Diff for /loncom/homework/response.pm between versions 1.66 and 1.140

version 1.66, 2002/10/14 20:37:27 version 1.140, 2006/06/19 16:32:35
Line 25 Line 25
 #  #
 # http://www.lon-capa.org/  # http://www.lon-capa.org/
 #  #
 # 11/23,11/24,11/28 Gerd Kortemeyer  
 # Guy Albertelli  
 # 08/04,08/07 Gerd Kortemeyer  
   
 package Apache::response;  package Apache::response;
 use strict;  use strict;
   use Apache::lonlocal;
   use Apache::lonnet;
   
 BEGIN {  BEGIN {
   &Apache::lonxml::register('Apache::response',('responseparam','parameter','dataresponse'));      &Apache::lonxml::register('Apache::response',('responseparam','parameter','dataresponse','customresponse','mathresponse'));
 }  }
   
 sub start_response {  sub start_response {
   my ($parstack,$safeeval)=@_;      my ($parstack,$safeeval)=@_;
   my $id= &Apache::lonxml::get_param('id',$parstack,$safeeval);      my $id = &Apache::lonxml::get_id($parstack,$safeeval);
   if ($id eq '') { $id = $Apache::lonxml::curdepth; }      if ($#Apache::inputtags::import > -1) {
   if ($#Apache::inputtags::import > -1) {   &Apache::lonxml::debug("Turning :$id: into");
     &Apache::lonxml::debug("Turning :$id: into");   $id = join('_',@Apache::inputtags::import).'_'.$id;
     $id = join('_',@Apache::inputtags::import).'_'.$id;   &Apache::lonxml::debug("New  :$id:");
     &Apache::lonxml::debug("New  :$id:");      }
   }      push (@Apache::inputtags::response,$id);
   push (@Apache::inputtags::response,$id);      push (@Apache::inputtags::responselist,$id);
   push (@Apache::inputtags::responselist,$id);      @Apache::inputtags::inputlist=();
   @Apache::inputtags::inputlist=();      if ($Apache::inputtags::part eq '' && 
   return $id;   !$Apache::lonhomework::ignore_response_errors) {
    &Apache::lonxml::error(&HTML::Entities::encode(&mt("Found a <*response> outside of a <part> in a <part>ed problem"),'<>&"'));
       }
       if ($Apache::inputtags::response_with_no_part &&
    $Apache::inputtags::part ne '0') {
    &Apache::lonxml::error(&HTML::Entities::encode(&mt("<*response>s are both inside of <part> and outside of <part>, this is not a valid problem, errors in grading may occur."),'<>&"').'<br />');
       }
       if ($Apache::inputtags::part eq '0') {
    $Apache::inputtags::response_with_no_part=1;
       }
       return $id;
 }  }
   
 sub end_response {  sub end_response {
   pop @Apache::inputtags::response;      #pop @Apache::inputtags::response;
   @Apache::inputtags::inputlist=();      @Apache::inputtags::inputlist=();
   return '';      return '';
 }  }
   
 sub start_hintresponse {  sub start_hintresponse {
   my ($parstack,$safeeval)=@_;      my ($parstack,$safeeval)=@_;
   my $id= &Apache::lonxml::get_param('id',$parstack,$safeeval);      my $id = &Apache::lonxml::get_id($parstack,$safeeval);
   if ($id eq '') { $id = $Apache::lonxml::curdepth; }      push (@Apache::inputtags::hint,$id);
   push (@Apache::inputtags::response,$id);      push (@Apache::inputtags::hintlist,$id);
   push (@Apache::inputtags::paramstack,[%Apache::inputtags::params]);      push (@Apache::inputtags::paramstack,[%Apache::inputtags::params]);
   return $id;      return $id;
 }  }
   
 sub end_hintresponse {  sub end_hintresponse {
   pop @Apache::inputtags::response;      pop @Apache::inputtags::hint;
   %Apache::inputtags::params=@{pop(@Apache::inputtags::paramstack)};      if (defined($Apache::inputtags::paramstack[-1])) {
   return '';   %Apache::inputtags::params=
       @{ pop(@Apache::inputtags::paramstack) };
       }
       return '';
   }
   
   my @randomseeds;
   sub pushrandomnumber {
       my $rand_alg=&Apache::lonnet::get_rand_alg();
       if (!$rand_alg || $rand_alg eq '32bit' || $rand_alg eq '64bit' ||
    $rand_alg eq '64bit2') {
    # do nothing
       } else {
    my @seed=&Math::Random::random_get_seed();
    push(@randomseeds,\@seed);
       }
       &Apache::response::setrandomnumber(@_);
   }
   sub poprandomnumber {
       my $rand_alg=&Apache::lonnet::get_rand_alg();
       if (!$rand_alg || $rand_alg eq '32bit' || $rand_alg eq '64bit' ||
    $rand_alg eq '64bit2') {
    return;
       }
       my $seed=pop(@randomseeds);
       if ($seed) {
    &Math::Random::random_set_seed(@$seed);
       } else {
    &Apache::lonxml::error("Unable to restore random algorithm.");
       }
 }  }
   
 # used by response to set the non-safe space random number generator to something  
 # that is stable and unique based on the part number and response number  
 sub setrandomnumber {  sub setrandomnumber {
   my $rndseed;      my ($ignore_id2) = @_;
   if ($ENV{'request.state'} eq "construct") {      my $rndseed;
     $rndseed=$ENV{'form.rndseed'};      $rndseed=&Apache::structuretags::setup_rndseed();
     if (!$rndseed) { $rndseed=time; }      if (!defined($rndseed)) { $rndseed=&Apache::lonnet::rndseed(); }
   } else {      &Apache::lonxml::debug("randseed $rndseed");
     my ($symb,$courseid,$domain,$name) = &Apache::lonxml::whichuser();      #  $rndseed=unpack("%32i",$rndseed);
     $rndseed=&Apache::lonnet::rndseed($symb,$courseid,$domain,$name);      my $rand_alg=&Apache::lonnet::get_rand_alg();
   }      my ($rndmod,$rndmod2);
   &Apache::lonxml::debug("randseed $rndseed");  
 #  $rndseed=unpack("%32i",$rndseed);      my ($id1,$id2,$shift_amt);
   $rndseed=$rndseed      if ($Apache::lonhomework::parsing_a_problem) {
     +(&Apache::lonnet::numval($Apache::inputtags::part) << 10);   $id1=$Apache::inputtags::part;
   if (defined($Apache::inputtags::response['-1'])) {   if (defined($Apache::inputtags::response[-1])) {
     $rndseed=$rndseed      $id2=$Apache::inputtags::response[-1];
       +&Apache::lonnet::numval($Apache::inputtags::response['-1']);   }
   }   $shift_amt=scalar(@Apache::inputtags::responselist);
   &Math::Random::random_set_seed_from_phrase($rndseed);      } elsif ($Apache::lonhomework::parsing_a_task) {
   &Apache::lonxml::debug("randseed $rndseed");   $id1=$Apache::bridgetask::dimension;
   return '';   if (!$ignore_id2 && defined($Apache::bridgetask::instance[-1])) {
       $id2=$Apache::bridgetask::instance[-1];
    }
    $shift_amt=scalar(@Apache::bridgetask::instance);
       } 
       &Apache::lonxml::debug("id1: $id1, id2: $id2, shift_amt: $shift_amt");
       if (!$rand_alg || $rand_alg eq '32bit' || $rand_alg eq '64bit' ||
    $rand_alg eq '64bit2') {
    $rndmod=(&Apache::lonnet::numval($id1) << 10);
    if (defined($id2)) { $rndmod+=&Apache::lonnet::numval($id2); }
       } elsif ($rand_alg eq '64bit3') {
    $rndmod=(&Apache::lonnet::numval2($id1) << 10);
    if (defined($id2)) { $rndmod+=&Apache::lonnet::numval2($id2); }
       } elsif ($rand_alg eq '64bit4') {
    my $shift=(4*$shift_amt)%30;
    $rndmod=(&Apache::lonnet::numval3($id1) << (($shift+15)%30));
    if (defined($id2)) {
       $rndmod+=(&Apache::lonnet::numval3($id2) << $shift );
    }
       } else {
    ($rndmod,$rndmod2)=&Apache::lonnet::digest("$id1,$id2");
       }
   
       if ($rndseed =~/([,:])/) {
    my $char=$1;
    use integer;
    my ($num1,$num2)=split(/\Q$char\E/,$rndseed);
    $num1+=$rndmod;
    $num2+= ((defined($rndmod2)) ? $rndmod2 : $rndmod);
    if($Apache::lonnet::_64bit) { $num1=(($num1<<32)>>32); $num2=(($num2<<32)>>32); }
    $rndseed=$num1.$char.$num2;
       } else {
    $rndseed+=$rndmod;
    if($Apache::lonnet::_64bit) {
       use integer;
       $rndseed=(($rndseed<<32)>>32);
    }
       }
       &Apache::lonxml::debug("randseed $rndmod $rndseed");
       &Apache::lonnet::setup_random_from_rndseed($rndseed);
       return '';
 }  }
   
 sub meta_parameter_write {  sub meta_parameter_write {
Line 101  sub meta_parameter_write { Line 177  sub meta_parameter_write {
     my $partref=$Apache::inputtags::part;      my $partref=$Apache::inputtags::part;
     my $result='<parameter part="'.$Apache::inputtags::part.'"';      my $result='<parameter part="'.$Apache::inputtags::part.'"';
     if (defined($Apache::inputtags::response[-1])) {      if (defined($Apache::inputtags::response[-1])) {
       $result.=            ' id="'.$Apache::inputtags::response[-1].'"';   $result.=            ' id="'.$Apache::inputtags::response[-1].'"';
       $partref.='_'.$Apache::inputtags::response[-1];   $partref.='_'.$Apache::inputtags::response[-1];
     }      }
     $result.=            ' name="'.$name.'"'.      $result.=            ' name="'.$name.'"'.
                          ' type="'.$type.'"'.                           ' type="'.$type.'"'.
 ($default?' default="'.$default.'"':'').  (defined($default)?' default="'.$default.'"':'').
 ($display?' display="'.$display.' [Part: '.$partref.']"':'')  (defined($display)?' display="'.$display.' [Part: '.$partref.']"':'')
              .'></parameter>'               .'></parameter>'
              ."\n";               ."\n";
     return $result;      return $result;
Line 117  sub meta_package_write { Line 193  sub meta_package_write {
     my $name=shift;      my $name=shift;
     my $result = '<parameter part="'.$Apache::inputtags::part.'"';      my $result = '<parameter part="'.$Apache::inputtags::part.'"';
     if(defined($Apache::inputtags::response[-1])) {      if(defined($Apache::inputtags::response[-1])) {
       $result.=              ' id="'.$Apache::inputtags::response[-1].'"';   $result.= ' id="'.$Apache::inputtags::response[-1].'"';
     }      }
     $result.=' package="'.$name.'"></parameter>'."\n";      $result.=' package="'.$name.'"></parameter>'."\n";
     return $result;      return $result;
Line 128  sub meta_stores_write { Line 204  sub meta_stores_write {
     my $partref=$Apache::inputtags::part;      my $partref=$Apache::inputtags::part;
     my $result = '<stores part="'.$Apache::inputtags::part.'"';      my $result = '<stores part="'.$Apache::inputtags::part.'"';
     if (defined($Apache::inputtags::response[-1])) {      if (defined($Apache::inputtags::response[-1])) {
       $result.=           ' id="'.$Apache::inputtags::response[-1].'"';   $result.=           ' id="'.$Apache::inputtags::response[-1].'"';
       $partref.='_'.$Apache::inputtags::response[-1];   $partref.='_'.$Apache::inputtags::response[-1];
     }      }
     $result.=          ' name="'.$name.'"'.      $result.=          ' name="'.$name.'"'.
                        ' type="'.$type.'"'.                         ' type="'.$type.'"'.
Line 144  sub mandatory_part_meta { Line 220  sub mandatory_part_meta {
 # output (to lonspreadsheet)  # output (to lonspreadsheet)
 # of each part  # of each part
 #  #
  return      return
 #    &meta_parameter_write('opendate','date_start','',  #    &meta_parameter_write('opendate','date_start','',
 #                          'Opening Date').  #                          'Opening Date').
 #    &meta_parameter_write('duedate','date_end','',  #    &meta_parameter_write('duedate','date_end','',
Line 155  sub mandatory_part_meta { Line 231  sub mandatory_part_meta {
 #                          'Available Points').  #                          'Available Points').
 #    &meta_parameter_write('maxtries','int_pos','',  #    &meta_parameter_write('maxtries','int_pos','',
 #                          'Maximum Number of Tries').  #                          'Maximum Number of Tries').
     &meta_package_write('part').   &meta_package_write('part').
     &meta_stores_write('solved','string',          &meta_stores_write('solved','string',
                           'Problem Status').     'Problem Status').
     &meta_stores_write('tries','int_zeropos',          &meta_stores_write('tries','int_zeropos',
                           'Number of Attempts').     'Number of Attempts').
     &meta_stores_write('awarded','float',          &meta_stores_write('awarded','float',
           'Partial Credit Factor');     'Partial Credit Factor');
 #  #
 # Note: responseid-specific data 'submission' and 'awarddetail'  # Note: responseid-specific data 'submission' and 'awarddetail'
 # not available to spreadsheet -> skip here  # not available to spreadsheet -> skip here
 #  #
 }  }
   
   sub meta_part_order {
       if (@Apache::inputtags::partlist) {
    my @parts=@Apache::inputtags::partlist;
    shift(@parts);
    return '<partorder>'.join(',',@parts).'</partorder>'."\n";
       } else {
    return '<partorder>0</partorder>'."\n";
       }
   }
   
   sub meta_response_order {
       if (@Apache::inputtags::responselist) {
    return '<responseorder>'.join(',',@Apache::inputtags::responselist).
       '</responseorder>'."\n";
       }
   }
   
 sub check_for_previous {  sub check_for_previous {
   my ($curresponse,$partid,$id) = @_;      my ($curresponse,$partid,$id) = @_;
   my %previous;      my %previous;
   $previous{'used'} = 0;      $previous{'used'} = 0;
   foreach my $key (reverse(sort(keys(%Apache::lonhomework::history)))) {      foreach my $key (sort(keys(%Apache::lonhomework::history))) {
     if ($key =~ /resource\.$partid\.$id\.submission/) {   if ($key =~ /resource\.$partid\.$id\.submission$/) {
       &Apache::lonxml::debug("Trying $key");      &Apache::lonxml::debug("Trying $key");
       my $pastresponse=$Apache::lonhomework::history{$key};      my $pastresponse=$Apache::lonhomework::history{$key};
       if ($pastresponse eq $curresponse) {      if ($pastresponse eq $curresponse) {
  $previous{'used'} = 1;   $previous{'used'} = 1;
  my $history;   my $history;
  if ( $key =~ /^(\d+):/ ) {   if ( $key =~ /^(\d+):/ ) {
   $history=$1;      $history=$1;
   $previous{'award'} = $Apache::lonhomework::history{"$history:resource.$partid.$id.awarddetail"};      $previous{'award'} = $Apache::lonhomework::history{"$history:resource.$partid.$id.awarddetail"};
   $previous{'last'}='0';      $previous{'last'}='0';
  } else {      push(@{ $previous{'version'} },$history);
   $previous{'award'} = $Apache::lonhomework::history{"resource.$partid.$id.awarddetail"};   } else {
   $previous{'last'}='1';      $previous{'award'} = $Apache::lonhomework::history{"resource.$partid.$id.awarddetail"};
       $previous{'last'}='1';
    }
    if (! $previous{'award'} ) { $previous{'award'} = 'UNKNOWN'; }
    &Apache::lonxml::debug("got a match :$previous{'award'}:$previous{'used'}:");
       }
  }   }
  if (! $previous{'award'} ) { $previous{'award'} = 'UNKNOWN'; }  
  &Apache::lonxml::debug("got a match :$previous{'award'}:$previous{'used'}:");  
  last;  
       }  
     }      }
   }      &Apache::lonhomework::showhash(%previous);
   return %previous;      return %previous;
 }  }
   
 sub handle_previous {  sub handle_previous {
   my ($previous,$ad)=@_;      my ($previous,$ad)=@_;
   if ($$previous{'used'} && ($$previous{'award'} eq $ad) ) {      if ($$previous{'used'} && ($$previous{'award'} eq $ad) ) {
     if ($$previous{'last'}) {   if ($$previous{'last'}) {
       push(@Apache::inputtags::previous,'PREVIOUSLY_LAST');      push(@Apache::inputtags::previous,'PREVIOUSLY_LAST');
     } else {      push(@Apache::inputtags::previous_version,$$previous{'version'});
       push(@Apache::inputtags::previous,'PREVIOUSLY_USED');   } elsif ($Apache::lonhomework::type ne 'survey') {
       push(@Apache::inputtags::previous,'PREVIOUSLY_USED');
       push(@Apache::inputtags::previous_version,$$previous{'version'});
    }
     }      }
   }  
 }  }
   
 sub view_or_modify {  sub view_or_modify {
   my ($symb,$courseid,$domain,$name) = &Apache::lonxml::whichuser();      my ($symb,$courseid,$domain,$name) = &Apache::lonxml::whichuser();
   my $myself=0;      my $myself=0;
   if ( ($name eq $ENV{'user.name'}) && ($domain eq $ENV{'user.domain'}) ) {      if ( ($name eq $env{'user.name'}) && ($domain eq $env{'user.domain'}) ) {
     $myself=1;   $myself=1;
   }      }
   my $vgr=&Apache::lonnet::allowed('vgr',$courseid);      my $vgr=&Apache::lonnet::allowed('vgr',$courseid);
   my $mgr=&Apache::lonnet::allowed('vgr',$courseid);      my $mgr=&Apache::lonnet::allowed('vgr',$courseid);
   if ($mgr) { return "M"; }      if ($mgr) { return "M"; }
   if ($vgr) { return "V"; }      if ($vgr) { return "V"; }
   if ($myself) { return "V"; }      if ($myself) { return "V"; }
   return '';      return '';
 }  }
   
 sub start_dataresponse {  sub start_dataresponse {
   my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
   my $id = &Apache::response::start_response($parstack,$safeeval);      my $id = &Apache::response::start_response($parstack,$safeeval);
   my $result;      my $result;
   if ($target eq 'web') {      if ($target eq 'web') {
     $result = $token->[2]->{'display'}.':';   $result = $token->[2]->{'display'}.':';
   } elsif ($target eq 'meta') {      } elsif ($target eq 'meta') {
     $result = &Apache::response::meta_stores_write($token->[2]->{'name'},   $result = &Apache::response::meta_stores_write($token->[2]->{'name'},
    $token->[2]->{'type'},         $token->[2]->{'type'},
    $token->[2]->{'display'});         $token->[2]->{'display'});
     $result .= &Apache::response::meta_package_write('dataresponse');   $result .= &Apache::response::meta_package_write('dataresponse');
   }      }
   return $result;      return $result;
 }  }
   
 sub end_dataresponse {  sub end_dataresponse {
   my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
   my $result;      my $result;
   if ( $target eq 'web' ) {      if ( $target eq 'web' ) {
   } elsif ($target eq 'grade' ) {      } elsif ($target eq 'grade' ) {
     if ( defined $ENV{'form.submitted'}) {   if ( defined $env{'form.submitted'}) {
       my ($symb,$courseid,$domain,$name)=&Apache::lonxml::whichuser();      my ($symb,$courseid,$domain,$name)=&Apache::lonxml::whichuser();
       my $allowed=&Apache::lonnet::allowed('mgr',$courseid);      my $allowed=&Apache::lonnet::allowed('mgr',$courseid);
       if ($allowed) {      if ($allowed) {
  &Apache::response::setup_params('dataresponse');   &Apache::response::setup_params('dataresponse',$safeeval);
  my $partid = $Apache::inputtags::part;   my $partid = $Apache::inputtags::part;
  my $id = $Apache::inputtags::response['-1'];   my $id = $Apache::inputtags::response['-1'];
  my $response = $ENV{'form.HWVAL'.$id};   my $response = $env{'form.HWVAL_'.$id};
  my $name = &Apache::lonxml::get_param('name',$parstack,$safeeval);   my $name = &Apache::lonxml::get_param('name',$parstack,$safeeval);
  if ( $response =~ /[^\s]/) {   if ( $response =~ /[^\s]/) {
   $Apache::lonhomework::results{"resource.$partid.$id.$name"}=      $Apache::lonhomework::results{"resource.$partid.$id.$name"}=$response;
     $response;      $Apache::lonhomework::results{"resource.$partid.$id.submission"}=$response;
   $Apache::lonhomework::results{      $Apache::lonhomework::results{"resource.$partid.$id.awarddetail"}='SUBMITTED';
  "resource.$partid.$id.submission"}=   }
   $response;      } else {
   $Apache::lonhomework::results{   $result='Not Permitted to change values.'
  "resource.$partid.$id.awarddetail"}=      }
   'SUBMITTED';   }
  }      }
       } else {      &Apache::response::end_response;
  $result='Not Permitted to change values.'      return $result;
       }  }
     }  
   }  sub start_customresponse {
   &Apache::response::end_response;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
   return $result;      my $id = &Apache::response::start_response($parstack,$safeeval);
       push(@Apache::lonxml::namespace,'customresponse');
       my $result;
       undef($Apache::response::custom_answer);
       &Apache::lonxml::register('Apache::response',('answer'));
       if ($target eq 'web') {
     if (  &Apache::response::show_answer() ) {
       my $answer = &Apache::lonxml::get_param('answerdisplay',$parstack,
      $safeeval);
       $Apache::inputtags::answertxt{$id}=$answer;
    }
       } elsif ($target eq 'edit') {
    $result.=&Apache::edit::tag_start($target,$token);
    $result.=&Apache::edit::text_arg('String to display for answer:',
    'answerdisplay',$token);
    $result.=&Apache::edit::end_row().&Apache::edit::start_spanning_row();
       } elsif ($target eq 'modified') {
    my $constructtag;
    $constructtag=&Apache::edit::get_new_args($token,$parstack,
     $safeeval,'answerdisplay');
    if ($constructtag) {
       $result = &Apache::edit::rebuild_tag($token);
       $result.=&Apache::edit::handle_insert();
    }
       } elsif ($target eq 'answer' || $target eq 'grade') {
    &Apache::response::reset_params();
       } elsif ($target eq 'meta') {
    $result .= &Apache::response::meta_package_write('customresponse');
       }
       return $result;
   }
   
   sub end_customresponse {
       my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
       my $result;
       my $part=$Apache::inputtags::part;
       my $id=$Apache::inputtags::response[-1];
       if ( $target eq 'grade' && &Apache::response::submitted() ) {
    my $response = &Apache::response::getresponse();
    if ( $response =~ /[^\s]/ && 
        $Apache::response::custom_answer_type eq 'loncapa/perl') {
       if (!$Apache::lonxml::default_homework_loaded) {
    &Apache::lonxml::default_homework_load($safeeval);
       }
       my %previous = &Apache::response::check_for_previous($response,
    $part,$id);
       $Apache::lonhomework::results{"resource.$part.$id.submission"}=
    $response;
       my $error;
       ${$safeeval->varglob('LONCAPA::customresponse_submission')}=
    $response;
       
       my $award = &Apache::run::run('{ my $submission=$LONCAPA::customresponse_submission;'.$Apache::response::custom_answer.'}',$safeeval);
       if (!&Apache::inputtags::valid_award($award)) {
    $error = $award;
    $award = 'ERROR';
       }
       &Apache::response::handle_previous(\%previous,$award);
       $Apache::lonhomework::results{"resource.$part.$id.awarddetail"}=
    $award;
       if ($error) {
    $Apache::lonhomework::results{"resource.$part.$id.awardmsg"}=
       $error;
       }
    }
       }
       pop(@Apache::lonxml::namespace);
       &Apache::lonxml::deregister('Apache::response',('answer'));
       &Apache::response::end_response();
       return $result;
   }
   
   
   sub start_mathresponse {
       my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
       my $id = &Apache::response::start_response($parstack,$safeeval);
       push(@Apache::lonxml::namespace,'mathresponse');
       my $result;
       undef($Apache::response::custom_answer);
       &Apache::lonxml::register('Apache::response',('answer'));
       if ($target eq 'web') {
     if (  &Apache::response::show_answer() ) {
       my $answer = &Apache::lonxml::get_param('answerdisplay',$parstack,
      $safeeval);
       $Apache::inputtags::answertxt{$id}=$answer;
    }
       } elsif ($target eq 'edit') {
    $result.=&Apache::edit::tag_start($target,$token);
    $result.=&Apache::edit::text_arg('String to display for answer:',
    'answerdisplay',$token);
    $result.=&Apache::edit::end_row().&Apache::edit::start_spanning_row();
       } elsif ($target eq 'modified') {
    my $constructtag;
    $constructtag=&Apache::edit::get_new_args($token,$parstack,
     $safeeval,'answerdisplay');
    if ($constructtag) {
       $result = &Apache::edit::rebuild_tag($token);
       $result.=&Apache::edit::handle_insert();
    }
       } elsif ($target eq 'answer' || $target eq 'grade') {
    &Apache::response::reset_params();
       } elsif ($target eq 'meta') {
    $result .= &Apache::response::meta_package_write('mathresponse');
       }
       return $result;
   }
   
   sub end_mathresponse {
       my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
       my $result;
       my $part=$Apache::inputtags::part;
       my $id=$Apache::inputtags::response[-1];
       if ( $target eq 'grade' && &Apache::response::submitted() ) {
    my $response = &Apache::response::getresponse();
    if ( $response =~ /[^\s]/ && 
        $Apache::response::custom_answer_type eq 'loncapa/perl') {
       if (!$Apache::lonxml::default_homework_loaded) {
    &Apache::lonxml::default_homework_load($safeeval);
       }
       my %previous = &Apache::response::check_for_previous($response,
    $part,$id);
       $Apache::lonhomework::results{"resource.$part.$id.submission"}=
    $response;
       my $error;
       ${$safeeval->varglob('LONCAPA::mathresponse_submission')}=
    $response;
       
       my $award = &Apache::run::run('{ my $submission=$LONCAPA::mathresponse_submission;'.$Apache::response::custom_answer.'}',$safeeval);
       if (!&Apache::inputtags::valid_award($award)) {
    $error = $award;
    $award = 'ERROR';
       }
       &Apache::response::handle_previous(\%previous,$award);
       $Apache::lonhomework::results{"resource.$part.$id.awarddetail"}=
    $award;
       if ($error) {
    $Apache::lonhomework::results{"resource.$part.$id.awardmsg"}=
       $error;
       }
    }
       }
       pop(@Apache::lonxml::namespace);
       &Apache::lonxml::deregister('Apache::response',('answer'));
       &Apache::response::end_response();
       return $result;
   }
   
   
   sub start_answer {
       my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
       my $result;
       $Apache::response::custom_answer=
    &Apache::lonxml::get_all_text_unbalanced("/answer",$parser);
       $Apache::response::custom_answer_type=
    lc(&Apache::lonxml::get_param('type',$parstack,$safeeval));
       $Apache::response::custom_answer_type =~ s/\s+//g;
       if ($target eq "edit" ) {
    $result=&Apache::edit::tag_start($target,$token,'Answer algorithm');
    $result.=&Apache::edit::editfield($token->[1],
     $Apache::response::custom_answer,
     '',80,4);
       } elsif ( $target eq "modified" ) {
    $result=$token->[4].&Apache::edit::modifiedfield('/answer',$parser);
       }
       return $result;
   }
   
   sub end_answer {
       my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
       if ($target eq 'edit' ) {
    return &Apache::edit::end_table();
       }
   }
   
   sub decide_package {
       my ($tagstack)=@_;
       my $package;
       if ($$tagstack[-1] eq 'parameter') {
    $package='part';
       } else {
    my $i=-1;
    while (defined($$tagstack[$i])) {
       if ($$tagstack[$i] =~ /(response|hint)$/) {
    $package=$$tagstack[$i];
    last;
       }
       $i--;
    }
       }
       return $package;
 }  }
   
 sub start_responseparam {  sub start_responseparam {
   my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
   my $result='';      my $result='';
   if ($target eq 'meta') {      if ($target eq 'meta') {
     $result = &meta_parameter_write($token->[2]->{'name'},   $result = &meta_parameter_write($token->[2]->{'name'},
     $token->[2]->{'type'},   $token->[2]->{'type'},
     $token->[2]->{'default'},   $token->[2]->{'default'},
     $token->[2]->{'description'});   $token->[2]->{'description'});
   } elsif ($target eq 'edit') {      } elsif ($target eq 'edit') {
     $result.=&Apache::edit::tag_start($target,$token);   $result.=&Apache::edit::tag_start($target,$token);
     $result.=&Apache::edit::text_arg('Name:','name',$token).   my $optionlist;
       &Apache::edit::text_arg('Type:','type',$token).   my $package=&decide_package($tagstack);
  &Apache::edit::text_arg('Description:','description',$token).   foreach my $key (sort(keys(%Apache::lonnet::packagetab))) {
   &Apache::edit::text_arg('Default:','default',$token).      if ($key =~ /^\Q$package\E&(.*)&display$/) {
     "</td></tr>";   $optionlist.='<option value="'.$1.'">'.
     $result.=&Apache::edit::end_table;      $Apache::lonnet::packagetab{$key}.'</option>';
   } elsif ($target eq 'modified') {      }
     my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,   }
  'name','type','description',   if (defined($optionlist)) {
  'default');      $result.='Use template: <select name="'.
     if ($constructtag) {   &Apache::edit::html_element_name('parameter_package').'">'.
       $result = &Apache::edit::rebuild_tag($token);      '<option value=""></option>'.$optionlist.'</select><br />';
       $result.=&Apache::edit::handle_insert();   }
     }   $result.=&Apache::edit::text_arg('Name:','name',$token).
   } elsif ($target eq 'grade' || $target eq 'answer' || $target eq 'web' ||      &Apache::edit::text_arg('Type:','type',$token).
    $target eq 'tex' || $target eq 'analyze' ) {   &Apache::edit::text_arg('Description:','description',$token).
     if ($ENV{'request.state'} eq 'construct') {      &Apache::edit::text_arg('Default:','default',$token).
       my $name    = &Apache::lonxml::get_param('name',$parstack,$safeeval);   "</td></tr>";
       my $default = &Apache::lonxml::get_param('default',$parstack,$safeeval);   $result.=&Apache::edit::end_table;
       if ($name) {$Apache::inputtags::params{$name}=$default;}      } elsif ($target eq 'modified') {
    my $constructtag=&Apache::edit::get_new_args($token,$parstack,
        $safeeval,'name','type',
        'description','default');
    my $element=&Apache::edit::html_element_name('parameter_package');
    if (defined($env{"form.$element"}) && $env{"form.$element"} ne '') {
       my $name=$env{"form.$element"};
       my $tag=&decide_package($tagstack);
       $token->[2]->{'name'}=$name;
       $token->[2]->{'type'}=
    $Apache::lonnet::packagetab{"$tag&$name&type"};
       $token->[2]->{'description'}=
    $Apache::lonnet::packagetab{"$tag&$name&display"};
       $token->[2]->{'default'}=
    $Apache::lonnet::packagetab{"$tag&$name&default"};
       $constructtag=1;
    }
    if ($constructtag) {
       $result = &Apache::edit::rebuild_tag($token);
       $result.=&Apache::edit::handle_insert();
    }
       } elsif ($target eq 'grade' || $target eq 'answer' || $target eq 'web' ||
        $target eq 'tex' || $target eq 'analyze' ) {
    if ($env{'request.state'} eq 'construct') {
       my $name   =&Apache::lonxml::get_param('name',$parstack,$safeeval);
       my $default=&Apache::lonxml::get_param('default',$parstack,
        $safeeval);
       if ($name) {$Apache::inputtags::params{$name}=$default;}
    }
     }      }
   }      return $result;
   return $result;  
 }  }
   
 sub end_responseparam {  sub end_responseparam {
   my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval)=@_;
   if ($target eq 'edit') { return ('','no'); }      if ($target eq 'edit') { return ('','no'); }
   return '';      return '';
 }  }
   
 sub start_parameter {  sub start_parameter {
   my $result = &start_responseparam(@_);      return &start_responseparam(@_);
   return $result;  
 }  }
   
 sub end_parameter {  sub end_parameter {
   my $result = &end_responseparam(@_);      return &end_responseparam(@_);
   return $result;  }
   
   sub reset_params {
       %Apache::inputtags::params=();
 }  }
   
 sub setup_params {  sub setup_params {
   my ($tag) = @_;      my ($tag,$safeeval) = @_;
   
   if ($ENV{'request.state'} eq 'construct') { return; }      if ($env{'request.state'} eq 'construct') { return; }
   %Apache::inputtags::params=();      my %paramlist=();
   my %paramlist=();      foreach my $key (keys(%Apache::lonnet::packagetab)) {
   foreach my $key (keys(%Apache::lonnet::packagetab)) {   if ($key =~ /^$tag/) {
     if ($key =~ /^$tag/) {      my ($package,$name) = split(/&/,$key);
       my ($package,$name) = split(/&/,$key);      $paramlist{$name}=1;
       $paramlist{$name}=1;   }
     }  
   }  
   foreach my $key (keys(%paramlist)) {  
     my $entry= 'resource.'.$Apache::inputtags::part;  
     if (defined($Apache::inputtags::response[-1])) {  
       $entry.='_'.$Apache::inputtags::response[-1];  
     }      }
     $entry.='.'.$key;      foreach my $key (keys(%paramlist)) {
     &Apache::lonxml::debug("looking for $entry");   my $entry= 'resource.'.$Apache::inputtags::part;
     my $value = &Apache::lonnet::EXT("$entry");   if (defined($Apache::inputtags::response[-1])) {
     &Apache::lonxml::debug("$key has value :$value:");      $entry.='_'.$Apache::inputtags::response[-1];
     if ($value eq 'con_lost' || $value =~ /^error:/) {   }
       &Apache::lonxml::debug("using nothing");   $entry.='.'.$key;
       $Apache::inputtags::params{$key}='';   &Apache::lonxml::debug("looking for $entry");
     } else {   my $value = &Apache::lonnet::EXT("$entry");
       &Apache::lonxml::debug("using value");   &Apache::lonxml::debug("$key has value :$value:");
       $Apache::inputtags::params{$key}=$value;   if ($value eq 'con_lost' || $value =~ /^error:/) {
       &Apache::lonxml::debug("using nothing");
       $Apache::inputtags::params{$key}='';
    } else {
       my $string="{return qq\0".$value."\0}";
       my $newvalue=&Apache::run::run($string,$safeeval,1);
       if (defined($newvalue)) { $value=$newvalue; }
       $Apache::inputtags::params{$key}=$value;
    }
     }      }
   }  
 }  }
   
   {
       my @answer_bits;
   
 sub answer_header {  sub answer_header {
   my ($type) = @_;      my ($type,$increment) = @_;
   my $result;      my $result;
   if ($type eq 'optionresponse' || $type eq 'radiobuttonresponse' ) {      if ($env{'form.answer_output_mode'} eq 'tex') {
     $result = '<table border="1"><tr><th>Answer for Part:'.   undef(@answer_bits);
       $Apache::inputtags::part. '</th></tr><tr>'."\n";   my $bit;
   } else {   if ($Apache::lonhomework::type eq 'exam') {
     $result = '<table border="1"><tr><td>Answer for Part:'.      $bit = ($Apache::lonxml::counter+$increment).') ';
       $Apache::inputtags::part. '</td>'."\n";   } else {
   }      $bit .= ' Answer for Part: \verb|'.
   return $result;   $Apache::inputtags::part.'| ';
    }
    push(@answer_bits,$bit);
       } else {
    $result  = '<table border="1"><tr>';
    if ($Apache::lonhomework::type eq 'exam') {
       $result .= '<td>'.($Apache::lonxml::counter+$increment). ')</td>';
    } else {
       $result .= '<td>Answer for Part:'.$Apache::inputtags::part.'</td>';
    }
    $result .= "\n";
       }
       return $result;
 }  }
   
 sub answer_part {  sub answer_part {
   my ($type,$answer) = @_;      my ($type,$answer) = @_;
   my $result;      my $result;
   if ($type eq 'optionresponse' || $type eq 'radiobuttonresponse') {      if ($env{'form.answer_output_mode'} eq 'tex') {
     $result = '<td>'.$answer.'</td>';   my $to_use='|';
   } else {   foreach my $value (32..126) {
     $result = '<td>'.$answer.'</td>';      my $char=pack('c',$value);
   }      if ($answer !~ /\Q$char\E/) {
   return $result;   $to_use=$char;
    last;
       }
    }
    if ($answer ne '') {
       push(@answer_bits,'\verb'.$to_use.$answer.$to_use);
    }
       } else {
    $result = '<td>'.$answer.'</td>';
       }
       return $result;
 }  }
   
 sub answer_footer {  sub answer_footer {
   my ($type) = @_;      my ($type) = @_;
   my $result;      my $result;
   if ($type eq 'optionresponse' || $type eq 'radiobuttonresponse') {      if ($env{'form.answer_output_mode'} eq 'tex') {
     $result = '</tr></table>';   my $columns = scalar(@answer_bits);
   } else {   $result  = ' \vskip 0 mm \noindent \begin{tabular}{|'.'c|'x$columns.'}\hline ';
     $result = '</tr></table>';   $result .= join(' & ',@answer_bits);
   }   $result .= ' \\\\ \\hline \end{tabular} \vskip 0 mm ';
   return $result;      } else {
    $result = '</tr></table>';
       }
       return $result;
   }
   
 }  }
   
 sub showallfoils {  sub showallfoils {
   my $return=0;      if (defined($env{'form.showallfoils'})) {
   if (defined($ENV{'form.showallfoils'}) &&   my ($symb)=&Apache::lonxml::whichuser();
       $ENV{'request.state'} eq 'construct') {   if (($env{'request.state'} eq 'construct') || 
     $return=1;      ($env{'user.adv'} && $symb eq '')      ||
   }              ($Apache::lonhomework::viewgrades) ) {
   return $return;      return 1;
    }
       }
       if ($Apache::lonhomework::type eq 'survey') { return 1; }
       return 0;
   }
   
   sub getresponse {
       my ($temp,$resulttype)=@_;
       my $formparm='form.HWVAL_'.$Apache::inputtags::response['-1'];
       my $response;
       if (!defined($temp)) {
    $temp=1;
       } else {
    $formparm.=":$temp";
       }
       my %let_to_num=('A'=>0,'B'=>1,'C'=>2,'D'=>3,'E'=>4,'F'=>5,'G'=>6,'H'=>7,
       'I'=>8,'J'=>9,'K'=>10,'L'=>11,'M'=>12,'N'=>13,'O'=>14,
       'P'=>15,'Q'=>16,'R'=>17,'S'=>18,'T'=>19,'U'=>20,'V'=>21,
       'W'=>22,'X'=>23,'Y'=>24,'Z'=>25);
       if ($env{'form.submitted'} eq 'scantron') {
    my $part  = $Apache::inputtags::part;
    my $id    = $Apache::inputtags::response[-1];
    $response = $env{'scantron.'.($Apache::lonxml::counter+$temp-1).
    '.answer'};
    # save bubbled letter for later
    $Apache::lonhomework::results{"resource.$part.$id.scantron"}.=
       $response;
    if ($resulttype ne 'letter') {
       if ($resulttype eq 'A is 1') {
    $response = $let_to_num{$response}+1;
       } else {
    $response = $let_to_num{$response};
       }
    }
       } else {
    $response = $env{$formparm};
       }
       return $response;
   }
   
   sub repetition {
       my $id = $Apache::inputtags::part;
       my $weight = &Apache::lonnet::EXT("resource.$id.weight");
       if (!defined($weight) || ($weight eq '')) { $weight=1; }
       my $repetition = int($weight/10);
       if ($weight % 10 != 0) { $repetition++; } 
       return $repetition;
   }
   
   sub scored_response {
       my ($part,$id)=@_;
       my $repetition=&repetition();
       my $score=0;
       for (my $i=0;$i<$repetition;$i++) {
    # A is 1, B is 2, etc. (get response return 0-9 and then we add 1)
    my $increase=&Apache::response::getresponse($i+1);
    if ($increase ne '') { $score+=$increase+1; }
       }
       my $weight = &Apache::lonnet::EXT("resource.$part.weight");
       if (!defined($weight) || $weight eq '' || $weight eq 0) { $weight = 1; }
       my $pcr=$score/$weight;
       $Apache::lonhomework::results{"resource.$part.$id.awarded"}=$pcr;
       $Apache::lonhomework::results{"resource.$part.$id.awarddetail"}=
    'ASSIGNED_SCORE';
       return $repetition;
   }
   
   sub whichorder {
       my ($max,$randomize,$showall,$hash)=@_;
       #&Apache::lonxml::debug("man $max randomize $randomize");
       if (!defined(@{ $$hash{'names'} })) { return; }
       my @names = @{ $$hash{'names'} };
       my @whichopt =();
       my (%top,@toplist,%bottom,@bottomlist);
       if (!($showall || ($randomize eq 'no'))) {
    my $current=0;
    foreach my $name (@names) {
       $current++;
       if ($$hash{"$name.location"} eq 'top') {
    $top{$name}=$current;
       } elsif ($$hash{"$name.location"} eq 'bottom') {
    $bottom{$name}=$current;
       }
    }
       }
       my $topcount=0;
       my $bottomcount=0;
       while (((scalar(@whichopt)+$topcount+$bottomcount) < $max || $showall)
      && ($#names > -1)) {
    #&Apache::lonxml::debug("Have $#whichopt max is $max");
    my $aopt;
    if ($showall || ($randomize eq 'no')) {
       $aopt=0;
    } else {
       $aopt=int(&Math::Random::random_uniform() * ($#names+1));
    }
    #&Apache::lonxml::debug("From $#whichopt $max $#names elms, picking $aopt");
    $aopt=splice(@names,$aopt,1);
    #&Apache::lonxml::debug("Picked $aopt");
    if ($top{$aopt}) {
       $toplist[$top{$aopt}]=$aopt;
       $topcount++;
    } elsif ($bottom{$aopt}) {
       $bottomlist[$bottom{$aopt}]=$aopt;
       $bottomcount++;
    } else {
       push (@whichopt,$aopt);
    }
       }
       for (my $i=0;$i<=$#toplist;$i++) {
    if ($toplist[$i]) { unshift(@whichopt,$toplist[$i]) }
       }
       for (my $i=0;$i<=$#bottomlist;$i++) {
    if ($bottomlist[$i]) { push(@whichopt,$bottomlist[$i]) }
       }
       return @whichopt;
   }
   
   sub show_answer {
       my $part   = $Apache::inputtags::part;
       my $award  = $Apache::lonhomework::history{"resource.$part.solved"};
       my $status = $Apache::inputtags::status[-1];
       return  ( ($award =~ /^correct/
          && lc($Apache::lonhomework::problemstatus) ne 'no')
         || $status eq "SHOW_ANSWER");
   }
   
   sub analyze_store_foilgroup {
       my ($shown,$attrs)=@_;
       my $part_id="$Apache::inputtags::part.$Apache::inputtags::response[-1]";
       foreach my $name (@{ $Apache::response::foilgroup{'names'} }) {
    if (defined($Apache::lonhomework::analyze{"$part_id.foil.value.$name"})) { next; }
    push (@{ $Apache::lonhomework::analyze{"$part_id.foils"} },$name);
    foreach my $attr (@$attrs) {
       $Apache::lonhomework::analyze{"$part_id.foil.".$attr.".$name"} =
    $Apache::response::foilgroup{"$name.".$attr};
    }
       }
       push (@{ $Apache::lonhomework::analyze{"$part_id.shown"} }, @{ $shown });
 }  }
   
   sub check_if_computed {
       my ($token,$parstack,$safeeval,$name)=@_;
       my $value = &Apache::lonxml::get_param($name,$parstack,$safeeval);
       if (ref($token->[2]) eq 'HASH' && $value ne $token->[2]{$name}) {
    my $part_id="$Apache::inputtags::part.$Apache::inputtags::response[-1]";
    $Apache::lonhomework::analyze{"$part_id.answercomputed"} = 1;
       }
   }
   
   sub pick_foil_for_concept {
       my ($target,$attrs,$hinthash,$parstack,$safeeval)=@_;
       if (not defined(@{ $Apache::response::conceptgroup{'names'} })) { return; }
       my @names = @{ $Apache::response::conceptgroup{'names'} };
       my $pick=int(&Math::Random::random_uniform() * ($#names+1));
       my $name=$names[$pick];
       push @{ $Apache::response::foilgroup{'names'} }, $name;
       foreach my $attr (@$attrs) {
    $Apache::response::foilgroup{"$name.".$attr} =
       $Apache::response::conceptgroup{"$name.".$attr};
       }
       my $concept = &Apache::lonxml::get_param('concept',$parstack,$safeeval);
       $Apache::response::foilgroup{"$name.concept"} = $concept;
       &Apache::lonxml::debug("Selecting $name in $concept");
       my $part_id="$Apache::inputtags::part.$Apache::inputtags::response[-1]";
       if ($target eq 'analyze') {
    push (@{ $Apache::lonhomework::analyze{"$part_id.concepts"} },
         $concept);
    $Apache::lonhomework::analyze{"$part_id.concept.$concept"}=
       $Apache::response::conceptgroup{'names'};
    foreach my $name (@{ $Apache::response::conceptgroup{'names'} }) {
       push (@{ $Apache::lonhomework::analyze{"$part_id.foils"} },
     $name);
       foreach my $attr (@$attrs) {
    $Apache::lonhomework::analyze{"$part_id.foil.$attr.$name"}=
       $Apache::response::conceptgroup{"$name.$attr"};
       }
    }
       }
       push(@{ $hinthash->{"$part_id.concepts"} },$concept);
       $hinthash->{"$part_id.concept.$concept"}=
    $Apache::response::conceptgroup{'names'};
   
   }
   
   sub get_response_param {
       my ($id,$name,$default)=@_;
       my $parameter;
       if ($env{'request.state'} eq 'construct' &&
    defined($Apache::inputtags::params{$name})) {
    $parameter=$Apache::inputtags::params{$name};
       } else {
    $parameter=&Apache::lonnet::EXT("resource.$id.$name");
       }
       if (!defined($parameter) || $parameter eq '') {
    $parameter = $default;
       }
       return $parameter;
   }
   
   sub submitted {
       my ($who)=@_;
       
       # when scatron grading any submission is a submission
       if ($env{'form.submitted'} eq 'scantron') { return 1; }
       # if the caller only cared if this was a scantron submission
       if ($who eq 'scantron') { return 0; }
       # if the Submit Answer button for this particular part was pressed
       my $partid=$Apache::inputtags::part;
       if (defined($env{'form.submit_'.$partid})) { return 1; }
       # Submit All button on a .page was pressed
       if (defined($env{'form.all_submit'})) { return 1; }
       # otherwise no submission occured
       return 0;
   }
   
   sub add_to_gradingqueue {
       my ($symb,$courseid,$domain,$name) = &Apache::lonxml::whichuser();
       if (   $courseid eq ''
    || $symb eq ''
    || $env{'request.state'} eq 'construct'
    || $Apache::lonhomework::type ne 'problem') {
    return;
       }
   
       my %queue_info = ( 'type' => 'problem',
          'time' => time);
   
       if (exists($Apache::lonhomework::history{"resource.0.checkedin.slot"})) {
    $queue_info{'slot'}=
        $Apache::lonhomework::history{"resource.0.checkedin.slot"};
       }
   
       my $result=&Apache::bridgetask::add_to_queue('gradingqueue',\%queue_info);
       if ($result ne 'ok') {
    &Apache::lonxml::error("add_to_queue said $result");
       }
   }
   
   # basically undef and 0 (both false) mean that they still have work to do
   # and all true values mean that they can't do any more work
   #
   # a return of undef means it is unattempted
   # a return of 0 means it is attmpted and wrong but still has tries
   # a return of 1 means it is marked correct
   # a return of 2 means they have exceed maximum number of tries
   # a return of 3 means it after the answer date
   sub check_status {
       my ($id)=@_;
       if (!$id) { $id=$Apache::linputtags::part; }
       my $curtime=&Apache::lonnet::EXT('system.time');
       my $opendate=&Apache::lonnet::EXT("resource.$id.opendate");
       my $duedate=&Apache::lonnet::EXT("resource.$id.duedate");
       my $answerdate=&Apache::lonnet::EXT("resource.$id.answerdate");
       if ( $opendate && $curtime > $opendate &&
            $duedate && $curtime > $duedate &&
            $answerdate && $curtime > $answerdate) {
           return 3;
       }
       my $status=&Apache::lonnet::EXT("user.resource.resource.$id.solved");
       if ($status =~ /^correct/) { return 1; }
       if (!$status) { return undef; }
       my $maxtries=&Apache::lonnet::EXT("resource.$id.maxtries");
       if ($maxtries eq '') { $maxtries=2; }
       my $curtries=&Apache::lonnet::EXT("user.resource.resource.$id.tries");
       if ($curtries < $maxtries) { return 0; }
       return 2;
   }
   
 1;  1;
 __END__  __END__
     

Removed from v.1.66  
changed lines
  Added in v.1.140


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>
500 Internal Server Error

Internal Server Error

The server encountered an internal error or misconfiguration and was unable to complete your request.

Please contact the server administrator at root@localhost to inform them of the time this error occurred, and the actions you performed just before this error.

More information about this error may be available in the server error log.