Diff for /loncom/homework/grades.pm between versions 1.41 and 1.44

version 1.41, 2002/07/25 21:25:38 version 1.44, 2002/08/02 21:10:03
Line 30 Line 30
 # 7/26 H.K. Ng  # 7/26 H.K. Ng
 # 8/20 Gerd Kortemeyer  # 8/20 Gerd Kortemeyer
 # Year 2002  # Year 2002
 # June, July 2002 H.K. Ng  # June-August H.K. Ng
 #  #
   
 package Apache::grades;  package Apache::grades;
Line 42  use Apache::loncommon; Line 42  use Apache::loncommon;
 use Apache::lonhomework;  use Apache::lonhomework;
 use Apache::lonmsg qw(:user_normal_msg);  use Apache::lonmsg qw(:user_normal_msg);
 use Apache::Constants qw(:common);  use Apache::Constants qw(:common);
 #use Time::HiRes qw( gettimeofday tv_interval );  
   
 sub moreinfo {  # ----- These first few routines are general use routines.-----
     my ($request,$reason) = @_;  #
     $request->print("Unable to process request: $reason");  # --- Retrieve the parts that matches stores_\d+ from the metadata file.---
     if ( $Apache::grades::viewgrades eq 'F' ) {  sub getpartlist {
  $request->print('<form action="/adm/grades" method="post">'."\n");      my ($url) = @_;
  if ($ENV{'form.url'}) {      my @parts =();
     $request->print('<input type="hidden" name="url" value="'.$ENV{'form.url'}.'" />'."\n");      my (@metakeys) = split(/,/,&Apache::lonnet::metadata($url,'keys'));
  }      foreach my $key (@metakeys) {
  if ($ENV{'form.symb'}) {   if ( $key =~ m/stores_([0-9]+)_.*/) {
     $request->print('<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />'."\n");      push(@parts,$key);
  }   }
 #    $request->print('<input type="hidden" name="command" value="submission" />'."\n");  
  $request->print('<input type="hidden" name="command" value="'.$ENV{'form.command'}.'" />'."\n");  
  $request->print("Student:".'<input type="text" name="student" value="'.$ENV{'form.student'}.'" />'."<br />\n");  
  $request->print("Domain:".'<input type="text" name="domain" value="'.$ENV{'user.domain'}.'" />'."<br />\n");  
  $request->print('<input type="submit" name="submit" value="ReSubmit" />'."<br />\n");  
  $request->print('</form>');  
     }  
     return '';  
 }  
   
 sub verifyreceipt {  
     my $request=shift;  
     my $courseid=$ENV{'request.course.id'};  
 #    my $cdom=$ENV{"course.$courseid.domain"};  
 #    my $cnum=$ENV{"course.$courseid.num"};  
     my $receipt=unpack("%32C*",$Apache::lonnet::perlvar{'lonHostID'}).'-'.  
  $ENV{'form.receipt'};  
     $receipt=~s/[^\-\d]//g;  
     my $symb=$ENV{'form.symb'};  
     unless ($symb) {  
  $symb=&Apache::lonnet::symbread($ENV{'form.url'});  
     }  
     if ((&Apache::lonnet::allowed('mgr',$courseid)) && ($symb)) {  
         $request->print('<h1>Verifying Submission Receipt '.$receipt.'</h1>');  
         my $matches=0;  
         my ($classlist) = &getclasslist('all','0');  
         foreach my $student ( sort(@{ $$classlist{'all'} }) ) {  
             my ($uname,$udom)=split(/\:/,$student);  
             if ($receipt eq   
  &Apache::lonnet::ireceipt($uname,$udom,$courseid,$symb)) {  
  $request->print('Matching '.$student.'<br>');  
  $matches++;  
    }  
         }  
         $request->printf('<p>'.$matches." match%s</p>",$matches <= 1 ? '' : 'es');  
 # needs to print who is matched  
     }      }
     return '';      return @parts;
 }  }
   
 sub student_gradeStatus {  # --- Get the symbolic name of a problem and the url
     my ($url,$udom,$uname,$partlist) = @_;  sub get_symb_and_url {
       my ($request) = @_;
       (my $url=$ENV{'form.url'}) =~ s-^http://($ENV{'SERVER_NAME'}|$ENV{'HTTP_HOST'})--;
     my $symb=($ENV{'form.symb'} ne '' ? $ENV{'form.symb'} : (&Apache::lonnet::symbread($url)));      my $symb=($ENV{'form.symb'} ne '' ? $ENV{'form.symb'} : (&Apache::lonnet::symbread($url)));
     my %record= &Apache::lonnet::restore($symb,$ENV{'request.course.id'},$udom,$uname);      if ($symb eq '') { $request->print("Unable to handle ambiguous references:$url:."); return ''; }
     my %partstatus = ();      return ($symb,$url);
     foreach (@$partlist) {  
  my ($status,$foo)=split(/_/,$record{"resource.$_.solved"},2);  
  $status = 'nothing' if ($status eq '');  
  $partstatus{$_} = $status;  
  $partstatus{"resource.$_.submitted_by"} = $record{"resource.$_.submitted_by"}   
  if ($record{"resource.$_.submitted_by"} ne '');  
     }  
     return %partstatus;  
 }  }
   
   # --- Retrieve the fullname for a user. Return lastname, first middle ---
   # --- Generation is attached next to the lastname if it exists. ---
 sub get_fullname {  sub get_fullname {
     my ($uname,$udom) = @_;      my ($uname,$udom) = @_;
     my %name=&Apache::lonnet::get('environment', ['lastname','generation',      my %name=&Apache::lonnet::get('environment', ['lastname','generation',
Line 124  sub get_fullname { Line 83  sub get_fullname {
     return $fullname;      return $fullname;
 }  }
   
   #--- Get the partlist and the response type for a given problem. ---
   #--- Indicate if a response type is coded handgraded or not. ---
 sub response_type {  sub response_type {
     my ($url) = shift;      my ($url) = shift;
     my $allkeys = &Apache::lonnet::metadata($url,'keys');      my $allkeys = &Apache::lonnet::metadata($url,'keys');
 #    print "allkeys=>$allkeys<br>";  
     my %seen = ();      my %seen = ();
     my (@partlist,%handgrade);      my (@partlist,%handgrade);
     foreach (split(/,/,&Apache::lonnet::metadata($url,'packages'))) {      foreach (split(/,/,&Apache::lonnet::metadata($url,'packages'))) {
  if (/^\w+response_\d{1,2}.*/) {   if (/^\w+response_\d+.*/) {
     my ($responsetype,$part) = split(/_/,$_,2);      my ($responsetype,$part) = split(/_/,$_,2);
     my ($partid,$respid) = split(/_/,$part);      my ($partid,$respid) = split(/_/,$part);
     $handgrade{$part} = $responsetype.':'.($allkeys =~ /parameter_$part\_handgrade/ ? 'yes' : 'no');      $handgrade{$part} = $responsetype.':'.($allkeys =~ /parameter_$part\_handgrade/ ? 'yes' : 'no');
Line 143  sub response_type { Line 103  sub response_type {
     return \@partlist,\%handgrade;      return \@partlist,\%handgrade;
 }  }
   
   #--- Prints a message on screen if a user did something wrong
   #--- Operator error ---
   sub userError {
       my ($request, $reason, $step) = @_;
       $request->print('<h3><font color="red">LON-CAPA User Error</font></h3><br />'."\n");
       $request->print('<b>Reason: </b>'.$reason.'<br /><br />'."\n");
       $request->print('<b>Step: </b>'.($step ne '' ? $step : 'Use your browser back button to correct')
       .'<br /><br />'."\n");
       return '';
   }
   
   #--- Dumps the class list with usernames,list of sections,
   #--- section, ids and fullnames for each user.
   sub getclasslist {
       my ($getsec,$hideexpired) = @_;
       my $now = time;
       my %classlist=&Apache::lonnet::dump('classlist',
    $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
    $ENV{'course.'.$ENV{'request.course.id'}.'.num'});
       # codes to check for fields in the classlist
       # should contain end:start:id:section:fullname
       for (keys %classlist) {
    my (@fields) = split(/:/,$classlist{$_});
    %classlist   = &reformat_classlist(\%classlist) if (scalar(@fields) <= 2);
    last;
       }
   
       my (@holdsec,@sections,%allids,%stusec,%fullname);
       foreach (keys(%classlist)) {
    my ($end,$start,$id,$section,$fullname)=split(/:/,$classlist{$_});
    # still a student?
    if (($hideexpired) && ($end) && ($end < $now)) {
       next;
    }
    $section = ($section ne '' ? $section : 'no');
    push @holdsec,$section;
    if ($getsec eq 'all' || $getsec eq $section) {
       push (@{ $classlist{$getsec} }, $_);
       $allids{$_}  =$id;
       $stusec{$_}  =$section;
       $fullname{$_}=$fullname;
    }
       }
       my %seen = ();
       foreach my $item (@holdsec) {
    push (@sections, $item) unless $seen{$item}++;
       }
       return (\%classlist,\@sections,\%allids,\%stusec,\%fullname);
   }
   
   # add id, section and fullname to the classlist.db
   # done to maintain backward compatibility with older versions
   sub reformat_classlist {
       my ($classlist) = shift;
       foreach (sort keys(%$classlist)) {
    my ($unam,$udom) = split(/:/);
    my $section      = &Apache::lonnet::usection($udom,$unam,$ENV{'request.course.id'});
    my $fullname     = &get_fullname ($unam,$udom);
    my %userid       = &Apache::lonnet::idrget($udom,($unam));
    $$classlist{$_}  = $$classlist{$_}.':'.$userid{$unam}.':'.$section.':'.$fullname;
       }
       my $putresult = &Apache::lonnet::put
    ('classlist',\%$classlist,
    $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
    $ENV{'course.'.$ENV{'request.course.id'}.'.num'});
   
       return %$classlist;
   }
   
   #find user domain
   sub finduser {
       my ($name) = @_;
       my $domain = '';
       if ( $Apache::grades::viewgrades eq 'F' ) {
    my %classlist=&Apache::lonnet::dump('classlist',
       $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
       $ENV{'course.'.$ENV{'request.course.id'}.'.num'});
    my (@fields) = grep /^$name:/, keys %classlist;
    ($name, $domain) = split(/:/,$fields[0]);
    return ($name,$domain);
       } else {
    return ($ENV{'user.name'},$ENV{'user.domain'});
       }
   }
   
   #--- Prompts a user to enter a username.
   sub moreinfo {
       my ($request,$reason) = @_;
       $request->print("Unable to process request: $reason");
       if ( $Apache::grades::viewgrades eq 'F' ) {
    $request->print('<form action="/adm/grades" method="post">'."\n");
    if ($ENV{'form.url'}) {
       $request->print('<input type="hidden" name="url" value="'.$ENV{'form.url'}.'" />'."\n");
    }
    if ($ENV{'form.symb'}) {
       $request->print('<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />'."\n");
    }
    $request->print('<input type="hidden" name="command" value="'.$ENV{'form.command'}.'" />'."\n");
    $request->print("Student:".'<input type="text" name="student" value="'.$ENV{'form.student'}.'" />'."<br />\n");
    $request->print("Domain:".'<input type="text" name="domain" value="'.$ENV{'user.domain'}.'" />'."<br />\n");
    $request->print('<input type="submit" name="submit" value="ReSubmit" />'."<br />\n");
    $request->print('</form>');
       }
       return '';
   }
   
   #--- Retrieve the grade status of a student for all the parts
   sub student_gradeStatus {
       my ($url,$symb,$udom,$uname,$partlist) = @_;
       my %record     = &Apache::lonnet::restore($symb,$ENV{'request.course.id'},$udom,$uname);
       my %partstatus = ();
       foreach (@$partlist) {
    my ($status,$foo)    = split(/_/,$record{"resource.$_.solved"},2);
    $status              = 'nothing' if ($status eq '');
    $partstatus{$_}      = $status;
    my $subkey           = "resource.$_.submitted_by";
    $partstatus{$subkey} = $record{$subkey} if ($record{$subkey} ne '');
       }
       return %partstatus;
   }
   
   
   #------------------ End of general use routines --------------------
   #-------------------------------------------------------------------
   
   #------------------------------------ Receipt Verification Routines
   #--- Check whether a receipt number is valid.---
   sub verifyreceipt {
       my $request  = shift;
   
       my $courseid = $ENV{'request.course.id'};
       my $receipt  = unpack("%32C*",$Apache::lonnet::perlvar{'lonHostID'}).'-'.
    $ENV{'form.receipt'};
       $receipt     =~ s/[^\-\d]//g;
       my $url      = $ENV{'form.url'};
       my $symb     = $ENV{'form.symb'};
       unless ($symb) {
    $symb    = &Apache::lonnet::symbread($url);
       }
   
       my $jscript='<script type="text/javascript" language="javascript">'."\n".
    '    function viewOneStudent(user,domain) {'."\n".
    ' document.onestudent.student.value = user;'."\n".
    ' document.onestudent.userdom.value = domain;'."\n".
    ' document.onestudent.submit();'."\n".
    '    }'."\n".
    '</script>'."\n";
       $jscript.= '<form action="/adm/grades" method="post" name="onestudent">'."\n".
    '<input type="hidden" name="symb"    value="'.$symb.'" />'."\n".
    '<input type="hidden" name="url"     value="'.$url.'" />'."\n".
    '<input type="hidden" name="command" value="submission" />'."\n".
    '<input type="hidden" name="student" value="" />'."\n".
    '<input type="hidden" name="userdom" value="" />'."\n".
    '</form>'."\n";
   
       my $title.='<h2><font color="#339933">Verifying Submission Receipt '.
    $receipt.'</h2></font>'."\n".
    '<font size=+1><b>Resource: </b>'.$ENV{'form.url'}.'</font><br><br>'."\n";
   
       my ($string,$contents,$matches) = ('','',0);
       my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist('all','0');
       
       foreach (sort {$$fullname{$a} cmp $$fullname{$b} } keys %$fullname) {
    my ($uname,$udom)=split(/\:/);
    if ($receipt eq 
       &Apache::lonnet::ireceipt($uname,$udom,$courseid,$symb)) {
       $contents.='<tr bgcolor="#ffffe6"><td>&nbsp;'."\n".
    '<a href="javascript:viewOneStudent(\''.$uname.'\',\''.$udom.
    '\')"; TARGET=_self>'.$$fullname{$_}.'</a>&nbsp;</td>'."\n".
    '<td>&nbsp;'.$uname.'&nbsp;</td>'.
    '<td>&nbsp;'.$udom.'&nbsp;</td></tr>'."\n";
       
       $matches++;
    }
       }
       if ($matches == 0) {
    $string = $title.'No match found for the above receipt.';
       } else {
    $string = $jscript.$title.
       'The above receipt matches the following student'.
       ($matches <= 1 ? '.' : 's.')."\n".
       '<table border="0"><tr><td bgcolor="#777777">'."\n".
       '<table border="0"><tr bgcolor="#e6ffff">'."\n".
       '<td><b>&nbsp;Fullname&nbsp;</b></td>'."\n".
       '<td><b>&nbsp;Username&nbsp;</b></td>'."\n".
       '<td><b>&nbsp;Domain&nbsp;</b></td></tr>'."\n".
       $contents.
       '</table></td></tr></table>'."\n";
       }
       return $string.&show_grading_menu_form ($symb,$url);
   }
   
   #--- This is called by a number of programs.
   #--- Called from the Grading Menu - View/Grade an individual student
   #--- Also called directly when one clicks on the subm button 
   #    on the problem page.
 sub listStudents {  sub listStudents {
     my ($request) = shift;      my ($request) = shift;
     my $cdom      =$ENV{"course.$ENV{'request.course.id'}.domain"};      my $cdom      = $ENV{"course.$ENV{'request.course.id'}.domain"};
     my $cnum      =$ENV{"course.$ENV{'request.course.id'}.num"};      my $cnum      = $ENV{"course.$ENV{'request.course.id'}.num"};
     my $getsec    =$ENV{'form.section'} eq '' ? 'all' : $ENV{'form.section'};      my $getsec    = $ENV{'form.section'} eq '' ? 'all' : $ENV{'form.section'};
     my $submitonly=$ENV{'form.submitonly'} eq '' ? 'all' : $ENV{'form.submitonly'};      my $submitonly= $ENV{'form.submitonly'} eq '' ? 'all' : $ENV{'form.submitonly'};
   
     my $result='<h2><font color="#339933">&nbsp;View Submissions for a Student or a Group of Students</font></h2>';      my $result='<h2><font color="#339933">&nbsp;'.
    'View Submissions for a Student or a Group of Students</font></h2>';
     $result.='<table border="0">';      $result.='<table border="0">';
     $result.='<tr><td colspan=3><font size=+1><b>Resource: </b>'.$ENV{'form.url'}.'</font></td></tr>';      $result.='<tr><td colspan=3><font size=+1>'.
    '<b>Resource: </b>'.$ENV{'form.url'}.'</font></td></tr>';
     my ($partlist,$handgrade) = &response_type($ENV{'form.url'});      my ($partlist,$handgrade) = &response_type($ENV{'form.url'});
     for (sort keys(%$handgrade)) {      for (sort keys(%$handgrade)) {
  my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});   my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});
  $ENV{'form.handgrade'} = 'yes' if ($handgrade eq 'yes');   $ENV{'form.handgrade'} = 'yes' if ($handgrade eq 'yes');
  $result.='<tr><td><b>Part id: </b>'.$_.'</td>'.   $result.='<tr><td><b>Part </b>'.(split(/_/))[0].'</td>'.
     '<td><b>Type: </b>'.$responsetype.'</td>'.      '<td><b>Type: </b>'.$responsetype.'</td>'.
     '<td><b>Handgrade: </b>'.$handgrade.'</font></td></tr>';      '<td><b>Handgrade: </b>'.$handgrade.'</font></td></tr>';
     }      }
Line 188  ENDTABLEST Line 345  ENDTABLEST
  $request->print('<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />'."\n");   $request->print('<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />'."\n");
     }      }
     $request->print('<input type="hidden" name="command" value="processGroup" />'."\n");      $request->print('<input type="hidden" name="command" value="processGroup" />'."\n");
       
     my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($getsec,'0');      my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($getsec,'0');
           
     $result='<table border="0"><tr><td bgcolor="#777777">'.      $result='<table border="0"><tr><td bgcolor="#777777">'.
  '<table border="0"><tr bgcolor="#e6ffff">'.   '<table border="0"><tr bgcolor="#e6ffff">'.
  '<td><b>&nbsp;Select&nbsp;</b></td><td><b>&nbsp;Username&nbsp;</b></td>'.   '<td><b>&nbsp;Select&nbsp;</b></td><td><b>&nbsp;Fullname&nbsp;</b></td>'.
  '<td><b>&nbsp;Fullname&nbsp;</b></td><td><b>&nbsp;Domain&nbsp;</b></td>';   '<td><b>&nbsp;Username&nbsp;</b></td><td><b>&nbsp;Domain&nbsp;</b></td>';
     foreach (sort(@$partlist)) {      foreach (sort(@$partlist)) {
  $result.='<td><b>&nbsp;Part ID '.$_.' Status&nbsp;</b></td>';   $result.='<td><b>&nbsp;Part '.(split(/_/))[0].' Status&nbsp;</b></td>';
     }      }
     $request->print($result.'</tr>'."\n");      $request->print($result.'</tr>'."\n");
   
     foreach my $student (sort(@{ $$classlist{$getsec} }) ) {      foreach my $student (sort {$$fullname{$a} cmp $$fullname{$b} } keys %$fullname) {
  my ($uname,$udom) = split(/:/,$student);   my ($uname,$udom) = split(/:/,$student);
  my (%status) = &student_gradeStatus($ENV{'form.url'},$udom,$uname,$partlist);   my (%status) = &student_gradeStatus($ENV{'form.url'},
       $ENV{'form.symb'},$udom,$uname,$partlist);
  my $statusflg = '';   my $statusflg = '';
  foreach (keys(%status)) {   foreach (keys(%status)) {
     $statusflg = 1 if ($status{$_} ne 'nothing');      $statusflg = 1 if ($status{$_} ne 'nothing');
     my ($foo,$partid,$foo) = split(/\./,$_);      my ($foo,$partid,$foo1) = split(/\./,$_);
     if ($status{'resource.'.$partid.'.submitted_by'} ne '') {      if ($status{'resource.'.$partid.'.submitted_by'} ne '') {
  $statusflg = '';   $statusflg = '';
  $request->print('<input type="hidden" name="'.   $request->print('<input type="hidden" name="'.
Line 220  ENDTABLEST Line 378  ENDTABLEST
     $result='<tr bgcolor="#ffffe6">'.      $result='<tr bgcolor="#ffffe6">'.
  '<td align="center"><input type=checkbox name="stuinfo" value="'.   '<td align="center"><input type=checkbox name="stuinfo" value="'.
  $student.':'.$$fullname{$student}.'"></td>'."\n".   $student.':'.$$fullname{$student}.'"></td>'."\n".
  '<td>&nbsp;'.$uname.'&nbsp;</td>'."\n".  
  '<td>&nbsp;'.$$fullname{$student}.'&nbsp;</td>'."\n".   '<td>&nbsp;'.$$fullname{$student}.'&nbsp;</td>'."\n".
    '<td>&nbsp;'.$uname.'&nbsp;</td>'."\n".
  '<td align="middle">&nbsp;'.$udom.'&nbsp;</td>'."\n";   '<td align="middle">&nbsp;'.$udom.'&nbsp;</td>'."\n";
           
     foreach (sort keys(%status)) {      foreach (sort keys(%status)) {
Line 233  ENDTABLEST Line 391  ENDTABLEST
     }      }
     $request->print('</table></td></tr></table>');      $request->print('</table></td></tr></table>');
     $request->print('<input type="submit" name="submit" value="View/Grade" /><form />');      $request->print('<input type="submit" name="submit" value="View/Grade" /><form />');
       return '';
 }  }
   
   #---- Called from the listStudents routine
   #     Displays the submissions for one student or a group of students
 sub processGroup {  sub processGroup {
     my ($request)  = shift;      my ($request)  = shift;
     my $ctr        = 0;      my $ctr        = 0;
Line 247  sub processGroup { Line 408  sub processGroup {
     }      }
     foreach (@stuchecked) {      foreach (@stuchecked) {
  my ($uname,$udom,$fullname) = split(/:/);   my ($uname,$udom,$fullname) = split(/:/);
  $ENV{'form.student'} = $uname;   $ENV{'form.student'}        = $uname;
  $ENV{'form.fullname'} = $fullname;   $ENV{'form.userdom'}        = $udom;
    $ENV{'form.fullname'}       = $fullname;
  &submission($request,$ctr,$total);   &submission($request,$ctr,$total);
  $ctr++;   $ctr++;
     }      }
     return '';      return '';
 }  }
   
 sub userError {  #------------------------------------------------------------------------------------
     my ($request, $reason, $step) = @_;  #
     $request->print('<h3><font color="red">LON-CAPA User Error</font></h3><br />'."\n");  #-------------------------- Next few routines handles grading by student, essentially
     $request->print('<b>Reason: </b>'.$reason.'<br /><br />'."\n");  #                           handles essay response type problem/part
     $request->print('<b>Step: </b>'.($step ne '' ? $step : 'Use your browser back button to correct')  #
     .'<br /><br />'."\n");  #--- Javascript to handle the submission page functionality ---
     return '';  sub sub_page_js {
 }      my $request = shift;
       $request->print(<<SUBJAVASCRIPT);
 #FIXME - needs to handle multiple matches  <script type="text/javascript" language="javascript">
 sub finduser {    function updateRadio(radioButton,formtextbox,formsel,scores,weight) {
     my ($name) = @_;       var pts = formtextbox.value;
     my $domain = '';       var resetbox =false;
     if ( $Apache::grades::viewgrades eq 'F' ) {       if (isNaN(pts) || pts < 0) {
  my ($classlist) = &getclasslist('all','0');          alert("A number equal or greater than 0 is expected. Entered value = "+pts);
  foreach ( sort(@{ $$classlist{'all'} }) ) {          for (var i=0; i<radioButton.length; i++) {
     my ($posname,$posdomain) = split(/:/);             if (radioButton[i].checked) {
     if ($posname =~ $name) { $name=$posname; $domain=$posdomain; last; }        formtextbox.value = i;
         resetbox = true;
      }
  }   }
  return ($name,$domain);   if (!resetbox) {
     } else {     formtextbox.value = "";
  return ($ENV{'user.name'},$ENV{'user.domain'});   }
    return;
        }
   
        if (pts > weight) {
           var resp = confirm("You entered a value ("+pts+
             ") greater than the weight for the part. Accept?");
           if (resp == false) {
              formtextbox.value = "";
              return;
          }
     }      }
 }  
   
 sub getclasslist {      for (var i=0; i<radioButton.length; i++) {
     my ($getsec,$hideexpired) = @_;   radioButton[i].checked=false;
     my $now = time;   if (pts == i) {
     my %classlist=&Apache::lonnet::dump('classlist',     radioButton[i].checked=true;
  $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},  
  $ENV{'course.'.$ENV{'request.course.id'}.'.num'});  
     my (@holdsec,@sections,%allids,%stusec,%fullname);  
     foreach (keys(%classlist)) {  
  my ($end,$start,$id,$section,$fullname)=split(/:/,$classlist{$_});  
        # still a student?  
  if (($hideexpired) && ($end) && ($end < $now)) {  
     next;  
  }   }
  $section = ($section ne '' ? $section : 'no');      }
  push @holdsec,$section;      updateSelect(formsel);
  if ($getsec eq 'all' || $getsec eq $section) {      scores.value = "0";
     push (@{ $classlist{$getsec} }, $_);    }
     $allids{$_}=$id;  
     $stusec{$_}=$section;    function writeBox(formrad,formsel,pts,scores) {
     $fullname{$_}=$fullname;      formrad.value = pts;
       scores.value = "0";
       updateSelect(formsel,pts);
       return;
     }
   
     function clearRadBox(radioButton,formbox,formsel,scores) {
       for (var i=0; i<formsel.length; i++) {
    if (formsel[i].selected) {
       var selectx=i;
  }   }
     }      }
     my %seen = ();      if (selectx == scores.value) { return };
     foreach my $item (@holdsec) {      formbox.value = "";
  push (@sections, $item) unless $seen{$item}++;      for (var i=0; i<radioButton.length; i++) {
    radioButton[i].checked=false;
     }      }
     return (\%classlist,\@sections,\%allids,\%stusec,\%fullname);      scores.value = selectx;
 }    }
   
 sub getpartlist {    function updateSelect(formsel) {
     my ($url) = @_;      formsel[0].selected = true;
     my @parts =();      return;
     my (@metakeys) = split(/,/,&Apache::lonnet::metadata($url,'keys'));    }
     foreach my $key (@metakeys) {  
  if ( $key =~ m/stores_([0-9]+)_.*/) {  //===================== Show list of keywords ====================
     push(@parts,$key);    function keywords(keyform) {
  }      var keywds = keyform.value;
       var nret = prompt("Keywords list, separated by a space. Add/delete to list if desired.",keywds);
       if (nret==null) return;
       keyform.value = nret;
   
       document.SCORE.refresh.value = "on";
       if (document.SCORE.keywords.value != "") {
    document.SCORE.submit();
     }      }
     return @parts;      return;
 }    }
   
 #FIXME need to look at the metadata <stores> spec on what type of data to accept and provide an  //===================== Script to view submitted by ==================
 #interface based on that, also do that to above function.    function viewSubmitter(submitter) {
 sub setstudentgrade {      document.SCORE.refresh.value = "on";
     my ($url,$symb,$courseid,$student,@parts) = @_;      document.SCORE.NCT.value = "1";
     my $result ='';      document.SCORE.unamedom0.value = submitter;
     my ($stuname,$domain) = split(/:/,$student);      document.SCORE.submit();
     my %record=&Apache::lonnet::restore($symb,$courseid,$domain,$stuname);      return;
     my %newrecord;    }
   
     foreach my $part (@parts) {  //===================== Script to add keyword(s) ==================
  my ($temp,$part,$type)=split(/_/,$part);    function getSel() {
  my $oldscore=$record{"resource.$part.$type"};      if (document.getSelection) txt = document.getSelection();
  my $newscore=$ENV{"form.GRADE.$student.$part.$type"};      else if (document.selection) txt = document.selection.createRange().text;
  if ($type eq 'solved') {      else return;
     my $update=0;      var cleantxt = txt.replace(new RegExp('([\\f\\n\\r\\t\\v ])+', 'g')," ");
     if ($newscore eq 'nothing' ) {      if (cleantxt=="") {
  if ($oldscore ne '') {   alert("Select a word or group of words from document and then click this link.");
     $update=1;   return;
     $newscore = '';  
  }  
     } elsif ($oldscore !~ m/^$newscore/) {  
  $update=1;  
  $result.="Updating $stuname to $newscore<br />\n";  
  if ($newscore eq 'correct')   { $newscore = 'correct_by_override'; }  
  if ($newscore eq 'incorrect') { $newscore = 'incorrect_by_override'; }  
  if ($newscore eq 'excused')   { $newscore = 'excused'; }  
  if ($newscore eq 'ungraded')  { $newscore = 'ungraded_attempted'; }  
     } else {  
  #$result.="$stuname:$part:$type:unchanged  $oldscore to $newscore:<br />\n";  
     }  
     if ($update) { $newrecord{"resource.$part.$type"}=$newscore; }  
  } else {  
     if ($oldscore ne $newscore) {  
  $newrecord{"resource.$part.$type"}=$newscore;  
  $result.="Updating $student"."'s status for $part.$type to $newscore<br />\n";  
     } else {  
  #$result.="$stuname:$part:$type:unchanged  $oldscore to $newscore:<br />\n";  
     }  
  }  
     }      }
     if ( scalar(keys(%newrecord)) > 0 ) {      var nret = prompt("Add selection to keyword list? Edit if desired.",cleantxt);
  $newrecord{'resource.regrader'}="$ENV{'user.name'}:$ENV{'user.domain'}";      if (nret==null) return;
 #       &Apache::lonnet::cstore(\%newrecord,$symb,$courseid,$domain,$stuname);      var curlist = document.SCORE.keywords.value;
       document.SCORE.keywords.value = curlist+" "+nret;
       document.SCORE.refresh.value = "on";
       if (document.SCORE.keywords.value != "") {
    document.SCORE.submit();
       }
       return;
     }
   
  $result.="Stored away ".scalar(keys(%newrecord))." elements.<br />\n";  //====================== Script for composing message ==============
     function msgCenter(msgform,usrctr,fullname) {
       var Nmsg  = msgform.savemsgN.value;
       savedMsgHeader(Nmsg,usrctr,fullname);
       var subject = msgform.msgsub.value;
       var rtrchk  = eval("document.SCORE.includemsg"+usrctr);
       var msgchk = rtrchk.value;
       re = /msgsub/;
       var shwsel = "";
       if (re.test(msgchk)) { shwsel = "checked" }
       displaySubject(subject,shwsel);
       for (var i=1; i<=Nmsg; i++) {
    var testpt = "savemsg"+i+",";
    re = /testpt/;
    shwsel = "";
    if (re.test(msgchk)) { shwsel = "checked" }
    var message = eval("document.SCORE.savemsg"+i+".value");
    displaySavedMsg(i,message,shwsel);
     }      }
     return $result;      newmsg = eval("document.SCORE.newmsg"+usrctr+".value");
 }      shwsel = "";
       re = /newmsg/;
       if (re.test(msgchk)) { shwsel = "checked" }
       newMsg(newmsg,shwsel);
       msgTail(); 
       return;
     }
   
 sub print_hash {    function savedMsgHeader(Nmsg,usrctr,fullname) {
     my ($request, $hash) = @_;      var height = 30*Nmsg+250;
     $request->print('<table border=1><tr><td>Key</td><td>Value</td></tr>');      var scrollbar = "no";
     for (sort keys (%$hash)) {      if (height > 600) {
  $request->print('<tr><td>'.$_.'</td><td>'.$$hash{$_}.'&nbsp;</td></tr>');   height = 600;
    scrollbar = "yes";
     }      }
     $request->print('</table>');  /*    if (window.pWin)
     return '';   window.pWin.close(); */
       pWin = window.open('', 'MessageCenter', 'toolbar=no,location=no,scrollbars='+scrollbar+',screenx=70,screeny=75,width=600,height='+height);
       pWin.document.write("<html><head>");
       pWin.document.write("<title>Message Central</title>");
   
       pWin.document.write("<script language=javascript>");
       pWin.document.write("function checkInput() {");
       pWin.document.write("  opener.document.SCORE.msgsub.value = document.msgcenter.msgsub.value;");
       pWin.document.write("  var nmsg   = opener.document.SCORE.savemsgN.value;");
       pWin.document.write("  var usrctr = document.msgcenter.usrctr.value;");
       pWin.document.write("  var newval = eval(\\"opener.document.SCORE.newmsg\\"+usrctr);");
       pWin.document.write("  newval.value = document.msgcenter.newmsg.value;");
   
       pWin.document.write("  var msgchk = \\"\\";");
       pWin.document.write("  if (document.msgcenter.subchk.checked) {");
       pWin.document.write("     msgchk = \\"msgsub,\\";");
       pWin.document.write("  }");
       pWin.document.write(   "for (var i=1; i<=nmsg; i++) {");
       pWin.document.write("      var opnmsg = eval(\\"opener.document.SCORE.savemsg\\"+i);");
       pWin.document.write("      var frmmsg = eval(\\"document.msgcenter.msg\\"+i);");
       pWin.document.write("      opnmsg.value = frmmsg.value;");
       pWin.document.write("      var chkbox = eval(\\"document.msgcenter.msgn\\"+i);");
       pWin.document.write("      if (chkbox.checked) {");
       pWin.document.write("         msgchk += \\"savemsg\\"+i+\\",\\";");
       pWin.document.write("      }");
       pWin.document.write("  }");
       pWin.document.write("  if (document.msgcenter.newmsgchk.checked) {");
       pWin.document.write("     msgchk += \\"newmsg\\"+usrctr;");
       pWin.document.write("  }");
       pWin.document.write("  var includemsg = eval(\\"opener.document.SCORE.includemsg\\"+usrctr);");
       pWin.document.write("  includemsg.value = msgchk;");
   
       pWin.document.write("  self.close()");
   
       pWin.document.write("}");
   
       pWin.document.write("<");
       pWin.document.write("/script>");
   
       pWin.document.write("</head><body bgcolor=white>");
   
       pWin.document.write("<form action=\\"inactive\\" name=\\"msgcenter\\">");
       pWin.document.write("<input value=\\""+usrctr+"\\" name=\\"usrctr\\" type=\\"hidden\\">");
       pWin.document.write("<font color=\\"green\\" size=+1>&nbsp;Compose Message for \"+fullname+\"</font><br><br>");
   
       pWin.document.write("<table border=0 width=100%><tr><td bgcolor=\\"#777777\\">");
       pWin.document.write("<table border=0 width=100%><tr bgcolor=\\"#ddffff\\">");
       pWin.document.write("<td><b>Type</b></td><td><b>Include</b></td><td><b>Message</td></tr>");
 }  }
 #      function displaySubject(msg,shwsel) {
 # --------------------------- show submissions of a student, option to grade --------      pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");
       pWin.document.write("<td>Subject</td>");
       pWin.document.write("<td align=\\"center\\"><input name=\\"subchk\\" type=\\"checkbox\\"" +shwsel+"></td>");
       pWin.document.write("<td><input name=\\"msgsub\\" type=\\"text\\" value=\\""+msg+" \\"size=\\"60\\" maxlength=\\"80\\"></td></tr>");
   }
   
   function displaySavedMsg(ctr,msg,shwsel) {
       pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");
       pWin.document.write("<td align=\\"center\\">"+ctr+"</td>");
       pWin.document.write("<td align=\\"center\\"><input name=\\"msgn"+ctr+"\\" type=\\"checkbox\\"" +shwsel+"></td>");
       pWin.document.write("<td><input name=\\"msg"+ctr+"\\" type=\\"text\\" value=\\""+msg+" \\" size=\\"60\\" maxlength=\\"80\\"></td></tr>");
   }
   
     function newMsg(newmsg,shwsel) {
       pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");
       pWin.document.write("<td align=\\"center\\">New</td>");
       pWin.document.write("<td align=\\"center\\"><input name=\\"newmsgchk\\" type=\\"checkbox\\"" +shwsel+"></td>");
       pWin.document.write("<td><input name=\\"newmsg\\" type=\\"text\\" onchange=\\"javascript:this.form.newmsgchk.checked=true\\" value=\\""+newmsg+" \\" size=\\"60\\" maxlength=\\"80\\"></td></tr>");
   }
   
     function msgTail() {
       pWin.document.write("</table>");
       pWin.document.write("</td></tr></table>&nbsp;");
       pWin.document.write("<input type=\\"button\\" value=\\"Save\\" onClick=\\"javascript:checkInput()\\">&nbsp;&nbsp;");
       pWin.document.write("<input type=\\"button\\" value=\\"Cancel\\" onClick=\\"self.close()\\"><br><br>");
       pWin.document.write("</form>");
       pWin.document.write("</body></html>");
   }
   
   //====================== Script for keyword highlight options ==============
     function kwhighlight() {
       var kwclr    = document.SCORE.kwclr.value;
       var kwsize   = document.SCORE.kwsize.value;
       var kwstyle  = document.SCORE.kwstyle.value;
       var redsel = "";
       var grnsel = "";
       var blusel = "";
       if (kwclr=="red")   {var redsel="checked"};
       if (kwclr=="green") {var grnsel="checked"};
       if (kwclr=="blue")  {var blusel="checked"};
       var sznsel = "";
       var sz1sel = "";
       var sz2sel = "";
       if (kwsize=="0")  {var sznsel="checked"};
       if (kwsize=="+1") {var sz1sel="checked"};
       if (kwsize=="+2") {var sz2sel="checked"};
       var synsel = "";
       var syisel = "";
       var sybsel = "";
       if (kwstyle=="")    {var synsel="checked"};
       if (kwstyle=="<i>") {var syisel="checked"};
       if (kwstyle=="<b>") {var sybsel="checked"};
       highlightCentral();
       highlightbody('red','red',redsel,'0','normal',sznsel,'','normal',synsel);
       highlightbody('green','green',grnsel,'+1','+1',sz1sel,'<i>','italic',syisel);
       highlightbody('blue','blue',blusel,'+2','+2',sz2sel,'<b>','bold',sybsel);
       highlightend();
       return;
     }
   
   
     function highlightCentral() {
       hwdWin = window.open('', 'KeywordHighlightCentral', 'toolbar=no,location=no,scrollbars=no,width=400,height=300,screenx=100,screeny=75');
       hwdWin.document.write("<html><head>");
       hwdWin.document.write("<title>Highlight Central</title>");
   
       hwdWin.document.write("<script language=javascript>");
       hwdWin.document.write("function updateChoice(flag) {");
       hwdWin.document.write("  opener.document.SCORE.kwclr.value = radioSelection(document.hlCenter.kwdclr);");
       hwdWin.document.write("  opener.document.SCORE.kwsize.value = radioSelection(document.hlCenter.kwdsize);");
       hwdWin.document.write("  opener.document.SCORE.kwstyle.value = radioSelection(document.hlCenter.kwdstyle);");
       hwdWin.document.write("  opener.document.SCORE.refresh.value = \\"on\\";");
       hwdWin.document.write("  if (opener.document.SCORE.keywords.value!=\\"\\"){");
       hwdWin.document.write("     opener.document.SCORE.submit();");
       hwdWin.document.write("  }");
       hwdWin.document.write("  self.close()");
       hwdWin.document.write("}");
   
       hwdWin.document.write("function radioSelection(radioButton) {");
       hwdWin.document.write("    var selection=null;");
       hwdWin.document.write("    for (var i=0; i<radioButton.length; i++) {");
       hwdWin.document.write("        if (radioButton[i].checked) {");
       hwdWin.document.write("            selection=radioButton[i].value;");
       hwdWin.document.write("            return selection;");
       hwdWin.document.write("        }");
       hwdWin.document.write("    }");
       hwdWin.document.write("}");
   
       hwdWin.document.write("<");
       hwdWin.document.write("/script>");
   
       hwdWin.document.write("</head><body bgcolor=white>");
   
       hwdWin.document.write("<form action=\\"inactive\\" name=\\"hlCenter\\">");
       hwdWin.document.write("<font color=\\"green\\" size=+1>&nbsp;Keyword Highlight Options</font><br><br>");
   
       hwdWin.document.write("<table border=0 width=100%><tr><td bgcolor=\\"#777777\\">");
       hwdWin.document.write("<table border=0 width=100%><tr bgcolor=\\"#ddffff\\">");
       hwdWin.document.write("<td><b>Text Color</b></td><td><b>Font Size</b></td><td><b>Font Style</td></tr>");
     }
   
     function highlightbody(clrval,clrtxt,clrsel,szval,sztxt,szsel,syval,sytxt,sysel) { 
       hwdWin.document.write("<tr bgcolor=\\"#ffffdd\\">");
       hwdWin.document.write("<td align=\\"left\\">");
       hwdWin.document.write("<input name=\\"kwdclr\\" type=\\"radio\\" value=\\""+clrval+"\\" "+clrsel+">&nbsp;"+clrtxt+"</td>");
       hwdWin.document.write("<td align=\\"left\\">");
       hwdWin.document.write("<input name=\\"kwdsize\\" type=\\"radio\\" value=\\""+szval+"\\" "+szsel+">&nbsp;"+sztxt+"</td>");
       hwdWin.document.write("<td align=\\"left\\">");
       hwdWin.document.write("<input name=\\"kwdstyle\\" type=\\"radio\\" value=\\""+syval+"\\" "+sysel+">&nbsp;"+sytxt+"</td>");
       hwdWin.document.write("</tr>");
     }
   
     function highlightend() { 
       hwdWin.document.write("</table>");
       hwdWin.document.write("</td></tr></table>&nbsp;");
   //    hwdWin.document.write("<input type=\\"button\\" value=\\"Save\\" onClick=\\"javascript:updateChoice(0)\\">&nbsp;&nbsp;");
       hwdWin.document.write("<input type=\\"button\\" value=\\"Save\\" onClick=\\"javascript:updateChoice(1)\\">&nbsp;&nbsp;");
       hwdWin.document.write("<input type=\\"button\\" value=\\"Cancel\\" onClick=\\"self.close()\\"><br><br>");
       hwdWin.document.write("</form>");
       hwdWin.document.write("</body></html>");
     }
   
   </script>
   SUBJAVASCRIPT
   }
   
   
   # --------------------------- show submissions of a student, option to grade 
 sub submission {  sub submission {
     my ($request,$counter,$total) = @_;      my ($request,$counter,$total) = @_;
   
     (my $url=$ENV{'form.url'})=~s-^http://($ENV{'SERVER_NAME'}|$ENV{'HTTP_HOST'})--;      (my $url=$ENV{'form.url'})=~s-^http://($ENV{'SERVER_NAME'}|$ENV{'HTTP_HOST'})--;
     if ($ENV{'form.student'} eq '') { &moreinfo($request,'Need student login id'); return ''; }  #    if ($ENV{'form.student'} eq '') { &moreinfo($request,'Need student login id'); return ''; }
     my ($uname,$udom) = &finduser($ENV{'form.student'});      my ($uname,$udom)     = ($ENV{'form.student'},$ENV{'form.userdom'});
     if ($uname eq '') { &moreinfo($request,'Unable to find student'); return ''; }      ($uname,$udom)        = &finduser($uname) if $udom eq '';
       $ENV{'form.fullname'} = &get_fullname ($uname,$udom) if $ENV{'form.fullname'} eq '';
   #    if ($uname eq '') { &moreinfo($request,'Unable to find student'); return ''; }
   
     my $symb=($ENV{'form.symb'} ne '' ? $ENV{'form.symb'} : (&Apache::lonnet::symbread($url)));      my $symb=($ENV{'form.symb'} ne '' ? $ENV{'form.symb'} : (&Apache::lonnet::symbread($url)));
     if ($symb eq '') { $request->print("Unable to handle ambiguous references:$url:."); return ''; }      if ($symb eq '') { $request->print("Unable to handle ambiguous references:$url:."); return ''; }
Line 401  sub submission { Line 769  sub submission {
  $request->print('<h2>&nbsp;<font color="#339933">Submission Record</font></h2>'.   $request->print('<h2>&nbsp;<font color="#339933">Submission Record</font></h2>'.
  '<font size=+1>&nbsp;<b>Resource: </b>'.$url.'</font>');   '<font size=+1>&nbsp;<b>Resource: </b>'.$url.'</font>');
   
  # option to display problem, only once else it cause problems with the form later    # option to display problem, only once else it cause problems 
  # since the problem has a form.          # with the form later since the problem has a form.
  if ($ENV{'form.vProb'} eq 'yes') {   if ($ENV{'form.vProb'} eq 'yes') {
     my $rendered=&Apache::loncommon::get_student_view($symb,$uname,$udom,      my $rendered=&Apache::loncommon::get_student_view($symb,$uname,$udom,
       $ENV{'request.course.id'});        $ENV{'request.course.id'});
Line 410  sub submission { Line 778  sub submission {
    $ENV{'request.course.id'});     $ENV{'request.course.id'});
     my $result.='<table border="0" width="100%"><tr><td bgcolor="#777777">';      my $result.='<table border="0" width="100%"><tr><td bgcolor="#777777">';
     $result.='<table border="0" width="100%"><tr><td bgcolor="#e6ffff">';      $result.='<table border="0" width="100%"><tr><td bgcolor="#e6ffff">';
     $result.='<b>Student\'s view of the problem</b></td></tr><tr><td bgcolor="#ffffff">'.$rendered.'<br />';      $result.='<b> View of the problem for '.$ENV{'form.fullname'}.
    '</b></td></tr><tr><td bgcolor="#ffffff">'.$rendered.'<br />';
     $result.='<b>Correct answer:</b><br />'.$companswer;      $result.='<b>Correct answer:</b><br />'.$companswer;
     $result.='</td></tr></table>';      $result.='</td></tr></table>';
     $result.='</td></tr></table><br />';      $result.='</td></tr></table><br />';
     $request->print($result);      $request->print($result);
  }   }
   
  # kwclr is the only variable that is guaranteed to be non blank if this subroutine has been called once.   # kwclr is the only variable that is guaranteed to be non blank 
           # if this subroutine has been called once.
  my %keyhash = ();   my %keyhash = ();
  if ($ENV{'form.kwclr'} eq '') {   if ($ENV{'form.kwclr'} eq '') {
     %keyhash = &Apache::lonnet::dump('nohist_handgrade',      %keyhash = &Apache::lonnet::dump('nohist_handgrade',
Line 434  sub submission { Line 804  sub submission {
     $ENV{'form.savemsgN'} = $keyhash{$symb.'_savemsgN'} ne '' ? $keyhash{$symb.'_savemsgN'} : '0';      $ENV{'form.savemsgN'} = $keyhash{$symb.'_savemsgN'} ne '' ? $keyhash{$symb.'_savemsgN'} : '0';
   
  }   }
   
  $request->print('<form action="/adm/grades" method="post" name="SCORE">'."\n".   $request->print('<form action="/adm/grades" method="post" name="SCORE">'."\n".
  '<input type="hidden" name="command"    value="handgrade" />'."\n".   '<input type="hidden" name="command"    value="handgrade" />'."\n".
  '<input type="hidden" name="refresh"    value="off" />'."\n".   '<input type="hidden" name="refresh"    value="off" />'."\n".
Line 474  sub submission { Line 845  sub submission {
 KEYWORDS  KEYWORDS
         }          }
     }      }
   
     my %record = &Apache::lonnet::restore($symb,$ENV{'request.course.id'},$udom,$uname);      my %record = &Apache::lonnet::restore($symb,$ENV{'request.course.id'},$udom,$uname);
     my ($partlist,$handgrade) = &response_type($url);      my ($partlist,$handgrade) = &response_type($url);
 #    &print_hash($request,\%record);  
   
     # Student info      # Display student info
     $request->print(($counter == 0 ? '' : '<br />'));      $request->print(($counter == 0 ? '' : '<br />'));
     my $fullname = ($ENV{'form.fullname'} ne '' ? $ENV{'form.fullname'} : &get_fullname($uname,$udom));  
     my $result='<table border="0" width=100%><tr><td bgcolor="#777777">'.      my $result='<table border="0" width=100%><tr><td bgcolor="#777777">'.
  '<table border="0" width=100%><tr bgcolor="#ffffff"><td>';   '<table border="0" width=100%><tr bgcolor="#edffff"><td>';
   
     $result.='<table border="0"><tr bgcolor="#ffffff"><td><b>Fullname: </b>'.$fullname.  #    $result.='<table border="0"><tr bgcolor="#ffffff"><td><b>Fullname: </b>'.$ENV{'form.fullname'}.
  '</td><td>&nbsp; &nbsp;<b>Username: </b>'.$uname.      $result.='<b>Fullname: </b>'.$ENV{'form.fullname'}.
  '</td><td>&nbsp; &nbsp;<b>Domain: </b>'.$udom.'</td></tr>';   '<font color="#999999">&nbsp; &nbsp;Username: '.$uname.'</font>'.
    '<font color="#999999">&nbsp; &nbsp;Domain: '.$udom.'</font><br />';
   
       # If this is handgraded, then check for collaborators
       my $col_flag = 0;
     if ($ENV{'form.handgrade'} eq 'yes') {      if ($ENV{'form.handgrade'} eq 'yes') {
  my @col_list;   my @col_list;
  ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist('all','0');   ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist('all','0');
  for (keys (%$handgrade)) {   for (keys (%$handgrade)) {
     my $ncol = &Apache::lonnet::EXT('resource.'.$_.'.maxcollaborators',$symb,$udom,$uname);      my $ncol = &Apache::lonnet::EXT('resource.'.$_.
       '.maxcollaborators',$symb,$udom,$uname);
     if ($ncol > 0) {      if ($ncol > 0) {
  s/\_/\./g;   s/\_/\./g;
  if ($record{'resource.'.$_.'.collaborators'} ne '') {   if ($record{'resource.'.$_.'.collaborators'} ne '') {
     my (@collaborators) = split(/,?\s+/,$record{'resource.'.$_.'.collaborators'});      my (@collaborators) = split(/,?\s+/,
    $record{'resource.'.$_.'.collaborators'});
     my (@badcollaborators);      my (@badcollaborators);
     if (scalar(@collaborators) != 0) {      if (scalar(@collaborators) != 0) {
  $result.='<tr bgcolor="#ffffff"><td colspan=3><b>Collaborators: </b>';   $result.='<b>Collaborators: </b>';
  foreach my $collaborator (@collaborators) {   foreach my $collaborator (@collaborators) {
     $collaborator = $collaborator =~ /\@|:/ ?       $collaborator = $collaborator =~ /\@|:/ ? 
  (split(/@|:/,$collaborator))[0] : $collaborator;   (split(/@|:/,$collaborator))[0] : $collaborator;
Line 507  KEYWORDS Line 883  KEYWORDS
  push @badcollaborators,$collaborator;   push @badcollaborators,$collaborator;
  next;   next;
     }      }
       $col_flag++;
     push @col_list, $collaborator;      push @col_list, $collaborator;
     $result.=$$fullname{$collaborator.':'.$udom}.' ('.$collaborator.') &nbsp; &nbsp;';      $result.=$$fullname{$collaborator.':'.$udom}.'&nbsp; &nbsp; &nbsp;';
  }   }
  $result.='</td></tr>'."\n";   $result.='<br />'."\n";
  $result.='<tr bgcolor="#ffbbbb"><td colspan=2>'.   $result.='<table border="0"><tr bgcolor="#ffbbbb"><td>'.
     'This student has submitted '.(scalar (@badcollaborators) > 1 ? '' : 'an').      'This student has submitted '.
       (scalar (@badcollaborators) > 1 ? '' : 'an').
     ' invalid collaborator'.(scalar (@badcollaborators) > 1 ? 's. ' : '. ').      ' invalid collaborator'.(scalar (@badcollaborators) > 1 ? 's. ' : '. ').
     (join ', ',@badcollaborators).'</td></tr>'       (join ', ',@badcollaborators).'</td></tr></table>' 
     if (scalar(@badcollaborators) > 0);      if (scalar(@badcollaborators) > 0);
   
  $result.='<tr bgcolor="#ffbbbb"><td colspan=2>'.   $result.='<table border="0"><tr bgcolor="#ffbbbb"><td>'.
     'This student has submitted too many collaborators. Maximum is '.      'This student has submitted too many collaborators. Maximum is '.
     $ncol.'.</td></tr>' if (scalar(@collaborators) > $ncol);      $ncol.'.</td></tr></table>' if (scalar(@collaborators) > $ncol);
  $result.='<input type="hidden" name="collaborator'.$counter.   $result.='<input type="hidden" name="collaborator'.$counter.
     '" value="'.(join ':',@col_list).'" />'."\n";      '" value="'.(join ':',@col_list).'" />'."\n";
     }      }
Line 527  KEYWORDS Line 905  KEYWORDS
     }      }
  }   }
     }      }
     $request->print($result.'</table>'."\n");      $request->print($result."\n");
   
     # print student answer      # print student answer/submission
       # Options are (1) Handgaded submission only
       #             (2) Last submission, includes submission that is not handgraded 
       #                  (for multi-response type part)
       #             (3) Last submission plus the parts info
       #             (4) The whole record for this student
     if ($ENV{'form.lastSub'} =~ /^(lastonly|hdgrade)$/) {      if ($ENV{'form.lastSub'} =~ /^(lastonly|hdgrade)$/) {
  if ($ENV{'form.'.$uname.':'.$udom.':submitted_by'}) {   if ($ENV{'form.'.$uname.':'.$udom.':submitted_by'}) {
     my $submitby='</td></tr><tr><td bgcolor="#e6ffff">'.      my $submitby=''.
  '<b>Collaborative submission by: </b>'.   '<b>Collaborative submission by: </b>'.
  '<a href="javascript:viewSubmitter(\''.$ENV{'form.'.$uname.':'.$udom.':submitted_by'}.   '<a href="javascript:viewSubmitter(\''.
    $ENV{'form.'.$uname.':'.$udom.':submitted_by'}.
  '\')"; TARGET=_self>'.   '\')"; TARGET=_self>'.
  $$fullname{$ENV{'form.'.$uname.':'.$udom.':submitted_by'}}.'</a>';   $$fullname{$ENV{'form.'.$uname.':'.$udom.':submitted_by'}}.'</a>';
     $submitby.='</td></tr><tr><td bgcolor="#ffffff">'."\n";  
     $request->print($submitby);      $request->print($submitby);
  } else {   } else {
     my ($string,$timestamp)=&get_last_submission ($symb,$uname,$udom,$ENV{'request.course.id'});      my ($string,$timestamp)=
     my $lastsubonly.='</td></tr><tr><td bgcolor="#e6ffff"><b>Last Submission Only</b>'.   &get_last_submission ($symb,$uname,$udom,$ENV{'request.course.id'});
  ($$timestamp eq '' ? '' : '&nbsp; &nbsp; <b>Date Submitted:</b> '.$$timestamp).'</td></tr>';      my $lastsubonly.=''.
    ($$timestamp eq '' ? '' : '<b>Date Submitted:</b> '.
    $$timestamp).'';
     if ($$timestamp eq '') {      if ($$timestamp eq '') {
  $lastsubonly.='<tr><td bgcolor="#ffffe6">'.$$string[0].'</td></tr>';   $lastsubonly.='<tr><td bgcolor="#ffffe6">'.$$string[0].'</td></tr>';
     } else {      } else {
Line 551  KEYWORDS Line 936  KEYWORDS
  my ($partid,$respid) = /^resource\.(\d+)\.(\d+)\.submission/;   my ($partid,$respid) = /^resource\.(\d+)\.(\d+)\.submission/;
  if ($part eq ($partid.'_'.$respid)) {   if ($part eq ($partid.'_'.$respid)) {
     my ($ressub,$subval) = split(/:/,$_,2);      my ($ressub,$subval) = split(/:/,$_,2);
     $lastsubonly.='<tr><td bgcolor="#ffffe6"><b>Part ID</b> '.      $lastsubonly.='<tr><td bgcolor="#ffffe6"><b>Part '.
  $partid.' <b>Response ID</b> '.$respid.   $partid.'</b> <font color="#999999">( ID '.$respid.
  ' <b>Submission</b> '.&keywords_highlight($subval).'</td></tr>'   ' )</font>&nbsp; &nbsp;<b>Answer: </b>'.
    &keywords_highlight($subval).'</td></tr>'
  if ($ENV{'form.lastSub'} eq 'lastonly' ||    if ($ENV{'form.lastSub'} eq 'lastonly' || 
     ($ENV{'form.lastSub'} eq 'hdgrade' && $$handgrade{$part} =~ /:yes$/));      ($ENV{'form.lastSub'} eq 'hdgrade' && 
        $$handgrade{$part} =~ /:yes$/));
  }   }
     }      }
  }   }
Line 565  KEYWORDS Line 952  KEYWORDS
  }   }
     } else {      } else {
  $request->print(&Apache::loncommon::get_previous_attempt($symb,$uname,$udom,   $request->print(&Apache::loncommon::get_previous_attempt($symb,$uname,$udom,
  $ENV{'request.course.id'},$last,   $ENV{'request.course.id'},
  '.submission','Apache::grades::keywords_highlight'));   $last,'.submission',
    'Apache::grades::keywords_highlight'));
     }      }
           
     # view submission with no grading option      # return if view submission with no grading option
     if ($ENV{'form.showgrading'} eq '') {      if ($ENV{'form.showgrading'} eq '') {
  $request->print('</td></tr></table></td></tr></table></form>');   $request->print('</td></tr></table></td></tr></table></form>');
  return;   return;
     }      }
   
       # Grading options
     $result='<input type="hidden" name="newmsg'.$counter.'" value="" />'."\n".      $result='<input type="hidden" name="newmsg'.$counter.'" value="" />'."\n".
  '<input type="hidden" name="includemsg'.$counter.'" value="" />'."\n".   '<input type="hidden" name="includemsg'.$counter.'" value="" />'."\n".
  '<input type="hidden" name="unamedom'.$counter.'" value="'.$uname.':'.$udom.'" />'."\n";   '<input type="hidden" name="unamedom'.$counter.'" value="'.$uname.':'.$udom.'" />'."\n";
     $result.='&nbsp;<a href="javascript:msgCenter(document.SCORE,'.$counter.      $result.='&nbsp;<a href="javascript:msgCenter(document.SCORE,'.$counter.
  ',\''.$fullname.'\')"; TARGET=_self>Compose Message</a><br />'."\n" if ($ENV{'form.handgrade'} eq 'yes');   ',\''.$ENV{'form.fullname'}.'\')"; TARGET=_self>'.
    'Compose Message to student'.($col_flag > 1 ? 's' : '').'</a>'.
    '<br />&nbsp;(Message will be sent when you click on Save & Next below.)'."\n" 
    if ($ENV{'form.handgrade'} eq 'yes');
     $request->print($result);      $request->print($result);
   
     my %seen = ();      my %seen = ();
Line 591  KEYWORDS Line 983  KEYWORDS
  next if ($$handgrade{$_} =~ /:no$/);   next if ($$handgrade{$_} =~ /:no$/);
  push @partlist,$partid;   push @partlist,$partid;
  my $wgt    = &Apache::lonnet::EXT('resource.'.$partid.'.weight',$symb,$udom,$uname);   my $wgt    = &Apache::lonnet::EXT('resource.'.$partid.'.weight',$symb,$udom,$uname);
  my $wgtmsg = ($wgt > 0 ? '(problem weight)' : '<font color="red">problem weight assigned by computer</font>');   my $wgtmsg = ($wgt > 0 ? '(problem weight)' : 
         '<font color="red">problem weight assigned by computer</font>');
  $wgt       = ($wgt > 0 ? $wgt : '1');   $wgt       = ($wgt > 0 ? $wgt : '1');
  my $score  = ($record{'resource.'.$partid.'.awarded'} eq '' ?   my $score  = ($record{'resource.'.$partid.'.awarded'} eq '' ?
       '' : $record{'resource.'.$partid.'.awarded'}*$wgt);        '' : $record{'resource.'.$partid.'.awarded'}*$wgt);
   
  # display grading options  
  $result='<input type="hidden" name="WGT'.$counter.'_'.$partid.'" value="'.$wgt.'" />';   $result='<input type="hidden" name="WGT'.$counter.'_'.$partid.'" value="'.$wgt.'" />';
  $result.='<table border="0"><tr><td><b>Part </b>'.$partid.' <b>Points</b></td><td>';   $result.='<table border="0"><tr><td><b>Part </b>'.$partid.' <b>Points: </b></td><td>';
   
  my $ctr = 0;   my $ctr = 0;
  $result.='<table border="0"><tr>';  # display radio buttons in a nice table 10 across   $result.='<table border="0"><tr>';  # display radio buttons in a nice table 10 across
  while ($ctr<=$wgt) {   while ($ctr<=$wgt) {
     $result.= '<td><input type="radio" name="RADVAL'.$counter.'_'.$partid.'" '.      $result.= '<td><input type="radio" name="RADVAL'.$counter.'_'.$partid.'" '.
  'onclick="javascript:writeBox(this.form.GRADE_BOX'.$counter.'_'.$partid.   'onclick="javascript:writeBox(this.form.GD_BOX'.$counter.'_'.$partid.
  ',this.form.GRADE_SEL'.$counter.'_'.$partid.','.$ctr.   ',this.form.GD_SEL'.$counter.'_'.$partid.','.$ctr.
  ',this.form.stores'.$counter.'_'.$partid.')" '.   ',this.form.stores'.$counter.'_'.$partid.')" '.
  ($score eq $ctr ? 'checked':'').' /> '.$ctr."</td>\n";   ($score eq $ctr ? 'checked':'').' /> '.$ctr."</td>\n";
     $result.=(($ctr+1)%10 == 0 ? '</tr><tr>' : '');      $result.=(($ctr+1)%10 == 0 ? '</tr><tr>' : '');
Line 614  KEYWORDS Line 1005  KEYWORDS
  $result.='</tr></table>';   $result.='</tr></table>';
   
  $result.='</td><td>&nbsp;<b>or</b>&nbsp;</td>';   $result.='</td><td>&nbsp;<b>or</b>&nbsp;</td>';
  $result.='<td><input type="text" name="GRADE_BOX'.$counter.'_'.$partid.'"'.   $result.='<td><input type="text" name="GD_BOX'.$counter.'_'.$partid.'"'.
     ($score ne ''? ' value = "'.$score.'"':'').' size="4" '.      ($score ne ''? ' value = "'.$score.'"':'').' size="4" '.
     'onChange="javascript:updateRadio(this.form.RADVAL'.$counter.'_'.$partid.      'onChange="javascript:updateRadio(this.form.RADVAL'.$counter.'_'.$partid.
     ',this.form.GRADE_BOX'.$counter.'_'.$partid.      ',this.form.GD_BOX'.$counter.'_'.$partid.
     ',this.form.GRADE_SEL'.$counter.'_'.$partid.      ',this.form.GD_SEL'.$counter.'_'.$partid.
     ',this.form.stores'.$counter.'_'.$partid.')" /></td>'."\n";      ',this.form.stores'.$counter.'_'.$partid.
       ','.$wgt.')" /></td>'."\n";
  $result.='<td>/'.$wgt.' '.$wgtmsg.' </td><td>';   $result.='<td>/'.$wgt.' '.$wgtmsg.' </td><td>';
   
  $result.='<select name="GRADE_SEL'.$counter.'_'.$partid.'" '.   $result.='<select name="GD_SEL'.$counter.'_'.$partid.'" '.
     'onChange="javascript:clearRadBox(this.form.RADVAL'.$counter.'_'.$partid.      'onChange="javascript:clearRadBox(this.form.RADVAL'.$counter.'_'.$partid.
     ',this.form.GRADE_BOX'.$counter.'_'.$partid.      ',this.form.GD_BOX'.$counter.'_'.$partid.
     ',this.form.GRADE_SEL'.$counter.'_'.$partid.      ',this.form.GD_SEL'.$counter.'_'.$partid.
     ',this.form.stores'.$counter.'_'.$partid.')" />'."\n".      ',this.form.stores'.$counter.'_'.$partid.')" />'."\n".
     '<option selected="on"> </option>'.      '<option selected="on"> </option>'.
     '<option>excused</option></select>'."&nbsp&nbsp\n";      '<option>excused</option></select>'."&nbsp&nbsp\n";
Line 655  KEYWORDS Line 1047  KEYWORDS
     return '';      return '';
 }  }
   
   #--- Retrieve the last submission for all the parts
 sub get_last_submission {  sub get_last_submission {
     my ($symb,$username,$domain,$course)=@_;      my ($symb,$username,$domain,$course)=@_;
     if ($symb) {      if ($symb) {
Line 669  sub get_last_submission { Line 1062  sub get_last_submission {
  }   }
     }      }
     foreach ((keys %lasthash)) {      foreach ((keys %lasthash)) {
  if ($_ =~ /\.submission$/) {push @string, (join(':',$_,$lasthash{$_}))}   if ($_ =~ /\.submission$/) {
       my ($partid,$foo) = split(/submission$/,$_);
       my $draft  = $lasthash{$partid.'awarddetail'} eq 'DRAFT' ?
    '<font color="red">Draft Copy</font> ' : '';
       push @string, (join(':',$_,$draft.$lasthash{$_}));
    }
  if ($_ =~ /timestamp/) {$timestamp = scalar(localtime($lasthash{$_}))};   if ($_ =~ /timestamp/) {$timestamp = scalar(localtime($lasthash{$_}))};
     }      }
  }   }
Line 678  sub get_last_submission { Line 1076  sub get_last_submission {
     }      }
 }  }
   
   #--- High light keywords, with style choosen by user.
 sub keywords_highlight {  sub keywords_highlight {
     my $string  = shift;      my $string    = shift;
     my $size    = $ENV{'form.kwsize'} eq '0' ? '' : 'size='.$ENV{'form.kwsize'};      my $size      = $ENV{'form.kwsize'} eq '0' ? '' : 'size='.$ENV{'form.kwsize'};
     my $styleon = $ENV{'form.kwstyle'} eq ''  ? '' : $ENV{'form.kwstyle'};      my $styleon   = $ENV{'form.kwstyle'} eq ''  ? '' : $ENV{'form.kwstyle'};
     (my $styleoff = $styleon) =~ s/\</\<\//;      (my $styleoff = $styleon) =~ s/\</\<\//;
     my @keylist = split(/[,\s+]/,$ENV{'form.keywords'});      my @keylist   = split(/[,\s+]/,$ENV{'form.keywords'});
     foreach (@keylist) {      foreach (@keylist) {
  $string =~ s/\b$_(\b|\.)/\<font color\=$ENV{'form.kwclr'} $size\>$styleon$_$styleoff\<\/font\>/gi;   $string =~ s/\b$_(\b|\.)/\<font color\=$ENV{'form.kwclr'} $size\>$styleon$_$styleoff\<\/font\>/gi;
     }      }
     return $string;      return $string;
 }  }
   
   #--- Called from submission routine
 sub processHandGrade {  sub processHandGrade {
     my ($request) = shift;      my ($request) = shift;
     my $url    = $ENV{'form.url'};      my $url    = $ENV{'form.url'};
Line 698  sub processHandGrade { Line 1098  sub processHandGrade {
     my $ngrade = $ENV{'form.NCT'};      my $ngrade = $ENV{'form.NCT'};
     my $ntstu  = $ENV{'form.NTSTU'};      my $ntstu  = $ENV{'form.NTSTU'};
   
       if ($button eq 'Save & Next') {
    my $ctr = 0;
    while ($ctr < $ngrade) {
       my ($uname,$udom) = split(/:/,$ENV{'form.unamedom'.$ctr});
       my ($errorflg) = &saveHandGrade($request,$url,$symb,$uname,$udom,$ctr);
       return '' if ($errorflg eq 'error');
   
       my $includemsg = $ENV{'form.includemsg'.$ctr};
       my ($subject,$message,$msgstatus) = ('','','');
       if ($includemsg =~ /savemsg|new$ctr/) {
    $subject = $ENV{'form.msgsub'} if ($includemsg =~ /^msgsub/);
    my (@msgnum) = split(/,/,$includemsg);
    foreach (@msgnum) {
       $message.=$ENV{'form.'.$_} if ($_ =~ /savemsg|newmsg/ && $_ ne '');
    }
    $message =~ s/\s+/ /g;
    $msgstatus = &Apache::lonmsg::user_normal_msg ($uname,$udom,
          $ENV{'form.msgsub'},$message);
       }
       if ($ENV{'form.collaborator'.$ctr}) {
    my (@collaborators) = split(/:/,$ENV{'form.collaborator'.$ctr});
    foreach (@collaborators) {
       &saveHandGrade($request,$url,$symb,$_,$udom,$ctr,
      $ENV{'form.unamedom'.$ctr});
       if ($message ne '') {
    $msgstatus = &Apache::lonmsg::user_normal_msg ($_,$udom,
          $ENV{'form.msgsub'},
          $message);
       }
    }
       }
       $ctr++;
    }
       }
   
       # Keywords sorted in alphabatical order
     my $loginuser = $ENV{'user.name'}.':'.$ENV{'user.domain'};      my $loginuser = $ENV{'user.name'}.':'.$ENV{'user.domain'};
     my %keyhash = ();      my %keyhash = ();
     $ENV{'form.keywords'}           =~ s/,\s{0,}|\s+/ /g;      $ENV{'form.keywords'}           =~ s/,\s{0,}|\s+/ /g;
     $ENV{'form.keywords'}           =~ s/^\s+|\s+$//;      $ENV{'form.keywords'}           =~ s/^\s+|\s+$//;
       my (@keywords) = sort(split(/\s+/,$ENV{'form.keywords'}));
       $ENV{'form.keywords'} = join(' ',@keywords);
     $keyhash{$symb.'_keywords'}     = $ENV{'form.keywords'};      $keyhash{$symb.'_keywords'}     = $ENV{'form.keywords'};
     $keyhash{$symb.'_subject'}      = $ENV{'form.msgsub'};      $keyhash{$symb.'_subject'}      = $ENV{'form.msgsub'};
     $keyhash{$loginuser.'_kwclr'}   = $ENV{'form.kwclr'};      $keyhash{$loginuser.'_kwclr'}   = $ENV{'form.kwclr'};
     $keyhash{$loginuser.'_kwsize'}  = $ENV{'form.kwsize'};      $keyhash{$loginuser.'_kwsize'}  = $ENV{'form.kwsize'};
     $keyhash{$loginuser.'_kwstyle'} = $ENV{'form.kwstyle'};      $keyhash{$loginuser.'_kwstyle'} = $ENV{'form.kwstyle'};
   
       # message center - Order of message gets changed. Blank line is eliminated.
       # New messages are saved in ENV for the next student.
       # All messages are saved in nohist_handgrade.db
     my ($ctr,$idx) = (1,1);      my ($ctr,$idx) = (1,1);
     while ($ctr <= $ENV{'form.savemsgN'}) {      while ($ctr <= $ENV{'form.savemsgN'}) {
  if ($ENV{'form.savemsg'.$ctr} ne '') {   if ($ENV{'form.savemsg'.$ctr} ne '') {
Line 732  sub processHandGrade { Line 1173  sub processHandGrade {
  $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},   $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},
  $ENV{'course.'.$ENV{'request.course.id'}.'.num'});   $ENV{'course.'.$ENV{'request.course.id'}.'.num'});
   
       # Called by Save & Refresh from Highlight Attribute Window
     if ($ENV{'form.refresh'} eq 'on') {      if ($ENV{'form.refresh'} eq 'on') {
  my $ctr = 0;   my $ctr = 0;
  $ENV{'form.NTSTU'}=$ngrade;   $ENV{'form.NTSTU'}=$ngrade;
  while ($ctr < $ngrade) {   while ($ctr < $ngrade) {
     ($ENV{'form.student'},my $udom) = split(/:/,$ENV{'form.unamedom'.$ctr});      ($ENV{'form.student'},$ENV{'form.userdom'}) = split(/:/,$ENV{'form.unamedom'.$ctr});
     &submission($request,$ctr,$ngrade-1);      &submission($request,$ctr,$ngrade-1);
     $ctr++;      $ctr++;
  }   }
  return '';   return '';
     }      }
   
     if ($button eq 'Save & Next') {      # Get the next/previous one or group of students
  my $ctr = 0;  
  while ($ctr < $ngrade) {  
     my ($uname,$udom) = split(/:/,$ENV{'form.unamedom'.$ctr});  
     my ($errorflg) = &saveHandGrade($request,$url,$symb,$uname,$udom,$ctr);  
     return '' if ($errorflg eq 'error');  
   
     my $includemsg = $ENV{'form.includemsg'.$ctr};  
     my ($subject,$message,$msgstatus) = ('','','');  
     if ($includemsg =~ /savemsg|new$ctr/) {  
  $subject = $ENV{'form.msgsub'} if ($includemsg =~ /^msgsub/);  
  my (@msgnum) = split(/,/,$includemsg);  
  foreach (@msgnum) {  
     $message.=$ENV{'form.'.$_} if ($_ =~ /savemsg|newmsg/ && $_ ne '');  
  }  
  $message =~ s/\s+/ /g;  
  $msgstatus = &Apache::lonmsg::user_normal_msg ($uname,$udom,  
        $ENV{'form.msgsub'},$message);  
     }  
     if ($ENV{'form.collaborator'.$ctr}) {  
  my (@collaborators) = split(/:/,$ENV{'form.collaborator'.$ctr});  
  foreach (@collaborators) {  
     &saveHandGrade($request,$url,$symb,$_,$udom,$ctr,  
    $ENV{'form.unamedom'.$ctr});  
     if ($message ne '') {  
  $msgstatus = &Apache::lonmsg::user_normal_msg ($_,$udom,  
        $ENV{'form.msgsub'},  
        $message);  
     }  
  }  
     }  
     $ctr++;  
  }  
     }  
     my $firststu = $ENV{'form.unamedom0'};      my $firststu = $ENV{'form.unamedom0'};
     my $laststu = $ENV{'form.unamedom'.($ngrade-1)};      my $laststu = $ENV{'form.unamedom'.($ngrade-1)};
     $ctr = 2;      $ctr = 2;
Line 785  sub processHandGrade { Line 1194  sub processHandGrade {
  $ctr++;   $ctr++;
  $laststu = $firststu if ($ctr > $ngrade);   $laststu = $firststu if ($ctr > $ngrade);
     }      }
     my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($ENV{'form.section'},'0');  
   
       my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($ENV{'form.section'},'0');
     my (@parsedlist,@nextlist);      my (@parsedlist,@nextlist);
     my ($nextflg) = 0;      my ($nextflg) = 0;
     foreach ( sort(@{ $$classlist{$ENV{'form.section'}} }) ) {      foreach (sort {$$fullname{$a} cmp $$fullname{$b} } keys %$fullname) {
  if ($nextflg == 1 && $button =~ /Next$/) {   if ($nextflg == 1 && $button =~ /Next$/) {
     push @parsedlist,$_;      push @parsedlist,$_;
  }   }
Line 805  sub processHandGrade { Line 1214  sub processHandGrade {
     foreach my $student (@parsedlist) {      foreach my $student (@parsedlist) {
  my ($uname,$udom) = split(/:/,$student);   my ($uname,$udom) = split(/:/,$student);
  if ($ENV{'form.submitonly'} eq 'yes') {   if ($ENV{'form.submitonly'} eq 'yes') {
     my (%status) = &student_gradeStatus($ENV{'form.url'},$udom,$uname,$partlist) ;      my (%status) = &student_gradeStatus($ENV{'form.url'},$symb,$udom,$uname,$partlist) ;
     my $statusflg = '';      my $statusflg = '';
     foreach (keys(%status)) {      foreach (keys(%status)) {
  $statusflg = 1 if ($status{$_} ne 'nothing');   $statusflg = 1 if ($status{$_} ne 'nothing');
  my ($foo,$partid,$foo) = split(/\./,$_);   my ($foo,$partid,$foo1) = split(/\./);
  $statusflg = '' if ($status{'resource.'.$partid.'.submitted_by'} ne '');   $statusflg = '' if ($status{'resource.'.$partid.'.submitted_by'} ne '');
     }      }
     next if ($statusflg eq '');      next if ($statusflg eq '');
Line 823  sub processHandGrade { Line 1232  sub processHandGrade {
   
     foreach (sort @nextlist) {      foreach (sort @nextlist) {
  my ($uname,$udom,$submitter) = split(/:/);   my ($uname,$udom,$submitter) = split(/:/);
  $ENV{'form.student'} = $uname;   $ENV{'form.student'}  = $uname;
    $ENV{'form.userdom'}  = $udom;
  $ENV{'form.fullname'} = $$fullname{$_};   $ENV{'form.fullname'} = $$fullname{$_};
 # $ENV{'form.'.$_.':submitted_by'} = $submitter;  # $ENV{'form.'.$_.':submitted_by'} = $submitter;
 # print "submitter=$ENV{'form.'.$_.':submitted_by'}= $submitter:<br>";  # print "submitter=$ENV{'form.'.$_.':submitted_by'}= $submitter:<br>";
Line 840  sub processHandGrade { Line 1250  sub processHandGrade {
     return '';      return '';
 }  }
   
   #---- Save the score and award for each student, if changed
 sub saveHandGrade {  sub saveHandGrade {
     my ($request,$url,$symb,$stuname,$domain,$newflg,$submitter) = @_;      my ($request,$url,$symb,$stuname,$domain,$newflg,$submitter) = @_;
 #   my %record=&Apache::lonnet::restore($symb,$ENV{'request.course.id'},$domain,$stuname);      my %record=&Apache::lonnet::restore($symb,$ENV{'request.course.id'},$domain,$stuname);
     my %newrecord;      my %newrecord;
     foreach (split(/:/,$ENV{'form.partlist'.$newflg})) {      foreach (split(/:/,$ENV{'form.partlist'.$newflg})) {
  if ($ENV{'form.GRADE_SEL'.$newflg.'_'.$_} eq 'excused') {   if ($ENV{'form.GD_SEL'.$newflg.'_'.$_} eq 'excused') {
     $newrecord{'resource.'.$_.'.solved'} = 'excused';      $newrecord{'resource.'.$_.'.solved'} = 'excused' 
    if ($record{'resource.'.$_.'.solved'} ne 'excused');
  } else {   } else {
     my $pts    = ($ENV{'form.GRADE_BOX'.$newflg.'_'.$_} ne '' ?       my $pts = ($ENV{'form.GD_BOX'.$newflg.'_'.$_} ne '' ? 
   $ENV{'form.GRADE_BOX'.$newflg.'_'.$_} : $ENV{'form.RADVAL'.$newflg.'_'.$_});         $ENV{'form.GD_BOX'.$newflg.'_'.$_} : 
          $ENV{'form.RADVAL'.$newflg.'_'.$_});
     if ($pts eq '') {      if ($pts eq '') {
  &userError($request,'No point was assigned for part id '.$_.' and for username '.$stuname.'.');   &userError($request,'No point was assigned for part '.$_.
      ' and for username '.$stuname.'.');
  return 'error';   return 'error';
     }      }
     my $wgt    = $ENV{'form.WGT'.$newflg.'_'.$_} eq '' ? 1 : $ENV{'form.WGT'.$newflg.'_'.$_};      my $wgt = $ENV{'form.WGT'.$newflg.'_'.$_} eq '' ? 1 : 
    $ENV{'form.WGT'.$newflg.'_'.$_};
     my $partial= $pts/$wgt;      my $partial= $pts/$wgt;
     $newrecord{'resource.'.$_.'.awarded'}  = $partial;      $newrecord{'resource.'.$_.'.awarded'}  = $partial 
    if ($record{'resource.'.$_.'.awarded'} ne $partial);
       my $reckey = 'resource.'.$_.'.solved';
     if ($partial == 0) {      if ($partial == 0) {
  $newrecord{'resource.'.$_.'.solved'} = 'incorrect_by_override';   $newrecord{$reckey} = 'incorrect_by_override' 
       if ($record{$reckey} ne 'incorrect_by_override');
     } else {      } else {
  $newrecord{'resource.'.$_.'.solved'} = 'correct_by_override';   $newrecord{$reckey} = 'correct_by_override' 
       if ($record{$reckey} ne 'correct_by_override');
     }      }
     $newrecord{'resource.'.$_.'.submitted_by'} = $submitter if ($submitter);      $newrecord{'resource.'.$_.'.submitted_by'} = $submitter 
    if ($submitter && ($record{'resource.'.$_.'.submitted_by'} ne $submitter));
  }   }
     }      }
       
     if ( scalar(keys(%newrecord)) > 0 ) {      if (scalar(keys(%newrecord)) > 0) {
  $newrecord{'resource.regrader'}="$ENV{'user.name'}:$ENV{'user.domain'}";   $newrecord{'resource.regrader'}="$ENV{'user.name'}:$ENV{'user.domain'}";
 # &print_hash($request,\%newrecord);   &Apache::lonnet::cstore(\%newrecord,$symb,
  &Apache::lonnet::cstore(\%newrecord,$symb,$ENV{'request.course.id'},$domain,$stuname);   $ENV{'request.course.id'},$domain,$stuname);
     }      }
     return '';      return '';
 }  }
   
 sub get_symb_and_url {  #--------------------------------------------------------------------------------------
     my ($request) = @_;  #
     (my $url=$ENV{'form.url'}) =~ s-^http://($ENV{'SERVER_NAME'}|$ENV{'HTTP_HOST'})--;  #-------------------------- Next few routines handles grading by section or whole class
     my $symb=($ENV{'form.symb'} ne '' ? $ENV{'form.symb'} : (&Apache::lonnet::symbread($url)));  #
     if ($symb eq '') { $request->print("Unable to handle ambiguous references:$url:."); return ''; }  #--- Javascript to handle grading by section or whole class
     return ($symb,$url);  sub viewgrades_js {
 }      my ($request) = shift;
   
 sub show_grading_menu_form {  
     my ($symb,$url)=@_;  
     my $result.='<form action="/adm/grades" method="post">'."\n".  
  '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".  
  '<input type="hidden" name="url" value="'.$url.'" />'."\n".  
  '<input type="hidden" name="command" value="gradingmenu" />'."\n".  
  '<input type="submit" name="submit" value="Grading Menu" />'."\n".  
  '</form>'."\n";  
     return $result;  
 }  
   
 sub gradingmenu {      $request->print(<<VIEWJAVASCRIPT);
     my ($request) = @_;  <script type="text/javascript" language="javascript">
     my ($symb,$url)=&get_symb_and_url($request);      function viewOneStudent(user,domain) {
     if (!$symb) {return '';}   document.onestudent.student.value = user;
     my $result='<h2>&nbsp;<font color="#339933">Select a Grading Method</font></h2>';   document.onestudent.userdom.value = domain;
     $result.='<table border="0">';   document.onestudent.submit();
     $result.='<tr><td colspan=3><font size=+1><b>Resource: </b>'.$url.'</font></td></tr>';  
     my ($partlist,$handgrade) = &response_type($url);  
     my ($resptype,$hdgrade)=('','no');  
     for (sort keys(%$handgrade)) {  
  my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});  
  $resptype = $responsetype;  
  $hdgrade = $handgrade if ($handgrade eq 'yes');  
  $result.='<tr><td><b>Part id: </b>'.$_.'</td>'.  
     '<td><b>Type: </b>'.$responsetype.'</td>'.  
     '<td><b>Handgrade: </b>'.$handgrade.'</font></td></tr>';  
     }      }
     $result.='</table>';  
     $result.=&view_edit_entire_class_form($symb,$url).'<br />';  
     $result.=&upcsvScores_form($symb,$url).'<br />';  
     $result.=&viewGradeaStu_form($symb,$url,$resptype,$hdgrade).'<br />';  
     $result.=&verifyReceipt_form($symb,$url).'<br />';  
     $result.=&view_classlist_form($symb,$url);  
   
     return $result;  
 }  
   
 sub view_classlist_form {      function writePoint(partid,weight,point) {
     my ($symb,$url)=@_;   var radioButton = eval("document.classgrade.RADVAL_"+partid);
     my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";   var textbox = eval("document.classgrade.TEXTVAL_"+partid);
     $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";   if (point == "textval") {
     $result.='&nbsp;<b>View Class List</b></td></tr>'."\n";      var point = eval("document.classgrade.TEXTVAL_"+partid+".value");
     $result.='<tr bgcolor=#ffffe6><td>'."\n";      if (isNaN(point) || point < 0) {
     $result.='<form action="/adm/grades" method="post">'."\n".   alert("A number equal or greater than 0 is expected. Entered value = "+point);
  '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".   var resetbox = false;
  '<input type="hidden" name="url" value="'.$url.'" />'."\n".   for (var i=0; i<radioButton.length; i++) {
  '<input type="hidden" name="command" value="viewclasslist" />'."\n";      if (radioButton[i].checked) {
     $result.='&nbsp;<input type="submit" name="submit" value="View Class" /></form>'."\n";   textbox.value = i;
     $result.='</td></tr></table>'."\n";   resetbox = true;
     $result.='</td></tr></table>'."\n";      }
     return $result;   }
 }   if (!resetbox) {
       textbox.value = "";
 sub viewclasslist {   }
     my ($request) = shift;   return;
     my ($coursedomain,$coursenum) = split(/_/,$ENV{'request.course.id'});      }
     my %classlist=&Apache::lonnet::dump('classlist',$coursedomain,$coursenum);      if (point > weight) {
     $request->print('<table border=1>');   var resp = confirm("You entered a value ("+point+
     foreach (sort keys(%classlist)) {     ") greater than the weight for the part. Accept?");
 #    my ($unam,$udom) = split(/:/,$_,2);   if (resp == false) {
 #    my $section = &Apache::lonnet::usection($udom,$unam,$ENV{'request.course.id'});      textbox.value = "";
 #    my $fullname = &get_fullname ($unam,$udom);      return;
 #    my @uname;   }
 #    $uname[0]=$unam;      }
 #    my %userid=&Apache::lonnet::idrget($udom,@uname);      for (var i=0; i<radioButton.length; i++) {
 #    my $value=$classlist{$_}.':'.$userid{$unam}.':'.$section.':'.$fullname;   radioButton[i].checked=false;
 #    $classlist{$_}=$value;   if (point == i) {
  $request->print('<tr><td>'.$_.' </td><td><pre> '.$classlist{$_}.'</pre></td></tr>');      radioButton[i].checked=true;
     }   }
     $request->print('</table>');      }
 #  my $putresult = &Apache::lonnet::put  
 #      ('classlist',\%classlist,  
 #       $ENV{'course.'.$ENV{'request.course.id'}.'.domain'},  
 #       $ENV{'course.'.$ENV{'request.course.id'}.'.num'});  
   
     return '';  
 }  
   
 sub view_edit_entire_class_form {   } else {
     my ($symb,$url)=@_;      textbox.value = point;
     my ($classlist,$sections) = &getclasslist('all','0');   }
     my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";   for (i=0;i<document.classgrade.total.value;i++) {
     $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";      var user = eval("document.classgrade.ctr"+i+".value");
     $result.='&nbsp;<b>View/Grade Entire Section/Class</b></td></tr>'."\n";      var scorename = eval("document.classgrade.GD_"+user+
     $result.='<tr bgcolor=#ffffe6><td>'."\n";   "_"+partid+"_aw");
     $result.='<form action="/adm/grades" method="post">'."\n".      var saveval   = eval("document.classgrade.GD_"+user+
  '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".   "_"+partid+"_sv_s.value");
  '<input type="hidden" name="url" value="'.$url.'" />'."\n".      var selname   = eval("document.classgrade.GD_"+user+"_"+partid+"_sv");
  '<input type="hidden" name="command" value="viewgrades" />'."\n";      if (saveval != "correct") {
     $result.='&nbsp;<b>Select section:</b> <select name="section">'."\n";   scorename.value = point;
     foreach (sort (@$sections)) {   if (selname[0].selected != true) {
  $result.= '<option>'.$_.'</option>'."\n";      selname[0].selected = true;
    }
       }
    }
    var selval   = eval("document.classgrade.SELVAL_"+partid);
    selval[0].selected = true;
     }      }
     $result.= '<option selected="on">all</select>'."<br />\n";  
 #    $result.='&nbsp;<b>Display students who has: </b>'.  
 # '<input type="radio" name="submitonly" value="yes" checked> submitted'.  
 # '<input type="radio" name="submitonly" value="all"> everybody <br /><br />';  
     $result.='&nbsp;<input type="submit" name="submit" value="View/Grade" /></form>'."\n";  
     $result.='</td></tr></table>'."\n";  
     $result.='</td></tr></table>'."\n";  
     return $result;  
 }  
   
 sub upcsvScores_form {      function writeRadText(partid,weight) {
     my ($symb,$url) = @_;   var selval   = eval("document.classgrade.SELVAL_"+partid);
     if (!$symb) {return '';}   var radioButton = eval("document.classgrade.RADVAL_"+partid);
     my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";   var textbox = eval("document.classgrade.TEXTVAL_"+partid);
     $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";   if (selval[1].selected) {
     $result.='&nbsp;<b>Specify a file containing the class scores for above resource</b></td></tr>'."\n";      for (var i=0; i<radioButton.length; i++) {
     $result.='<tr bgcolor=#ffffe6><td>'."\n";   radioButton[i].checked=false;
     my $upfile_select=&Apache::loncommon::upfile_select_html();  
   $result.=<<ENDUPFORM;  
 <form method="post" enctype="multipart/form-data" action="/adm/grades" name="gradesupload">  
 <input type="hidden" name="symb" value="$symb" />  
 <input type="hidden" name="url" value="$url" />  
 <input type="hidden" name="command" value="csvuploadmap" />  
 $upfile_select  
 <br />&nbsp;<input type="submit" name="submit" value="Upload Grades" />  
 </form>  
 ENDUPFORM  
     $result.='</td></tr></table>'."\n";  
     $result.='</td></tr></table>'."\n";  
     return $result;  
 }  
   
 sub viewGradeaStu_form {      }
     my ($symb,$url,$response,$handgrade) = @_;      textbox.value = "";
     my ($classlist,$sections) = &getclasslist('all','0');  
     my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";  
     $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";  
     $result.='&nbsp;<b>View/Grade an Individual Student\'s Submission</b></td></tr>'."\n";  
     $result.='<tr bgcolor=#ffffe6><td>'."\n";  
     $result.='<form action="/adm/grades" method="post">'."\n".  
  '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".  
  '<input type="hidden" name="url" value="'.$url.'" />'."\n".  
  '<input type="hidden" name="response" value="'.$response.'" />'."\n".  
  '<input type="hidden" name="handgrade" value="'.$handgrade.'" />'."\n".  
  '<input type="hidden" name="showgrading" value="yes" />'."\n".  
  '<input type="hidden" name="command" value="submission" />'."\n";  
   
     $result.='&nbsp;<b>Select section:</b> <select name="section">'."\n";      for (i=0;i<document.classgrade.total.value;i++) {
     foreach (sort (@$sections)) {   var user = eval("document.classgrade.ctr"+i+".value");
  $result.= '<option>'.$_.'</option>'."\n";   var scorename = eval("document.classgrade.GD_"+user+
        "_"+partid+"_aw");
    var saveval   = eval("document.classgrade.GD_"+user+
        "_"+partid+"_sv_s.value");
    var selname   = eval("document.classgrade.GD_"+user+
        "_"+partid+"_sv");
    if (saveval != "correct") {
       scorename.value = "";
       selname[1].selected = true;
    }
       }
    } else {
       for (i=0;i<document.classgrade.total.value;i++) {
    var user = eval("document.classgrade.ctr"+i+".value");
    var scorename = eval("document.classgrade.GD_"+user+
        "_"+partid+"_aw");
    var saveval   = eval("document.classgrade.GD_"+user+
        "_"+partid+"_sv_s.value");
    var selname   = eval("document.classgrade.GD_"+user+
        "_"+partid+"_sv");
    if (saveval != "correct") {
       scorename.value = eval("document.classgrade.GD_"+user+
        "_"+partid+"_aw_s.value");;
       selname[0].selected = true;
    }
       }
    }    
     }      }
     $result.= '<option selected="on">all</select>'."\n";  
     $result.='&nbsp;&nbsp;<b>Display students who has: </b>'.  
  '<input type="radio" name="submitonly" value="yes" checked> submitted'.  
  '<input type="radio" name="submitonly" value="all"> everybody <br />';  
     $result.='&nbsp;(Section "no" implies the students were not assigned a section.)<br />'   
  if (grep /no/,@$sections);  
       
     $result.='<br />&nbsp;<input type="submit" name="submit" value="View/Grade" />'."\n".  
  '</form>'."\n";  
     $result.='</td></tr></table>'."\n";  
     $result.='</td></tr></table>'."\n";  
     return $result;  
 }  
   
 sub verifyReceipt_form {  
     my ($symb,$url) = @_;  
     my $cdom=$ENV{"course.$ENV{'request.course.id'}.domain"};  
     my $cnum=$ENV{"course.$ENV{'request.course.id'}.num"};  
     my $hostver=unpack("%32C*",$Apache::lonnet::perlvar{'lonHostID'});  
   
     my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";      function changeSelect(partid,user) {
     $result.='<table width=100% border=0><tr><td bgcolor=#e6ffff>'."\n";   var selval = eval("document.classgrade.GD_"+user+'_'+partid+"_sv");
     $result.='&nbsp;<b>Verify a Submission Receipt Issued by this Server</td></tr>'."\n";   var textbox = eval("document.classgrade.GD_"+user+'_'+partid+"_aw");
     $result.='<tr bgcolor=#ffffe6><td>'."\n";   var point  = textbox.value;
     $result.='<form action="/adm/grades" method="post">'."\n";   var weight = eval("document.classgrade.weight_"+partid+".value");
     $result.='&nbsp;<tt>'.$hostver.'-<input type="text" name="receipt" size="4"></tt><br />'."\n";  
     $result.='&nbsp;<input type="submit" name="submit" value="Verify Receipt">'."\n";   if (isNaN(point) || point < 0) {
     $result.='<input type="hidden" name="command" value="verify">'."\n";      alert("A number equal or greater than 0 is expected. Entered value = "+point);
     if ($ENV{'form.url'}) {      textbox.value = "";
  $result.='<input type="hidden" name="url" value="'.$ENV{'form.url'}.'" />';      return;
     }   }
     if ($ENV{'form.symb'}) {   if (point > weight) {
  $result.='<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />';      var resp = confirm("You entered a value ("+point+
          ") greater than the weight of the part. Accept?");
       if (resp == false) {
    textbox.value = "";
    return;
       }
    }
    selval[0].selected = true;
     }      }
     $result.='</form>';  
     $result.='</td></tr></table>'."\n";  
     $result.='</td></tr></table>'."\n";  
     return $result;  
 }  
   
 sub viewgrades {      function changeOneScore(partid,user) {
     my ($request) = @_;   var selval = eval("document.classgrade.GD_"+user+'_'+partid+"_sv");
     $request->print(<<VIEWJAVASCRIPT);   if (selval[1].selected) {
 <script type="text/javascript" language="javascript">      var boxval = eval("document.classgrade.GD_"+user+'_'+partid+"_aw");
     function viewOneStudent(user) {      boxval.value = "";
  document.onestudent.student.value = user;   }
  document.onestudent.submit();  
     }      }
   
       function resetEntry(numpart) {
    for (ctpart=0;ctpart<numpart;ctpart++) {
       var partid = eval("document.classgrade.partid_"+ctpart+".value");
       var radioButton = eval("document.classgrade.RADVAL_"+partid);
       var textbox = eval("document.classgrade.TEXTVAL_"+partid);
       var selval  = eval("document.classgrade.SELVAL_"+partid);
       for (var i=0; i<radioButton.length; i++) {
    radioButton[i].checked=false;
   
     function writePoint(partid,weight,point) {      }
  for (i=0;i<document.classgrade.total.value;i++) {      textbox.value = "";
     var user = eval("document.classgrade.counter"+i+".value");      selval[0].selected = true;
     var scorename = eval("document.classgrade.GRADE_"+user+"_"+partid+"_awarded");  
     scorename.value = point;      for (i=0;i<document.classgrade.total.value;i++) {
    var user = eval("document.classgrade.ctr"+i+".value");
    var resetscore = eval("document.classgrade.GD_"+user+
         "_"+partid+"_aw");
    resetscore.value = eval("document.classgrade.GD_"+user+
    "_"+partid+"_aw_s.value");
   
    var saveselval   = eval("document.classgrade.GD_"+user+
        "_"+partid+"_sv_s.value");
   
    var selname   = eval("document.classgrade.GD_"+user+"_"+partid+"_sv");
    if (saveselval == "excused") {
       if (selname[1].selected == false) { selname[1].selected = true;}
    } else {
       if (selname[0].selected == false) {selname[0].selected = true};
    }
       }
  }   }
 }      }
   
 </script>  </script>
 VIEWJAVASCRIPT  VIEWJAVASCRIPT
   }
   
   #--- show scores for a section or whole class w/ option to change/update a score
   sub viewgrades {
       my ($request) = shift;
       &viewgrades_js($request);
   
     my ($symb,$url) = ($ENV{'form.symb'},$ENV{'form.url'});       my ($symb,$url) = ($ENV{'form.symb'},$ENV{'form.url'}); 
     $request->print ('<h2><font color="#339933">Manual Grading</font></h2>');      my $result='<h2><font color="#339933">Manual Grading</font></h2>';
   
     my $result='<table border="0">';      $result.='<font size=+1><b>Resource: </b>'.$ENV{'form.url'}.'</font>'."\n";
     $result.='<tr><td colspan=3><font size=+1><b>Resource: </b>'.$ENV{'form.url'}.  
  '</font></td></tr>'."\n";  
     my ($partlist,$handgrade) = &response_type($ENV{'form.url'});  
     my %weight = ();  
     for (sort keys(%$handgrade)) {  
  my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});  
  my ($partid,$respid) = split (/_/);  
  my $wgt = &Apache::lonnet::EXT('resource.'.$partid.'.weight',$symb);  
  $weight{$partid} = $wgt eq '' ? '1' : $wgt;  
  $result.='<tr><td><b>Part id: </b>'.$partid.'</td>'.  
     '<td><b>Type: </b>'.$responsetype.'</td>'.  
     '<td><b>Handgrade: </b>'.$handgrade.'</font></td></tr>'."\n";  
     }  
     $request->print($result.'</table>'."\n");  
   
     #view individual student submission form - called using Javascript viewOneStudent      #view individual student submission form - called using Javascript viewOneStudent
     $result = '<form action="/adm/grades" method="post" name="onestudent">'."\n".      $result.= '<form action="/adm/grades" method="post" name="onestudent">'."\n".
  '<input type="hidden" name="symb"    value="'.$symb.'" />'."\n".   '<input type="hidden" name="symb"    value="'.$symb.'" />'."\n".
  '<input type="hidden" name="url"     value="'.$url.'" />'."\n".   '<input type="hidden" name="url"     value="'.$url.'" />'."\n".
  '<input type="hidden" name="command" value="submission" />'."\n".   '<input type="hidden" name="command" value="submission" />'."\n".
  '<input type="hidden" name="student" value="" />'."\n".   '<input type="hidden" name="student" value="" />'."\n".
    '<input type="hidden" name="userdom" value="" />'."\n".
  '</form>'."\n";   '</form>'."\n";
   
     #start the form      #beginning of class grading form
     $result.= '<form action="/adm/grades" method="post" name="classgrade">'."\n".      $result.= '<form action="/adm/grades" method="post" name="classgrade">'."\n".
  '<input type="hidden" name="symb"    value="'.$symb.'" />'."\n".   '<input type="hidden" name="symb"    value="'.$symb.'" />'."\n".
  '<input type="hidden" name="url"     value="'.$url.'" />'."\n".   '<input type="hidden" name="url"     value="'.$url.'" />'."\n".
  '<input type="hidden" name="command" value="editgrades" />'."\n".   '<input type="hidden" name="command" value="editgrades" />'."\n".
  '<input type="hidden" name="section" value="'.$ENV{'form.section'}.'" />'."\n";   '<input type="hidden" name="section" value="'.$ENV{'form.section'}.'" />'."\n";
     $result.='To assign the same score for all the students use the radio buttons or box below. '.  
  'To assign individual score fill in the score for each student in the table below.<br />';      $result.='To assign the same score for all the students use the radio buttons or '.
    'text box below. To assign scores individually fill in the score boxes for '.
    'each student in the table below. <font color="red">A part that has already '.
    'been graded does not get changed using the radio buttons or text box. '.
    'If needed, it has to be changed individually.</font>';
   
       #radio buttons/text box for assigning points for a section or class.
       #handles different parts of a problem
       my ($partlist,$handgrade) = &response_type($ENV{'form.url'});
       my %weight = ();
       my $ctsparts = 0;
     $result.='<table border="0">';      $result.='<table border="0">';
     for (sort keys (%weight)) {      for (sort keys(%$handgrade)) {
    my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});
    my ($partid,$respid) = split (/_/);
    my $wgt = &Apache::lonnet::EXT('resource.'.$partid.'.weight',$symb);
    $weight{$partid} = $wgt eq '' ? '1' : $wgt;
   
    $result.='<input type="hidden" name="partid_'.
       $ctsparts.'" value="'.$partid.'" />'."\n";
    $result.='<input type="hidden" name="weight_'.
       $partid.'" value="'.$weight{$partid}.'" />'."\n";
    $result.='<tr><td><b>Part  '.$partid.'&nbsp; &nbsp;Point:</b> </td><td>';
    $result.='<table border="0"><tr>';  
  my $ctr = 0;   my $ctr = 0;
  $result.='<tr><td><b>Part</b> '.$_.'</td><td>';   while ($ctr<=$weight{$partid}) { # display radio buttons in a nice table 10 across
  $result.='<table border="0"><tr>';  # display radio buttons in a nice table 10 across      $result.= '<td><input type="radio" name="RADVAL_'.$partid.'" '.
  while ($ctr<=$weight{$_}) {   'onclick="javascript:writePoint('.$partid.','.$weight{$partid}.
     $result.= '<td><input type="radio" name="RADVAL_'.$_.'" '.  
  'onclick="javascript:writePoint('.$_.','.$weight{$_}.  
  ','.$ctr.')" />'.$ctr."</td>\n";   ','.$ctr.')" />'.$ctr."</td>\n";
     $result.=(($ctr+1)%10 == 0 ? '</tr><tr>' : '');      $result.=(($ctr+1)%10 == 0 ? '</tr><tr>' : '');
     $ctr++;      $ctr++;
  }   }
  $result.='</tr></table>';   $result.='</tr></table>';
    $result.= '</td><td><b> or </b><input type="text" name="TEXTVAL_'.
       $partid.'" size="4" '.
       'onChange="javascript:writePoint('.$partid.','.$weight{$partid}.
       ',\'textval\')" /> /'.
       $weight{$partid}.' (problem weight)</td>'."\n";
    $result.= '</td><td><select name="SELVAL_'.$partid.'"'.
       'onChange="javascript:writeRadText('.$partid.','.$weight{$partid}.')" /> '.
       '<option selected="on"> </option>'.
       '<option>excused</option></select></td></tr>'."\n";
    $ctsparts++;
     }      }
     $result.='</tr></table>';      $result.='</table><input type="hidden" name="totalparts" value="'.$ctsparts.'" />';
       $result.='<input type="button" value="Reset" '.
     $result.= '<input type="submit" name="submit"  value="Submit Changes" />'."\n".   'onClick="javascript:resetEntry('.$ctsparts.');" TARGET=_self> &nbsp; &nbsp;';
  '<table border=0><tr><td bgcolor="#777777">'."\n".  #    $result.='<input type="button" value="Submit Changes" '.
   # 'onClick="submit();" TARGET=_self />'."\n";
       $result.= '<input type="submit" name="submit"  value="Submit Changes" />'."\n";
   
       #table listing all the students in a section/class
       #header of table
       $result.= '<table border=0><tr><td bgcolor="#777777">'."\n".
  '<table border=0><tr bgcolor="#deffff">'.   '<table border=0><tr bgcolor="#deffff">'.
  '<td><b>Username</b></td><td><b>Fullname</b></td><td><b>Domain</b></td>'."\n";   '<td><b>Fullname</b></td><td><b>Username</b></td><td><b>Domain</b></td>'."\n";
     #get list of parts for this problem  
     my (@parts) = sort(&getpartlist($url));      my (@parts) = sort(&getpartlist($url));
     foreach my $part (@parts) {      foreach my $part (@parts) {
  my $display=&Apache::lonnet::metadata($url,$part.'.display');   my $display=&Apache::lonnet::metadata($url,$part.'.display');
Line 1151  VIEWJAVASCRIPT Line 1564  VIEWJAVASCRIPT
  if ($display =~ /^Partial Credit Factor/) {   if ($display =~ /^Partial Credit Factor/) {
     $_ = $display;      $_ = $display;
     my ($partid) = /.*?(\d+).*/;      my ($partid) = /.*?(\d+).*/;
     $result.='<td><b>Score Part '.$partid.'<br>(weight = '.$weight{$partid}.')</b></td>'."\n";      $result.='<td><b>Score Part '.$partid.'<br>(weight = '.
    $weight{$partid}.')</b></td>'."\n";
     next;      next;
  }   }
  $display =~ s/Problem Status/Grade Status/;   $display =~ s/Problem Status/Grade Status<br>/;
  $result.='<td><b>'.$display.'</b></td>'."\n";   $result.='<td><b>'.$display.'</b></td>'."\n";
     }      }
     $result.='</tr>';      $result.='</tr>';
   
     #get info for each student      #get info for each student
       #list all the students - with points and grade status
     my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($ENV{'form.section'},'0');      my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($ENV{'form.section'},'0');
     my $ctr = 0;      my $ctr = 0;
     foreach ( sort(@{ $$classlist{$ENV{'form.section'}} }) ) {      foreach (sort {$$fullname{$a} cmp $$fullname{$b} } keys %$fullname) {
  (my $username = $_) = split(/:/);   my ($uname,$udom) = split(/:/);
  $result.='<input type="hidden" name="counter'.$ctr.'" value="'.$username.'" />'."\n";   $result.='<input type="hidden" name="ctr'.$ctr.'" value="'.$uname.'" />'."\n";
  $result.=&viewstudentgrade($url,$symb,$ENV{'request.course.id'},   $result.=&viewstudentgrade($url,$symb,$ENV{'request.course.id'},
    $_,$$fullname{$_},\@parts,\%weight);     $_,$$fullname{$_},\@parts,\%weight);
  $ctr++;   $ctr++;
Line 1175  VIEWJAVASCRIPT Line 1591  VIEWJAVASCRIPT
     return $result;      return $result;
 }  }
   
   #--- call by previous routine to display each student
 sub viewstudentgrade {  sub viewstudentgrade {
     my ($url,$symb,$courseid,$student,$fullname,$parts,$weight) = @_;      my ($url,$symb,$courseid,$student,$fullname,$parts,$weight) = @_;
     my ($username,$domain) = split(/:/,$student);      my ($uname,$udom) = split(/:/,$student);
     my %record=&Apache::lonnet::restore($symb,$courseid,$domain,$username);      my %record=&Apache::lonnet::restore($symb,$courseid,$udom,$uname);
     my $result='<tr bgcolor="#ffffdd"><td>'.      my $result='<tr bgcolor="#ffffdd"><td>'.
  '<a href="javascript:viewOneStudent(\''.$username.'\')"; TARGET=_self>'.$username.'</a>'.   '<a href="javascript:viewOneStudent(\''.$uname.'\',\''.$udom.
  '</td><td>'.$fullname.'</td><td align="middle">'.$domain.'</td>'."\n";   '\')"; TARGET=_self>'.$fullname.'</a>'.
    '</td><td>'.$uname.'</td><td align="middle">'.$udom.'</td>'."\n";
     foreach my $part (@$parts) {      foreach my $part (@$parts) {
  my ($temp,$part,$type)=split(/_/,$part);   my ($temp,$part,$type)=split(/_/,$part);
  my $score=$record{"resource.$part.$type"};   my $score=$record{"resource.$part.$type"};
  next if $type eq 'tries';   next if $type eq 'tries';
  if ($type eq 'awarded') {   if ($type eq 'awarded') {
     my $pts = $score*$$weight{$part};      my $pts = $score eq '' ? '' : $score*$$weight{$part};
     $result.='<td align="middle"><input type="text" name="GRADE_'.$username.'_'.$part.'_'.$type.      $result.='<input type="hidden" name="'.
  '" value="'.$pts.'" size="4" /></td>'."\n";   'GD_'.$uname.'_'.$part.'_aw_s" value="'.$pts.'" />'."\n";
 #    $result.='<td align="middle"><input type="text" name="GRADE.'.$student.'.'.$part.'.'.$type.      $result.='<td align="middle"><input type="text" name="'.
 # '" value="'.$pts.'" size="4" /></td>'."\n";   'GD_'.$uname.'_'.$part.'_aw" '.
 # } elsif ($type eq 'tries') {   'onChange="javascript:changeSelect('.$part.',\''.$uname.
 #    $result.='<td align="middle">'.$score.'&nbsp;</td>'."\n";   '\')" value="'.$pts.'" size="4" /></td>'."\n";
  } elsif ($type eq 'solved') {   } elsif ($type eq 'solved') {
     my ($status,$foo)=split(/_/,$score,2);      my ($status,$foo)=split(/_/,$score,2);
     $result.="<td align=\"middle\"><select name=\"GRADE.$student.$part.$type\">\n";  
     my $optsel = '<option>correct</option><option>incorrect</option><option>excused</option>'.  
  '<option>ungraded</option><option>nothing</option>'."\n";  
     $status = 'nothing' if ($status eq '');      $status = 'nothing' if ($status eq '');
     $optsel =~ s/<option>$status/<option selected="on">$status/;      $result.='<input type="hidden" name="'.
    'GD_'.$uname.'_'.$part.'_sv_s" value="'.$status.'" />'."\n";
       $result.='<td align="middle"><select name="'.
    'GD_'.$uname.'_'.$part.'_sv" '.
    'onChange="javascript:changeOneScore('.$part.',\''.$uname.'\')" >'."\n";
       my $optsel = '<option selected="on"> </option><option>excused</option>'."\n";
       $optsel = '<option> </option><option selected="on">excused</option>'."\n"
    if ($status eq 'excused');
     $result.=$optsel;      $result.=$optsel;
     $result.="</select></td>\n";      $result.="</select></td>\n";
  }   }
Line 1209  sub viewstudentgrade { Line 1631  sub viewstudentgrade {
     return $result;      return $result;
 }  }
   
   #--- change scores for all the students in a section/class
   #    record does not get update if unchanged
 sub editgrades {  sub editgrades {
     my ($request) = @_;      my ($request) = @_;
   
     my $symb=$ENV{'form.symb'};      my $symb=$ENV{'form.symb'};
     if ($symb eq '') { $request->print("Unable to handle ambiguous references:$symb:$ENV{'form.url'}"); return ''; }      my $url =$ENV{'form.url'};
     my $url=$ENV{'form.url'};      my $title='<h2><font color="#339933">Current Grade Status</font></h2>';
       $title.='<font size=+1><b>Resource: </b>'.$ENV{'form.url'}.'</font><br />'."\n";
     my $result.='<form action="/adm/grades" method="post">'."\n".      $title.='<font size=+1><b>Section: </b>'.$ENV{'form.section'}.'</font>'."\n";
  '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".      $title.= &show_grading_menu_form ($symb,$url);
  '<input type="hidden" name="url" value="'.$url.'" />'."\n".      my $result= '<table border="0"><tr><td bgcolor="#777777">'."\n";
  '<input type="hidden" name="command" value="viewgrades" />'."\n".      $result.= '<table border="0"><tr bgcolor="#deffff">'.
  '<input type="submit" name="submit" value="See Grades" /> <br />'."\n";   '<td rowspan=2><b>Username</b></td><td rowspan=2><b>Fullname</b></td>'."\n";
       
     my (@parts) = &getpartlist($url);      my %scoreptr = (
     my ($classlist) = &getclasslist($ENV{'form.section'},'0');      'correct'  =>'correct_by_override',
     foreach my $student ( sort(@{ $$classlist{$ENV{'form.section'}} }) ) {      'incorrect'=>'incorrect_by_override',
  $result.=&setstudentgrade($url,$symb,$ENV{'request.course.id'},$student,@parts);      'excused'  =>'excused',
     }      'ungraded' =>'ungraded_attempted',
       'nothing'  => '',
     $result.='<input type="submit" name="submit" value="See Grades" /></table></form>';      );
     return $result;      my ($classlist,$seclist,$ids,$stusec,$fullname) = &getclasslist($ENV{'form.section'},'0');
 }  
   
 sub sub_page_js {  
     my $request = shift;  
     $request->print(<<SUBJAVASCRIPT);  
 <script type="text/javascript" language="javascript">  
   function updateRadio(radioButton,formtextbox,formsel,scores) {  
      var pts = formtextbox.value;  
      var resetbox =false;  
      if (isNaN(pts) || pts < 0) {  
  alert("A number equal or greater than 0 is expected. Entered value = "+pts);  
  for (var i=0; i<radioButton.length; i++) {  
    if (radioButton[i].checked) {  
       formtextbox.value = i;  
       resetbox = true;  
    }  
  }  
  if (!resetbox) {  
    formtextbox.value = "";  
  }  
  return;  
     }  
   
     for (var i=0; i<radioButton.length; i++) {  
  radioButton[i].checked=false;  
  if (pts == i) {  
    radioButton[i].checked=true;  
  }  
     }  
     updateSelect(formsel);  
     scores.value = "0";  
   }  
   
   function writeBox(formrad,formsel,pts,scores) {  
     formrad.value = pts;  
     scores.value = "0";  
     updateSelect(formsel,pts);  
     return;  
   }  
   
   function clearRadBox(radioButton,formbox,formsel,scores) {  
     for (var i=0; i<formsel.length; i++) {  
  if (formsel[i].selected) {  
     var selectx=i;  
  }  
     }  
     if (selectx == scores.value) { return };  
     formbox.value = "";  
     for (var i=0; i<radioButton.length; i++) {  
  radioButton[i].checked=false;  
     }  
     scores.value = selectx;  
   }  
   
   function updateSelect(formsel) {  
     formsel[0].selected = true;  
     return;  
   }  
   
 //===================== Show list of keywords ====================  
   function keywords(keyform) {  
     var keywds = keyform.value;  
     var nret = prompt("Keywords list, separated by a space. Add/delete to list if desired.",keywds);  
     if (nret==null) return;  
     keyform.value = nret;  
     return;  
   }  
   
 //===================== Script to view submitted by ==================  
   function viewSubmitter(submitter) {  
     document.SCORE.refresh.value = "on";  
     document.SCORE.NCT.value = "1";  
     document.SCORE.unamedom0.value = submitter;  
     document.SCORE.submit();  
     return;  
   }  
   
 //===================== Script to add keyword(s) ==================  
   function getSel() {  
     if (document.getSelection) txt = document.getSelection();  
     else if (document.selection) txt = document.selection.createRange().text;  
     else return;  
     var cleantxt = txt.replace(new RegExp('([\\f\\n\\r\\t\\v ])+', 'g')," ");  
     if (cleantxt=="") {  
  alert("Select a word or group of words from document and then click this link.");  
  return;  
     }  
     var nret = prompt("Add selection to keyword list? Edit if desired.",cleantxt);  
     if (nret==null) return;  
     var curlist = document.SCORE.keywords.value;  
     document.SCORE.keywords.value = curlist+" "+nret;  
     return;  
   }  
   
 //====================== Script for composing message ==============  
   function msgCenter(msgform,usrctr,fullname) {  
     var Nmsg  = msgform.savemsgN.value;  
     savedMsgHeader(Nmsg,usrctr,fullname);  
     var subject = msgform.msgsub.value;  
     var rtrchk  = eval("document.SCORE.includemsg"+usrctr);  
     var msgchk = rtrchk.value;  
 //    alert("checked=>"+msgchk);  
     re = /msgsub/;  
     var shwsel = "";  
     if (re.test(msgchk)) { shwsel = "checked" }  
     displaySubject(subject,shwsel);  
     for (var i=1; i<=Nmsg; i++) {  
  var testpt = "savemsg"+i+",";  
  re = /testpt/;  
  shwsel = "";  
  if (re.test(msgchk)) { shwsel = "checked" }  
  var message = eval("document.SCORE.savemsg"+i+".value");  
  displaySavedMsg(i,message,shwsel);  
     }  
     newmsg = eval("document.SCORE.newmsg"+usrctr+".value");  
     shwsel = "";  
     re = /newmsg/;  
     if (re.test(msgchk)) { shwsel = "checked" }  
     newMsg(newmsg,shwsel);  
     msgTail();   
     return;  
   }  
   
   function savedMsgHeader(Nmsg,usrctr,fullname) {      my (@partid);
     var height = 30*Nmsg+250;      my %weight = ();
     var scrollbar = "no";      my ($i,$ctr,$count,$rec_update) = (0,0,0,0);
     if (height > 600) {      while ($ctr < $ENV{'form.totalparts'}) {
  height = 600;   my $partid = $ENV{'form.partid_'.$ctr};
  scrollbar = "yes";   push @partid,$partid;
    $weight{$partid} = $ENV{'form.weight_'.$partid};
    $ctr++;
    $result .= '<td colspan = 2 align="center"><b>Part '.$partid.
       '</b> (Weight = '.$weight{$partid}.')</td>';
     }      }
 /*    if (window.pWin)      $result .= '</tr><tr bgcolor="#deffff">';
  window.pWin.close(); */      foreach (@partid) {
     pWin = window.open('', 'MessageCenter', 'toolbar=no,location=no,scrollbars='+scrollbar+',screenx=70,screeny=75,width=600,height='+height);   $result .= '<td align="center">&nbsp;<b>Old Score</b>&nbsp;</td>'.
     pWin.document.write("<html><head>");      '<td align="center">&nbsp;<b>New Score</b>&nbsp;</td>';
     pWin.document.write("<title>Message Central</title>");      }
       $result .= '</tr>'."\n";
     pWin.document.write("<script language=javascript>");  
     pWin.document.write("function checkInput() {");      for ($i=0; $i<$ENV{'form.total'}; $i++) {
     pWin.document.write("  opener.document.SCORE.msgsub.value = document.msgcenter.msgsub.value;");   my $user = $ENV{'form.ctr'.$i};
     pWin.document.write("  var nmsg   = opener.document.SCORE.savemsgN.value;");   my %newrecord;
     pWin.document.write("  var usrctr = document.msgcenter.usrctr.value;");   my $updateflag = 0;
     pWin.document.write("  var newval = eval(\\"opener.document.SCORE.newmsg\\"+usrctr);");   my @userdom = grep /^$user:/,keys %$classlist;
     pWin.document.write("  newval.value = document.msgcenter.newmsg.value;");   my ($foo,$udom) = split(/:/,$userdom[0]);
   
     pWin.document.write("  var msgchk = \\"\\";");   $result .= '<tr bgcolor="#ffffde"><td>'.$user.'&nbsp;</td><td>'.
     pWin.document.write("  if (document.msgcenter.subchk.checked) {");      $$fullname{$userdom[0]}.'&nbsp;</td>';
     pWin.document.write("     msgchk = \\"msgsub,\\";");  
     pWin.document.write("  }");   foreach (@partid) {
     pWin.document.write(   "for (var i=1; i<=nmsg; i++) {");      my $old_aw    = $ENV{'form.GD_'.$user.'_'.$_.'_aw_s'};
     pWin.document.write("      var opnmsg = eval(\\"opener.document.SCORE.savemsg\\"+i);");      my $old_part  = $old_aw eq '' ? '' : $old_aw/$weight{$_};
     pWin.document.write("      var frmmsg = eval(\\"document.msgcenter.msg\\"+i);");      my $old_score = $scoreptr{$ENV{'form.GD_'.$user.'_'.$_.'_sv_s'}};
     pWin.document.write("      opnmsg.value = frmmsg.value;");  
     pWin.document.write("      var chkbox = eval(\\"document.msgcenter.msgn\\"+i);");      my $awarded   = $ENV{'form.GD_'.$user.'_'.$_.'_aw'};
     pWin.document.write("      if (chkbox.checked) {");      my $partial   = $awarded eq '' ? '' : $awarded/$weight{$_};
     pWin.document.write("         msgchk += \\"savemsg\\"+i+\\",\\";");      my $score;
     pWin.document.write("      }");      if ($partial eq '') {
     pWin.document.write("  }");   $score = $scoreptr{$ENV{'form.GD_'.$user.'_'.$_.'_sv_s'}};
     pWin.document.write("  if (document.msgcenter.newmsgchk.checked) {");      } elsif ($partial > 0) {
     pWin.document.write("     msgchk += \\"newmsg\\"+usrctr;");   $score = 'correct_by_override';
     pWin.document.write("  }");      } elsif ($partial == 0) {
     pWin.document.write("  var includemsg = eval(\\"opener.document.SCORE.includemsg\\"+usrctr);");   $score = 'incorrect_by_override';
     pWin.document.write("  includemsg.value = msgchk;");      }
       $score = 'excused' if (($ENV{'form.GD_'.$user.'_'.$_.'_sv'} eq 'excused') &&
 //    pWin.document.write("  alert(\\"slected=\\"+msgchk)");     ($score ne 'excused'));
     pWin.document.write("  self.close()");      $result .= '<td align="center">'.$old_aw.'&nbsp;</td>'.
    '<td align="center">'.$awarded.
     pWin.document.write("}");   ($score eq 'excused' ? $score : '').'&nbsp;</td>';
   
     pWin.document.write("<");      next if ($old_part eq $partial && $old_score eq $score);
     pWin.document.write("/script>");  
       $updateflag = 1;
     pWin.document.write("</head><body bgcolor=white>");      $newrecord{'resource.'.$_.'.awarded'}  = $partial if $partial ne '';
       $newrecord{'resource.'.$_.'.solved'}   = $score;
     pWin.document.write("<form action=\\"inactive\\" name=\\"msgcenter\\">");      $rec_update++;
     pWin.document.write("<input value=\\""+usrctr+"\\" name=\\"usrctr\\" type=\\"hidden\\">");   }
     pWin.document.write("<font color=\\"green\\" size=+1>&nbsp;Compose Message for \"+fullname+\"</font><br><br>");   $result .= '</tr>'."\n";
    if ($updateflag) {
     pWin.document.write("<table border=0 width=100%><tr><td bgcolor=\\"#777777\\">");      $count++;
     pWin.document.write("<table border=0 width=100%><tr bgcolor=\\"#ddffff\\">");      $newrecord{'resource.regrader'}="$ENV{'user.name'}:$ENV{'user.domain'}";
     pWin.document.write("<td><b>Type</b></td><td><b>Include</b></td><td><b>Message</td></tr>");      &Apache::lonnet::cstore(\%newrecord,$symb,$ENV{'request.course.id'},
 }      $udom,$user);
     function displaySubject(msg,shwsel) {   }
     pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");      }
     pWin.document.write("<td>Subject</td>");      $result .= '</table></td></tr></table>'."\n";
     pWin.document.write("<td align=\\"center\\"><input name=\\"subchk\\" type=\\"checkbox\\"" +shwsel+"></td>");      my $msg = '<b>Number of records updated = '.$rec_update.
     pWin.document.write("<td><input name=\\"msgsub\\" type=\\"text\\" value=\\""+msg+" \\"size=\\"60\\" maxlength=\\"80\\"></td></tr>");   ' for '.$count.' student'.($count <= 1 ? '' : 's').'.</b><br />'.
 }   '<b>Total number of students = '.$ENV{'form.total'}.'</b><br />';
       return $title.$msg.$result;
 function displaySavedMsg(ctr,msg,shwsel) {  
     pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");  
     pWin.document.write("<td align=\\"center\\">"+ctr+"</td>");  
     pWin.document.write("<td align=\\"center\\"><input name=\\"msgn"+ctr+"\\" type=\\"checkbox\\"" +shwsel+"></td>");  
     pWin.document.write("<td><input name=\\"msg"+ctr+"\\" type=\\"text\\" value=\\""+msg+" \\" size=\\"60\\" maxlength=\\"80\\"></td></tr>");  
 }  
   
   function newMsg(newmsg,shwsel) {  
     pWin.document.write("<tr bgcolor=\\"#ffffdd\\">");  
     pWin.document.write("<td align=\\"center\\">New</td>");  
     pWin.document.write("<td align=\\"center\\"><input name=\\"newmsgchk\\" type=\\"checkbox\\"" +shwsel+"></td>");  
     pWin.document.write("<td><input name=\\"newmsg\\" type=\\"text\\" onchange=\\"javascript:this.form.newmsgchk.checked=true\\" value=\\""+newmsg+" \\" size=\\"60\\" maxlength=\\"80\\"></td></tr>");  
 }  
   
   function msgTail() {  
     pWin.document.write("</table>");  
     pWin.document.write("</td></tr></table>&nbsp;");  
     pWin.document.write("<input type=\\"button\\" value=\\"Save\\" onClick=\\"javascript:checkInput()\\">&nbsp;&nbsp;");  
     pWin.document.write("<input type=\\"button\\" value=\\"Cancel\\" onClick=\\"self.close()\\"><br><br>");  
     pWin.document.write("</form>");  
     pWin.document.write("</body></html>");  
 }  }
   #------------- end of section for handling grading by section/class ---------
   #
   #----------------------------------------------------------------------------
   
 //====================== Script for keyword highlight options ==============  
   function kwhighlight() {  
     var kwclr    = document.SCORE.kwclr.value;  
     var kwsize   = document.SCORE.kwsize.value;  
     var kwstyle  = document.SCORE.kwstyle.value;  
     var redsel = "";  
     var grnsel = "";  
     var blusel = "";  
     if (kwclr=="red")   {var redsel="checked"};  
     if (kwclr=="green") {var grnsel="checked"};  
     if (kwclr=="blue")  {var blusel="checked"};  
     var sznsel = "";  
     var sz1sel = "";  
     var sz2sel = "";  
     if (kwsize=="0")  {var sznsel="checked"};  
     if (kwsize=="+1") {var sz1sel="checked"};  
     if (kwsize=="+2") {var sz2sel="checked"};  
     var synsel = "";  
     var syisel = "";  
     var sybsel = "";  
     if (kwstyle=="")    {var synsel="checked"};  
     if (kwstyle=="<i>") {var syisel="checked"};  
     if (kwstyle=="<b>") {var sybsel="checked"};  
     highlightCentral();  
     highlightbody('red','red',redsel,'0','normal',sznsel,'','normal',synsel);  
     highlightbody('green','green',grnsel,'+1','+1',sz1sel,'<i>','italic',syisel);  
     highlightbody('blue','blue',blusel,'+2','+2',sz2sel,'<b>','bold',sybsel);  
     highlightend();  
     return;  
   }  
   
   
   function highlightCentral() {  
     hwdWin = window.open('', 'KeywordHighlightCentral', 'toolbar=no,location=no,scrollbars=no,width=400,height=300,screenx=100,screeny=75');  
     hwdWin.document.write("<html><head>");  
     hwdWin.document.write("<title>Highlight Central</title>");  
   
     hwdWin.document.write("<script language=javascript>");  
     hwdWin.document.write("function updateChoice(flag) {");  
     hwdWin.document.write("  opener.document.SCORE.kwclr.value = radioSelection(document.hlCenter.kwdclr);");  
     hwdWin.document.write("  opener.document.SCORE.kwsize.value = radioSelection(document.hlCenter.kwdsize);");  
     hwdWin.document.write("  opener.document.SCORE.kwstyle.value = radioSelection(document.hlCenter.kwdstyle);");  
 //    hwdWin.document.write("     var kwords=opener.document.SCORE.keywords.value;");  
 //    hwdWin.document.write("     alert(\\"keywords=\\"+opener.document.SCORE.keywords.value);");  
 //    hwdWin.document.write("     return;");  
   
     hwdWin.document.write("  if (flag==1){");  
     hwdWin.document.write("     opener.document.SCORE.refresh.value = \\"on\\";");  
     hwdWin.document.write("     if (opener.document.SCORE.keywords.value!=\\"\\"){");  
     hwdWin.document.write("        opener.document.SCORE.submit();");  
     hwdWin.document.write("     }");  
     hwdWin.document.write("   }");  
     hwdWin.document.write("  self.close()");  
     hwdWin.document.write("}");  
   
     hwdWin.document.write("function radioSelection(radioButton) {");  
     hwdWin.document.write("    var selection=null;");  
     hwdWin.document.write("    for (var i=0; i<radioButton.length; i++) {");  
     hwdWin.document.write("        if (radioButton[i].checked) {");  
     hwdWin.document.write("            selection=radioButton[i].value;");  
     hwdWin.document.write("            return selection;");  
     hwdWin.document.write("        }");  
     hwdWin.document.write("    }");  
     hwdWin.document.write("}");  
   
     hwdWin.document.write("<");  
     hwdWin.document.write("/script>");  
   
     hwdWin.document.write("</head><body bgcolor=white>");  
   
     hwdWin.document.write("<form action=\\"inactive\\" name=\\"hlCenter\\">");  
     hwdWin.document.write("<font color=\\"green\\" size=+1>&nbsp;Keyword Highlight Options</font><br><br>");  
   
     hwdWin.document.write("<table border=0 width=100%><tr><td bgcolor=\\"#777777\\">");  
     hwdWin.document.write("<table border=0 width=100%><tr bgcolor=\\"#ddffff\\">");  
     hwdWin.document.write("<td><b>Text Color</b></td><td><b>Font Size</b></td><td><b>Font Style</td></tr>");  
   }  
   
   function highlightbody(clrval,clrtxt,clrsel,szval,sztxt,szsel,syval,sytxt,sysel) {   
     hwdWin.document.write("<tr bgcolor=\\"#ffffdd\\">");  
     hwdWin.document.write("<td align=\\"left\\">");  
     hwdWin.document.write("<input name=\\"kwdclr\\" type=\\"radio\\" value=\\""+clrval+"\\" "+clrsel+">&nbsp;"+clrtxt+"</td>");  
     hwdWin.document.write("<td align=\\"left\\">");  
     hwdWin.document.write("<input name=\\"kwdsize\\" type=\\"radio\\" value=\\""+szval+"\\" "+szsel+">&nbsp;"+sztxt+"</td>");  
     hwdWin.document.write("<td align=\\"left\\">");  
     hwdWin.document.write("<input name=\\"kwdstyle\\" type=\\"radio\\" value=\\""+syval+"\\" "+sysel+">&nbsp;"+sytxt+"</td>");  
     hwdWin.document.write("</tr>");  
   }  
   
   function highlightend() {   
     hwdWin.document.write("</table>");  
     hwdWin.document.write("</td></tr></table>&nbsp;");  
     hwdWin.document.write("<input type=\\"button\\" value=\\"Save\\" onClick=\\"javascript:updateChoice(0)\\">&nbsp;&nbsp;");  
     hwdWin.document.write("<input type=\\"button\\" value=\\"Save & Refresh\\" onClick=\\"javascript:updateChoice(1)\\">&nbsp;&nbsp;");  
     hwdWin.document.write("<input type=\\"button\\" value=\\"Cancel\\" onClick=\\"self.close()\\"><br><br>");  
     hwdWin.document.write("</form>");  
     hwdWin.document.write("</body></html>");  
   }  
   
 </script>  
 SUBJAVASCRIPT  
 }  
   
   #----------------------------------------------------------------------------
   #
   #-------------------------- Next few routines handles grading by csv upload
   #
   #--- Javascript to handle csv upload
 sub csvupload_javascript_reverse_associate {  sub csvupload_javascript_reverse_associate {
   return(<<ENDPICK);    return(<<ENDPICK);
   function verify(vf) {    function verify(vf) {
Line 1720  sub csvuploadassign { Line 1910  sub csvuploadassign {
     my ($symb,$url)=&get_symb_and_url($request);      my ($symb,$url)=&get_symb_and_url($request);
     if (!$symb) {return '';}      if (!$symb) {return '';}
     &Apache::loncommon::load_tmp_file($request);      &Apache::loncommon::load_tmp_file($request);
     my @gradedata=&Apache::loncommon::upfile_record_sep();      my @gradedata = &Apache::loncommon::upfile_record_sep();
     my @keyfields = split(/\,/,$ENV{'form.keyfields'});      my @keyfields = split(/\,/,$ENV{'form.keyfields'});
     my %fields=();      my %fields=();
     for (my $i=0; $i<=$ENV{'form.nfields'}; $i++) {      for (my $i=0; $i<=$ENV{'form.nfields'}; $i++) {
Line 1774  sub csvuploadassign { Line 1964  sub csvuploadassign {
     $request->print(&show_grading_menu_form($symb,$url));      $request->print(&show_grading_menu_form($symb,$url));
     return '';      return '';
 }  }
   #------------- end of section for handling csv file upload ---------
   #
   #-------------------------------------------------------------------
   
 sub send_header {  #-------------------------- Menu interface -------------------------
     my ($request)= @_;  #
     $request->print(&Apache::lontexconvert::header());  #--- Show a Grading Menu button - Calls the next routine ---
 #  $request->print("  sub show_grading_menu_form {
 #<script>      my ($symb,$url)=@_;
 #remotewindow=open('','homeworkremote');      my $result.='<form action="/adm/grades" method="post">'."\n".
 #remotewindow.close();   '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".
 #</script>");    '<input type="hidden" name="url" value="'.$url.'" />'."\n".
     $request->print('<body bgcolor="#FFFFFF">');   '<input type="hidden" name="command" value="gradingmenu" />'."\n".
    '<input type="submit" name="submit" value="Grading Menu" />'."\n".
    '</form>'."\n";
       return $result;
 }  }
   
 sub send_footer {  #--- Displays the main menu page -------
     my ($request)= @_;  sub gradingmenu {
     $request->print('</body>');      my ($request) = @_;
     $request->print(&Apache::lontexconvert::footer());      my ($symb,$url)=&get_symb_and_url($request);
       if (!$symb) {return '';}
       my $result='<h2>&nbsp;<font color="#339933">Select a Grading Method</font></h2>';
       $result.='<table border="0">';
       $result.='<tr><td colspan=3><font size=+1><b>Resource: </b>'.$url.'</font></td></tr>';
       my ($partlist,$handgrade) = &response_type($url);
       my ($resptype,$hdgrade)=('','no');
       for (sort keys(%$handgrade)) {
    my ($responsetype,$handgrade)=split(/:/,$$handgrade{$_});
    $resptype = $responsetype;
    $hdgrade = $handgrade if ($handgrade eq 'yes');
    $result.='<tr><td><b>Part </b>'.(split(/_/))[0].'</td>'.
       '<td><b>Type: </b>'.$responsetype.'</td>'.
       '<td><b>Handgrade: </b>'.$handgrade.'</font></td></tr>';
       }
       $result.='</table>';
       $result.=&view_edit_entire_class_form($symb,$url).'<br />';
       $result.=&upcsvScores_form($symb,$url).'<br />';
       $result.=&viewGradeaStu_form($symb,$url,$resptype,$hdgrade).'<br />';
       $result.=&verifyReceipt_form($symb,$url) 
    if ((&Apache::lonnet::allowed('mgr',$ENV{'request.course.id'})) && ($symb));
    
       return $result;
   }
   
   #--- Menu for grading a section or the whole class ---
   sub view_edit_entire_class_form {
       my ($symb,$url)=@_;
       my ($classlist,$sections) = &getclasslist('all','0');
       my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";
       $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";
       $result.='&nbsp;<b>View/Grade Entire Section/Class</b></td></tr>'."\n";
       $result.='<tr bgcolor=#ffffe6><td>'."\n";
       $result.='<form action="/adm/grades" method="post">'."\n".
    '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".
    '<input type="hidden" name="url" value="'.$url.'" />'."\n".
    '<input type="hidden" name="command" value="viewgrades" />'."\n";
       $result.='&nbsp;<b>Select section:</b> <select name="section">'."\n";
       foreach (sort (@$sections)) {
    $result.= '<option>'.$_.'</option>'."\n";
       }
       $result.='<option selected="on">all</select>'."<br />\n";
       $result.='&nbsp;<input type="submit" name="submit" value="View/Grade" /></form>'."\n";
       $result.='</td></tr></table>'."\n";
       $result.='</td></tr></table>'."\n";
       return $result;
   }
   
   #--- Menu to upload a csv scores ---
   sub upcsvScores_form {
       my ($symb,$url) = @_;
       if (!$symb) {return '';}
       my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";
       $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";
       $result.='&nbsp;<b>Specify a file containing the class scores for above resource</b></td></tr>'."\n";
       $result.='<tr bgcolor=#ffffe6><td>'."\n";
       my $upfile_select=&Apache::loncommon::upfile_select_html();
     $result.=<<ENDUPFORM;
   <form method="post" enctype="multipart/form-data" action="/adm/grades" name="gradesupload">
   <input type="hidden" name="symb" value="$symb" />
   <input type="hidden" name="url" value="$url" />
   <input type="hidden" name="command" value="csvuploadmap" />
   $upfile_select
   <br />&nbsp;<input type="submit" name="submit" value="Upload Grades" />
   </form>
   ENDUPFORM
       $result.='</td></tr></table>'."\n";
       $result.='</td></tr></table>'."\n";
       return $result;
   }
   
   #--- Handgrading problems ---
   sub viewGradeaStu_form {
       my ($symb,$url,$response,$handgrade) = @_;
       my ($classlist,$sections) = &getclasslist('all','0');
       my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";
       $result.='<table width=100% border=0><tr bgcolor="#e6ffff"><td>'."\n";
       $result.='&nbsp;<b>View/Grade an Individual Student\'s Submission</b></td></tr>'."\n";
       $result.='<tr bgcolor=#ffffe6><td>'."\n";
       $result.='<form action="/adm/grades" method="post">'."\n".
    '<input type="hidden" name="symb" value="'.$symb.'" />'."\n".
    '<input type="hidden" name="url" value="'.$url.'" />'."\n".
    '<input type="hidden" name="response" value="'.$response.'" />'."\n".
    '<input type="hidden" name="handgrade" value="'.$handgrade.'" />'."\n".
    '<input type="hidden" name="showgrading" value="yes" />'."\n".
    '<input type="hidden" name="command" value="submission" />'."\n";
   
       $result.='&nbsp;<b>Select section:</b> <select name="section">'."\n";
       foreach (sort (@$sections)) {
    $result.= '<option>'.$_.'</option>'."\n";
       }
       $result.= '<option selected="on">all</select>'."\n";
       $result.='&nbsp;&nbsp;<b>Display students who has: </b>'.
    '<input type="radio" name="submitonly" value="yes" checked> submitted'.
    '<input type="radio" name="submitonly" value="all"> everybody <br />';
       $result.='&nbsp;(Section "no" implies the students were not assigned a section.)<br />' 
    if (grep /no/,@$sections);
       
       $result.='<br />&nbsp;<input type="submit" name="submit" value="View/Grade" />'."\n".
    '</form>'."\n";
       $result.='</td></tr></table>'."\n";
       $result.='</td></tr></table>'."\n";
       return $result;
   }
   
   #--- Form to input a receipt number ---
   sub verifyReceipt_form {
       my ($symb,$url) = @_;
       my $cdom=$ENV{"course.$ENV{'request.course.id'}.domain"};
       my $cnum=$ENV{"course.$ENV{'request.course.id'}.num"};
       my $hostver=unpack("%32C*",$Apache::lonnet::perlvar{'lonHostID'});
   
       my $result.='<table width=100% border=0><tr><td bgcolor=#777777>'."\n";
       $result.='<table width=100% border=0><tr><td bgcolor=#e6ffff>'."\n";
       $result.='&nbsp;<b>Verify a Submission Receipt Issued by this Server</td></tr>'."\n";
       $result.='<tr bgcolor=#ffffe6><td>'."\n";
       $result.='<form action="/adm/grades" method="post">'."\n";
       $result.='&nbsp;<tt>'.$hostver.'-<input type="text" name="receipt" size="4"></tt><br />'."\n";
       $result.='&nbsp;<input type="submit" name="submit" value="Verify Receipt">'."\n";
       $result.='<input type="hidden" name="command" value="verify">'."\n";
       if ($ENV{'form.url'}) {
    $result.='<input type="hidden" name="url" value="'.$ENV{'form.url'}.'" />';
       }
       if ($ENV{'form.symb'}) {
    $result.='<input type="hidden" name="symb" value="'.$ENV{'form.symb'}.'" />';
       }
       $result.='</form>';
       $result.='</td></tr></table>'."\n";
       $result.='</td></tr></table>'."\n";
       return $result;
 }  }
   
 sub handler {  sub handler {
Line 1801  sub handler { Line 2126  sub handler {
  $request->content_type('text/html');   $request->content_type('text/html');
     }      }
     $request->send_http_header;      $request->send_http_header;
     return OK if $request->header_only;      return '' if $request->header_only;
     &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'});      &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'});
     my $url=$ENV{'form.url'};      my $url=$ENV{'form.url'};
     my $symb=$ENV{'form.symb'};      my $symb=$ENV{'form.symb'};
Line 1830  sub handler { Line 2155  sub handler {
       'grade_courseid' => $tcrsid,        'grade_courseid' => $tcrsid,
       'grade_symb' => $tsymb)));        'grade_symb' => $tsymb)));
     } else {      } else {
  $request->print('<h1>Not authorized: '.$token.'</h1>');   $request->print('<h2>Not authorized: '.$token.'</h2>');
     }                 }           
  } else {   } else {
     $request->print('<h1>Not a valid DocID: '.$token.'</h1>');      $request->print('<h2>Not a valid DocID: '.$token.'</h2>');
  }   }
     } else {      } else {
  $request->print(&Apache::lonxml::tokeninputfield());   $request->print(&Apache::lonxml::tokeninputfield());
Line 1863  sub handler { Line 2188  sub handler {
     $request->print(&viewclasslist($request));      $request->print(&viewclasslist($request));
  } elsif ($command eq 'csvuploadmap') {   } elsif ($command eq 'csvuploadmap') {
     $request->print(&csvuploadmap($request));      $request->print(&csvuploadmap($request));
 #    } elsif ($command eq 'receiptInput') {  
 #      &receiptInput($request);  
  } elsif ($command eq 'csvuploadassign') {   } elsif ($command eq 'csvuploadassign') {
     if ($ENV{'form.associate'} ne 'Reverse Association') {      if ($ENV{'form.associate'} ne 'Reverse Association') {
  $request->print(&csvuploadassign($request));   $request->print(&csvuploadassign($request));
Line 1881  sub handler { Line 2204  sub handler {
  }   }
     }      }
     &send_footer($request);      &send_footer($request);
     return OK;      return '';
   }
   
   sub send_header {
       my ($request)= @_;
       $request->print(&Apache::lontexconvert::header());
   #  $request->print("
   #<script>
   #remotewindow=open('','homeworkremote');
   #remotewindow.close();
   #</script>"); 
       $request->print('<body bgcolor="#FFFFFF">');
   }
   
   sub send_footer {
       my ($request)= @_;
       $request->print('</body>');
       $request->print(&Apache::lontexconvert::footer());
 }  }
   
 1;  1;

Removed from v.1.41  
changed lines
  Added in v.1.44


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>