Diff for /loncom/homework/lonhomework.pm between versions 1.362 and 1.378

version 1.362, 2016/05/30 02:45:32 version 1.378, 2023/07/23 16:11:52
Line 50  use Apache::chemresponse(); Line 50  use Apache::chemresponse();
 use Apache::functionplotresponse();  use Apache::functionplotresponse();
 use Apache::drawimage();  use Apache::drawimage();
 use Apache::loncapamath();  use Apache::loncapamath();
   use Apache::loncourseuser();
   use Apache::grades();
 use Apache::Constants qw(:common);  use Apache::Constants qw(:common);
 use Apache::loncommon();  use Apache::loncommon();
 use Apache::lonparmset();  use Apache::lonparmset();
   use Apache::lonnavmaps();
 use Apache::lonlocal;  use Apache::lonlocal;
   use LONCAPA qw(:DEFAULT :match);
   use LONCAPA::ltiutils();
 use Time::HiRes qw( gettimeofday tv_interval );  use Time::HiRes qw( gettimeofday tv_interval );
 use HTML::Entities();  use HTML::Entities();
 use File::Copy();  use File::Copy();
   
 # FIXME - improve commenting  # FIXME - improve commenting
   
   my $registered_cleanup;
   
 BEGIN {  BEGIN {
     &Apache::lonxml::register_insert();      &Apache::lonxml::register_insert();
Line 190  sub proctor_checked_in { Line 196  sub proctor_checked_in {
     if ($type eq 'Task') {      if ($type eq 'Task') {
  my $version=$Apache::lonhomework::history{'resource.0.version'};   my $version=$Apache::lonhomework::history{'resource.0.version'};
  $key ="resource.$version.0.checkedin";   $key ="resource.$version.0.checkedin";
     } elsif ($type eq 'problem') {      } elsif (($type eq 'problem') || ($type eq 'tool')) {
  $key ='resource.0.checkedin';   $key ='resource.0.checkedin';
     }      }
     # backward compatability, used to be username@domain,       # backward compatability, used to be username@domain, 
Line 205  sub proctor_checked_in { Line 211  sub proctor_checked_in {
     return 1;      return 1;
  }   }
     }      }
       
     return 0;      return 0;
 }  }
   
 sub check_slot_access {  sub check_slot_access {
     my ($id,$type,$symb)=@_;      my ($id,$type,$symb,$partlist)=@_;
   
     # does it pass normal muster      # does it pass normal muster
     my ($status,$datemsg)=&check_access($id,$symb);      my ($status,$datemsg)=&check_access($id,$symb);
       
     my $useslots = &Apache::lonnet::EXT("resource.0.useslots",$symb);      my $useslots = &Apache::lonnet::EXT("resource.0.useslots",$symb);
     if ($useslots ne 'resource' && $useslots ne 'map'       if ($useslots ne 'resource' && $useslots ne 'map' 
  && $useslots ne 'map_map') {   && $useslots ne 'map_map') {
Line 228  sub check_slot_access { Line 233  sub check_slot_access {
         $checkin = "resource.$version.0.checkedin";          $checkin = "resource.$version.0.checkedin";
     }      }
     my $checkedin = $Apache::lonhomework::history{$checkin};      my $checkedin = $Apache::lonhomework::history{$checkin};
     my ($returned_slot,$slot_name,$checkinslot,$ipused,$blockip,$now,$ip);      my ($returned_slot,$slot_name,$checkinslot,$ipused,$blockip,$now,$ip,
           $consumed_uniq);
     $now = time;      $now = time;
     $ip=$env{'request.host'} || $ENV{'REMOTE_ADDR'};      $ip=$ENV{'REMOTE_ADDR'} || $env{'request.host'};
   
     if ($checkedin) {      if ($checkedin) {
         $checkinslot = $Apache::lonhomework::history{"$checkin.slot"};          $checkinslot = $Apache::lonhomework::history{"$checkin.slot"};
         my %slot=&Apache::lonnet::get_slot($checkinslot);          my %slot=&Apache::lonnet::get_slot($checkinslot);
           $consumed_uniq = $slot{'uniqueperiod'};
         if ($slot{'iptied'}) {          if ($slot{'iptied'}) {
             $ipused = $Apache::lonhomework::history{"$checkin.ip"};              $ipused = $Apache::lonhomework::history{"$checkin.ip"};
             unless (($ip ne '') && ($ipused eq $ip)) {              unless (($ip ne '') && 
                       (($ipused eq $ip) || ($ENV{'REMOTE_ADDR'} eq '127.0.0.1'))) {
                 $blockip = $slot{'iptied'};                  $blockip = $slot{'iptied'};
                 $slot_name = $checkinslot;                  $slot_name = $checkinslot;
                 $returned_slot = \%slot;                  $returned_slot = \%slot;
Line 251  sub check_slot_access { Line 259  sub check_slot_access {
         } else {          } else {
             return ($status,$datemsg);              return ($status,$datemsg);
         }          }
     }       }
   
     if ($status eq 'CLOSED' ||      if ($status eq 'CLOSED' ||
  $status eq 'INVALID_ACCESS' ||   $status eq 'INVALID_ACCESS' ||
Line 261  sub check_slot_access { Line 269  sub check_slot_access {
     if ($env{'request.state'} eq "construct") {      if ($env{'request.state'} eq "construct") {
  return ($status,$datemsg);   return ($status,$datemsg);
     }      }
       
     if ($type eq 'Task') {      if ($type eq 'Task') {
  if ($checkedin &&   if ($checkedin &&
     $Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass') {      $Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass') {
Line 271  sub check_slot_access { Line 279  sub check_slot_access {
         return ('SHOW_ANSWER');          return ('SHOW_ANSWER');
             }              }
         }          }
       } elsif (($type eq 'problem') &&
                ($Apache::lonhomework::browse eq 'F') &&
                ($ENV{'REMOTE_ADDR'} eq '127.0.0.1') &&
                ($env{'form.grade_courseid'} eq $env{'request.course.id'}) &&
                (&Apache::lonnet::allowed('mgr',$env{'request.course.id'}))) {
           return ($status,$datemsg);
     }      }
   
     my $availablestudent = &Apache::lonnet::EXT("resource.0.availablestudent",$symb);      my $availablestudent = &Apache::lonnet::EXT("resource.0.availablestudent",$symb);
Line 348  sub check_slot_access { Line 362  sub check_slot_access {
  $is_correct =     $is_correct =  
     ($Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass'      ($Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass'
      || $Apache::lonhomework::history{"resource.0.solved"} =~ /^correct_/ );       || $Apache::lonhomework::history{"resource.0.solved"} =~ /^correct_/ );
     } elsif ($type eq 'problem') {      } elsif (($type eq 'problem') || ($type eq 'tool')) {
  $got_grade  = 1;          if ((ref($partlist) eq 'ARRAY') && (@{$partlist} > 0)) {
  $is_correct =              my ($numcorrect,$numgraded) = (0,0);
     ($Apache::lonhomework::history{"resource.0.solved"} =~/^correct_/);              foreach my $part (@{$partlist}) {
                   my $currtries = $Apache::lonhomework::history{"resource.$part.tries"};
                   my $maxtries = &Apache::lonnet::EXT("resource.$part.maxtries",$symb);
                   my $probstatus = &Apache::structuretags::get_problem_status($part);
                   my $earlyout;
                   unless (($probstatus eq 'no') ||
                           ($probstatus eq 'no_feedback_ever')) {
                       if ($Apache::lonhomework::history{"resource.$part.solved"} =~/^correct_/) {
                           $numcorrect ++;
                       } else {
                           $earlyout = 1;
                       }
                   }
                   if ($currtries == $maxtries) {
                       $earlyout = 1;
                   } else {
                       $numgraded ++;
                   }
                   last if ($earlyout);
               }
               my $numparts = scalar(@{$partlist});
               if ($numparts == $numcorrect) {
                   $is_correct = 1;
               }
               if ($numparts == $numgraded) {
                   $got_grade = 1;
               }
           } else {
               my $currtries = $Apache::lonhomework::history{"resource.0.tries"};
               my $maxtries = &Apache::lonnet::EXT("resource.0.maxtries",$symb);
               my $probstatus = &Apache::structuretags::get_problem_status('0');
               unless (($probstatus eq 'no') ||
                       ($probstatus eq 'no_feedback_ever')) {
                   $is_correct =
                       ($Apache::lonhomework::history{"resource.0.solved"} =~/^correct_/);
               }
               unless (($currtries == $maxtries) || ($is_correct)) {
                   $got_grade = 1;
               }
           }
     }      }
           
     &Apache::lonxml::debug(" slot is $slotstatus checkedin ($checkedin) got_grade ($got_grade) is_correct ($is_correct)");      &Apache::lonxml::debug(" slot is $slotstatus checkedin ($checkedin) got_grade ($got_grade) is_correct ($is_correct)");
Line 369  sub check_slot_access { Line 422  sub check_slot_access {
     # used to gain access to it to work on it, until the due date is reached, and the      # used to gain access to it to work on it, until the due date is reached, and the
     # problem then becomes CLOSED.  Therefore return the slotstatus -       # problem then becomes CLOSED.  Therefore return the slotstatus - 
     # (which will be one of: NOT_IN_A_SLOT, RESERVABLE, RESERVABLE_LATER, or NOTRESERVABLE).      # (which will be one of: NOT_IN_A_SLOT, RESERVABLE, RESERVABLE_LATER, or NOTRESERVABLE).
     if (!defined($slot_name) && $type eq 'problem') {  
       if (!defined($slot_name) && (($type eq 'problem') || ($type eq 'tool'))) {
         if ($slotstatus eq 'NOT_IN_A_SLOT') {          if ($slotstatus eq 'NOT_IN_A_SLOT') {
             if (!$num_usable_slots) {              if (!$num_usable_slots) {
                 if ($env{'request.course.id'}) {                  ($slotstatus,$datemsg) = &check_reservable_slot($slotstatus,$symb,$now,$checkedin,
                     my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};                                                                  $consumed_uniq);
                     my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};  
                     unless ($symb) {  
                         ($symb)=&Apache::lonnet::whichuser();  
                     }  
                     $slotstatus = 'NOTRESERVABLE';  
                     my ($reservable_now_order,$reservable_now,$reservable_future_order,  
                         $reservable_future) =   
                         &Apache::loncommon::get_future_slots($cnum,$cdom,$now,$symb);  
                     if ((ref($reservable_now_order) eq 'ARRAY') && (ref($reservable_now) eq 'HASH')) {  
                         if (@{$reservable_now_order} > 0) {  
                             $slotstatus = 'RESERVABLE';  
                             $datemsg = $reservable_now->{$reservable_now_order->[-1]}{'endreserve'};  
                         }  
                     }  
                     unless ($slotstatus eq 'RESERVABLE') {  
                         if ((ref($reservable_future_order) eq 'ARRAY') && (ref($reservable_future) eq 'HASH')) {  
                             if (@{$reservable_future_order} > 0) {  
                                 $slotstatus = 'RESERVABLE_LATER';  
                                 $datemsg = $reservable_future->{$reservable_future_order->[0]}{'startreserve'};  
                             }  
                         }  
                     }  
                 }  
             }              }
         }          }
         return ($slotstatus,$datemsg);          return ($slotstatus,$datemsg);
Line 418  sub check_slot_access { Line 449  sub check_slot_access {
     }      }
   
     if (($is_correct) && ($blockip ne 'answer')) {      if (($is_correct) && ($blockip ne 'answer')) {
  if ($type eq 'problem') {   if (($type eq 'problem') || ($type eq 'tool')) {
     return ($status);      return ($status);
  }   }
  return ('SHOW_ANSWER');   return ('SHOW_ANSWER');
Line 432  sub check_slot_access { Line 463  sub check_slot_access {
     return ($slotstatus,$datemsg,$slot_name,$returned_slot,$ipused);      return ($slotstatus,$datemsg,$slot_name,$returned_slot,$ipused);
 }  }
   
   sub check_reservable_slot {
       my ($slotstatus,$symb,$now,$checkedin,$consumed_uniq) = @_;
       my $datemsg;
       if ($slotstatus eq 'NOT_IN_A_SLOT') {
           if ($env{'request.course.id'}) {
               my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
               my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
               unless ($symb) {
                   ($symb)=&Apache::lonnet::whichuser();
               }
               $slotstatus = 'NOTRESERVABLE';
               my ($reservable_now_order,$reservable_now,$reservable_future_order,
                   $reservable_future) =
                   &Apache::loncommon::get_future_slots($cnum,$cdom,$now,$symb);
               if ((ref($reservable_now_order) eq 'ARRAY') && (ref($reservable_now) eq 'HASH')) {
                   if (@{$reservable_now_order} > 0) {
                       if ((!$checkedin) || (ref($consumed_uniq) ne 'ARRAY')) {
                           $slotstatus = 'RESERVABLE';
                           $datemsg = $reservable_now->{$reservable_now_order->[-1]}{'endreserve'};
                       } else {
                           my ($uniqstart,$uniqend,$useslot);
                           if (ref($consumed_uniq) eq 'ARRAY') {
                               ($uniqstart,$uniqend)=@{$consumed_uniq};
                           }
                           foreach my $slot (reverse(@{$reservable_now_order})) {
                               if ($reservable_now->{$slot}{'uniqueperiod'} =~ /^(\d+)\,(\d+)$/) {
                                   my ($new_uniq_start,$new_uniq_end) = ($1,$2);
                                   next if (!
                                       ($uniqstart < $new_uniq_start && $uniqend < $new_uniq_start) ||
                                       ($uniqstart > $new_uniq_end   &&  $uniqend > $new_uniq_end  ));
                               }
                               $useslot = $slot;
                               last;
                           }
                           if ($useslot) {
                               $slotstatus = 'RESERVABLE';
                               $datemsg = $reservable_now->{$useslot}{'endreserve'};
                           }
                       }
                   }
               }
               unless ($slotstatus eq 'RESERVABLE') {
                   if ((ref($reservable_future_order) eq 'ARRAY') && (ref($reservable_future) eq 'HASH')) {
                       if (@{$reservable_future_order} > 0) {
                           if ((!$checkedin) || (ref($consumed_uniq) ne 'ARRAY')) {
                               $slotstatus = 'RESERVABLE_LATER';
                               $datemsg = $reservable_future->{$reservable_future_order->[0]}{'startreserve'};
                           } else {
                               my ($uniqstart,$uniqend,$useslot);
                               if (ref($consumed_uniq) eq 'ARRAY') {
                                   ($uniqstart,$uniqend)=@{$consumed_uniq};
                               }
                               foreach my $slot (@{$reservable_future_order}) {
                                   if ($reservable_future->{$slot}{'uniqueperiod'} =~ /^(\d+),(\d+)$/) {
                                       my ($new_uniq_start,$new_uniq_end) = ($1,$2);
                                       next if (!
                                         ($uniqstart < $new_uniq_start && $uniqend < $new_uniq_start) ||
                                         ($uniqstart > $new_uniq_end   &&  $uniqend > $new_uniq_end  ));
                                   }
                                   $useslot = $slot;
                                   last;
                               }
                               if ($useslot) {
                                   $slotstatus = 'RESERVABLE_LATER';
                                   $datemsg = $reservable_future->{$useslot}{'startreserve'};
                               }
                           }
                       }
                   }
               }
           }
       }
       return ($slotstatus,$datemsg);
   }
   
 # JB, 9/24/2002: Any changes in this function may require a change  # JB, 9/24/2002: Any changes in this function may require a change
 # in lonnavmaps::resource::getDateStatus.  # in lonnavmaps::resource::getDateStatus.
 sub check_access {  sub check_access {
Line 691  sub setuppermissions { Line 797  sub setuppermissions {
  $env{'request.course.sec'} !~ /^\s*$/) {   $env{'request.course.sec'} !~ /^\s*$/) {
  $viewgrades = &Apache::lonnet::allowed('vgr',$env{'request.course.id'}.   $viewgrades = &Apache::lonnet::allowed('vgr',$env{'request.course.id'}.
                                                '/'.$env{'request.course.sec'});                                                 '/'.$env{'request.course.sec'});
           if ($viewgrades) {
               $Apache::lonhomework::viewgradessec = $env{'request.course.sec'};
           }
     }      }
     $Apache::lonhomework::viewgrades = $viewgrades;      $Apache::lonhomework::viewgrades = $viewgrades;
   
Line 708  sub setuppermissions { Line 817  sub setuppermissions {
  $modifygrades =    $modifygrades = 
     &Apache::lonnet::allowed('mgr',$env{'request.course.id'}.      &Apache::lonnet::allowed('mgr',$env{'request.course.id'}.
      '/'.$env{'request.course.sec'});       '/'.$env{'request.course.sec'});
           if ($modifygrades) {
               $Apache::lonhomework::modifygradessec = $env{'request.course.sec'};
           }
     }      }
     $Apache::lonhomework::modifygrades = $modifygrades;      $Apache::lonhomework::modifygrades = $modifygrades;
   
Line 726  sub setuppermissions { Line 838  sub setuppermissions {
 sub unset_permissions {  sub unset_permissions {
     undef($Apache::lonhomework::queuegrade);      undef($Apache::lonhomework::queuegrade);
     undef($Apache::lonhomework::modifygrades);      undef($Apache::lonhomework::modifygrades);
       undef($Apache::lonhomework::modifygradessec);
     undef($Apache::lonhomework::viewgrades);      undef($Apache::lonhomework::viewgrades);
       undef($Apache::lonhomework::viewgradessec);
     undef($Apache::lonhomework::browse);      undef($Apache::lonhomework::browse);
 }  }
   
Line 828  sub analyze_header { Line 942  sub analyze_header {
     my $js = &Apache::structuretags::setmode_javascript();      my $js = &Apache::structuretags::setmode_javascript();
   
     # Breadcrumbs      # Breadcrumbs
     my $brcrum = [{'href' => &Apache::loncommon::authorspace($request->uri),      my $text = 'Authoring Space';
                    'text' => 'Authoring Space'},      my $href = &Apache::loncommon::authorspace($request->uri);
       if ($env{'request.course.id'}) {
           my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
           my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
           if ($href eq "/priv/$cdom/$cnum/") {
               $text = 'Course Authoring Space';
           }
       }
       my $brcrum = [{'href' => $href,
                      'text' => $text},
                   {'href' => '',                    {'href' => '',
                    'text' => 'Problem Testing'},                     'text' => 'Problem Testing'},
                   {'href' => '',                    {'href' => '',
Line 1061  sub editxmlmode { Line 1184  sub editxmlmode {
             &Apache::lonhtmlcommon::dragmath_js("EditMathPopup");              &Apache::lonhtmlcommon::dragmath_js("EditMathPopup");
   
     # Breadcrumbs      # Breadcrumbs
     my $brcrum = [{'href' => &Apache::loncommon::authorspace($request->uri),      my $text = 'Authoring Space';
                    'text' => 'Authoring Space'},      my $href = &Apache::loncommon::authorspace($request->uri);
       if ($env{'request.course.id'}) {
           my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
           my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
           if ($href eq "/priv/$cdom/$cnum/") {
               $text = 'Course Authoring Space';
           }
       }
       my $brcrum = [{'href' => $href,
                      'text' => $text},
                   {'href' => '',                    {'href' => '',
                    'text' => 'Problem Editing'}];                     'text' => 'Problem Editing'}];
   
Line 1097  sub editxmlmode { Line 1229  sub editxmlmode {
   
     unless ($env{'environment.nocodemirror'}) {      unless ($env{'environment.nocodemirror'}) {
         # dropdown menus          # dropdown menus
         $result .= Apache::lonmenu::create_submenu("#", "",           $result .= Apache::lonmenu::create_submenu("#", "",
             &mt("Problem Templates"), template_dropdown_datastructure());              &mt("Problem Templates"), template_dropdown_datastructure());
   
         $result .= Apache::lonmenu::create_submenu("#", "",           $result .= Apache::lonmenu::create_submenu("#", "",
             &mt("Response Types"), responseblock_dropdown_datastructure());              &mt("Response Types"), responseblock_dropdown_datastructure());
   
         $result .= Apache::lonmenu::create_submenu("#", "",           $result .= Apache::lonmenu::create_submenu("#", "",
             &mt("Conditional Blocks"), conditional_scripting_datastructure());              &mt("Conditional Blocks"), conditional_scripting_datastructure());
   
         $result .= Apache::lonmenu::create_submenu("#", "",           $result .= Apache::lonmenu::create_submenu("#", "",
             &mt("Miscellaneous"), misc_datastructure());              &mt("Miscellaneous"), misc_datastructure());
     }      }
   
     $result .= Apache::lonmenu::create_submenu("#", "",       $result .= Apache::lonmenu::create_submenu("#", "",
         &mt("Help") . ' <img src="/adm/help/help.png" alt="' . &mt("Help") .          &mt("Help") . ' <img src="/adm/help/help.png" alt="' . &mt("Help") .
         '" style="vertical-align:text-bottom; height: auto; margin:0; "/>',           '" style="vertical-align:text-bottom; height: auto; margin:0; "/>',
         helpmenu_datastructure(),"");          helpmenu_datastructure(),"");
   
     $result.="</ol></div>";      $result.="</ol></div>";
            
     $result .= '</div></div>' .       $result .= '</div></div>' .
         &Apache::lonxml::message_location() .          &Apache::lonxml::message_location() .
         &Apache::loncommon::xmleditor_js() .          &Apache::loncommon::xmleditor_js() .
         '<textarea ' . &Apache::edit::element_change_detection() .          '<textarea ' . &Apache::edit::element_change_detection() .
Line 1153  sub editxmlmode { Line 1285  sub editxmlmode {
                 autoCloseBrackets: true,                  autoCloseBrackets: true,
                 height: "auto",                  height: "auto",
                 styleActiveLine: true,                  styleActiveLine: true,
                   
                 extraKeys: {                  extraKeys: {
                     "Tab": "indentMore",                      "Tab": "indentMore",
                     "Shift-Tab": "indentLess",                      "Shift-Tab": "indentLess",
Line 1174  sub editxmlmode { Line 1306  sub editxmlmode {
 #    Render the page in whatever target desired.  #    Render the page in whatever target desired.
 #  #
 sub renderpage {  sub renderpage {
     my ($request,$file,$targets,$return_string,$donebuttonmsg) = @_;      my ($request,$file,$targets,$return_string,$donebuttonmsg,$viewasuser,$symb) = @_;
   
     my @targets = @{$targets || [&get_target()]};      my @targets = @{$targets || [&get_target()]};
     &Apache::lonhomework::showhashsubset(\%env,'form.');      &Apache::lonhomework::showhashsubset(\%env,'form.');
Line 1215  sub renderpage { Line 1347  sub renderpage {
  if ($target eq 'answer') { &showhash(%Apache::lonhomework::history); }   if ($target eq 'answer') { &showhash(%Apache::lonhomework::history); }
  if ($target eq 'web') {&Apache::lonhomework::showhashsubset(\%env,'^form');}   if ($target eq 'web') {&Apache::lonhomework::showhashsubset(\%env,'^form');}
   
           if (($target eq 'web') && ($viewasuser ne '') && ($symb ne '')) {
               $env{'request.user_in_effect'} = $viewasuser;
           }
   
  &Apache::lonxml::debug("Should be parsing now");   &Apache::lonxml::debug("Should be parsing now");
  $result .= &Apache::lonxml::xmlparse($request, $target, $problem,   $result .= &Apache::lonxml::xmlparse($request, $target, $problem,
      &setup_vars($target),%mystyle);       &setup_vars($target),%mystyle);
Line 1240  sub renderpage { Line 1376  sub renderpage {
     #}      #}
 #    $request->print($result);  #    $request->print($result);
     $overall_result.=$result;      $overall_result.=$result;
               if (($target eq 'web') && ($viewasuser ne '') && ($symb ne '')) {
                   my ($vuname,$vudom) = split(/:/,$viewasuser);
                   $overall_result .= &Apache::grades::view_as_user($symb,$vuname,$vudom).
                                      '</body></html>';
               }
 #    $request->rflush();  #    $request->rflush();
  }   }
           if (($target eq 'web') && ($viewasuser ne '') && ($symb ne '')) {
               undef($env{'request.user_in_effect'});
           }
  #$request->print(":Result ends");   #$request->print(":Result ends");
  #my $td=&tv_interval($t0);   #my $td=&tv_interval($t0);
     }      }
     if (!$return_string) {      if (!$return_string) {
  &Apache::lonxml::add_messages(\$overall_result);   &Apache::lonxml::add_messages(\$overall_result);
  $request->print($overall_result);      $request->print($overall_result);   
  $request->rflush();      $request->rflush();
     } else {      } else {
  return $overall_result;   return $overall_result;
     }      }
Line 1267  sub finished_parsing { Line 1411  sub finished_parsing {
 # value 3: name of help topic ???  # value 3: name of help topic ???
 sub get_template_list{  sub get_template_list{
     my ($extension) = @_;      my ($extension) = @_;
       
     my @files = glob($Apache::lonnet::perlvar{'lonIncludes'}.      my @files = glob($Apache::lonnet::perlvar{'lonIncludes'}.
                      '/templates/*.'.$extension);                       '/templates/*.'.$extension);
     @files = map {[$_,&mt(&Apache::lonnet::metadata($_, 'title')),      @files = map {[$_,&mt(&Apache::lonnet::metadata($_, 'title')),
Line 1348  sub get_template_html { Line 1492  sub get_template_html {
 sub newproblem {  sub newproblem {
     my ($request) = @_;      my ($request) = @_;
   
  if ($env{'form.mode'} eq 'blank'){      if ($env{'form.mode'} eq 'blank'){
         my $dest = &Apache::lonnet::filelocation("",$request->uri);          my $dest = &Apache::lonnet::filelocation("",$request->uri);
         &File::Copy::copy('/home/httpd/html/res/adm/includes/templates/blank.problem',$dest);          my $templatefilename =
               $request->dir_config('lonIncludes').'/templates/blank.problem';
           &File::Copy::copy($templatefilename,$dest);
         &renderpage($request,$dest);          &renderpage($request,$dest);
         return;          return;
     }      }
       my $errormsg;
     if ($env{'form.template'}) {      if ($env{'form.template'}) {
  my $file = $env{'form.template'};          my $file;
  my $dest = &Apache::lonnet::filelocation("",$request->uri);          my ($extension) = ($env{'form.template'} =~ /\.(\w+)$/);
  &File::Copy::copy($file,$dest);          if ($extension) {
  &renderpage($request,$dest);              my @files = &get_template_list($extension);
  return;              foreach my $poss (@files) {
                   if (ref($poss) eq 'ARRAY') {
                       if ($env{'form.template'} eq $poss->[0]) {
                           $file = $env{'form.template'};
                           last;
                       }
                   }
               }
               if ($file) {
           my $dest = &Apache::lonnet::filelocation("",$request->uri);
           &File::Copy::copy($file,$dest);
           &renderpage($request,$dest);
           return;
               } else {
                   $errormsg = '<p class="LC_error">'.&mt('Invalid template file.').'</p>';
               }
           } else {
               $errormsg = '<p class="LC_error">'.&mt('Invalid template file; template needs to be a .problem, .library, or .task file.').'</p>';
           }
     }      }
   
     my ($extension) = ($request->uri =~ m/\.(\w+)$/);      my ($extension) = ($request->uri =~ m/\.(\w+)$/);
Line 1376  sub newproblem { Line 1541  sub newproblem {
     } else {      } else {
  my $url=&HTML::Entities::encode($request->uri,'<>&"');   my $url=&HTML::Entities::encode($request->uri,'<>&"');
  my $dest = &Apache::lonnet::filelocation("",$request->uri);   my $dest = &Apache::lonnet::filelocation("",$request->uri);
  my $errormsg;  
  my $instructions;   my $instructions;
         my $brcrum = [{'href' => &Apache::loncommon::authorspace($request->uri),          my $text = 'Authoring Space';
                        'text' => 'Authoring Space'},          my $href = &Apache::loncommon::authorspace($request->uri);
           if ($env{'request.course.id'}) {
               my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
               my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
               if ($href eq "/priv/$cdom/$cnum/") {
                   $text = 'Course Authoring Space';
               }
           }
           my $brcrum = [{'href' => $href,
                          'text' => $text},
                       {'href' => '',                        {'href' => '',
                        'text' => "Create New $extension"}];                         'text' => "Create New $extension"}];
  my $start_page =    my $start_page = 
Line 1427  sub update_construct_style { Line 1600  sub update_construct_style {
 }  }
   
 #  #
 # Sets interval for current user so time left will be zero, either for the entire folder   # Sets interval for current user so time left will be zero, either for the entire folder
 # containing the current resource, or just the resource, depending on value of first item  # containing the current resource, or just the resource, depending on value of first item
 # in interval array retrieved from EXT("resource.0.interval");  # in interval array retrieved from EXT("resource.0.interval");
 #  #
Line 1451  sub zero_timer { Line 1624  sub zero_timer {
                         $key =~ s/\s+$//;                          $key =~ s/\s+$//;
                         if ($env{'form.LC_interval_done_proctorpass'} ne $secret) {                          if ($env{'form.LC_interval_done_proctorpass'} ne $secret) {
                             return ('fail',                              return ('fail',
                                    &mt('Incorrect key entered by proctor'));                                      &mt('Incorrect key entered by proctor'));
                         }                          }
                     }                      }
                     if ($first_access+$timelimit > $now) {                      if ($first_access+$timelimit > $now) {
Line 1474  sub zero_timer { Line 1647  sub zero_timer {
                             return ('ok');                              return ('ok');
                         } else {                          } else {
                             return ('fail',&mt('Error ending timed event: [_1]',$result));                              return ('fail',&mt('Error ending timed event: [_1]',$result));
                         }                           }
                     } else {                      } else {
                         return ('fail',&mt('Timed event already ended'));                          return ('fail',&mt('Timed event already ended'));
                     }                      }
Line 1505  sub handler { Line 1678  sub handler {
     #check if we know where we are      #check if we know where we are
     if ($env{'request.course.fn'} && !&Apache::lonnet::symbread('','',1,1)) {      if ($env{'request.course.fn'} && !&Apache::lonnet::symbread('','',1,1)) {
  # if we are browsing we might not be able to know where we are   # if we are browsing we might not be able to know where we are
  if ($Apache::lonhomework::browse ne 'F' &&    if ($Apache::lonhomework::browse ne 'F' &&
     $env{'request.state'} ne "construct") {      $env{'request.state'} ne "construct") {
     #should know where we are, so ask      #should know where we are, so ask
     &unset_permissions();      &unset_permissions();
Line 1550  sub handler { Line 1723  sub handler {
     } else {      } else {
         # Set the event timer to zero if the "done button" was clicked.  The button is          # Set the event timer to zero if the "done button" was clicked.  The button is
         # part of the doneButton form created in lonmenu.pm          # part of the doneButton form created in lonmenu.pm
         my ($donebuttonresult,$donemsg);          my ($donebuttonresult,$donemsg,$viewasuser);
         if ($symb && $env{'form.LC_interval_done'} eq 'true') {            if ($symb && $env{'form.LC_interval_done'} eq 'true') {  
             ($donebuttonresult,$donemsg) = &zero_timer($symb);              ($donebuttonresult,$donemsg) = &zero_timer($symb);
             undef($env{'form.LC_interval_done'});              undef($env{'form.LC_interval_done'});
             undef($env{'form.LC_interval_done_proctorpass'});              undef($env{'form.LC_interval_done_proctorpass'});
         }          }
           if (($env{'form.LC_viewas'} ne '') && $symb && $env{'request.course.id'} &&
               ($Apache::lonhomework::viewgrades || $Apache::lonhomework::modifygrades)) {
               if ($env{'form.LC_viewas'} =~ /^($match_username):($match_domain)$/) {
                   my ($possuname,$possudom) = ($1,$2);
                   my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
                   my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
                   my ($canview,$posssec);
                   if ($env{'request.course.sec'} ne '') {
                       if ($Apache::lonhomework::modifygradessec eq $env{'request.course.sec'}) {
                           $canview = 'section';
                           $posssec = $env{'request.course.sec'};
                       } elsif ($Apache::lonhomework::viewgradessec eq $env{'request.course.sec'}) {
                           $canview = 'section';
                           $posssec = $env{'request.course.sec'};
                       }
                   }
                   my $crstype = &Apache::loncommon::course_type();
                   if (&Apache::loncourseuser::is_course_user($possudom,$possuname,$cdom,$cnum,
                                                              $canview,$crstype,$posssec)) {
                       $viewasuser = $possuname.':'.$possudom;
                   }
               }
               undef($env{'form.LC_viewas'});
           }
  # just render the page normally outside of construction space   # just render the page normally outside of construction space
  &Apache::lonxml::debug("not construct");   &Apache::lonxml::debug("not construct");
  &renderpage($request,$file,undef,undef,$donemsg);          undef(@Apache::lonhomework::ltipassback);
    &renderpage($request,$file,undef,undef,$donemsg,$viewasuser,$symb);
           if (@Apache::lonhomework::ltipassback) {
               unless ($registered_cleanup) {
                   my $handlers = $request->get_handlers('PerlCleanupHandler');
                   $request->set_handlers('PerlCleanupHandler' =>
                                          [\&do_ltipassback,@{$handlers}]);
               }
           }
     }      }
     #my $td=&tv_interval($t0);      #my $td=&tv_interval($t0);
     #&Apache::lonxml::debug("Spent $td seconds processing");      #&Apache::lonxml::debug("Spent $td seconds processing");
Line 1704  sub default_xml_tag { Line 1909  sub default_xml_tag {
   
 sub helpmenu_datastructure {  sub helpmenu_datastructure {
   
  my $width = 500;   # filename, title, width, height
  my $height = 600;  
   
  my $helpers = [   my $helpers = [
  ['Problem_LON-CAPA_Functions', &mt('Script Functions')],   ['Problem_LON-CAPA_Functions.hlp', &mt('Script Functions'), 800, 600],
  ['Greek_Symbols', &mt('Greek Symbols')],   ['Greek_Symbols.hlp', &mt('Greek Symbols'), 500, 600],
   ['Other_Symbols', &mt('Other Symbols')],    ['Other_Symbols.hlp', &mt('Other Symbols'), 500, 600],
  ['Authoring_Output_Tags', &mt('Output Tags')],   ['Authoring_Output_Tags.hlp', &mt('Output Tags'), 800, 600],
  ['Authoring_Multilingual_Problems',    ['Authoring_Multilingual_Problems.hlp', 
  &mt('How to create problems in different languages')]   &mt('How to create problems in different languages'), 800, 600],
    ['loncapa.html', &mt('Language reference'), 800, 600],
  ];   ];
   
  my $help_structure = [];   my $help_structure = [];
Line 1721  sub helpmenu_datastructure { Line 1925  sub helpmenu_datastructure {
  foreach my $count (0..(scalar(@{$helpers})-1)) {   foreach my $count (0..(scalar(@{$helpers})-1)) {
  my $filename = $helpers->[$count]->[0];   my $filename = $helpers->[$count]->[0];
  my $title = $helpers->[$count]->[1];   my $title = $helpers->[$count]->[1];
  my $href = &HTML::Entities::encode("javascript:openMyModal('/adm/help/$filename.hlp',$width,$height,'yes');");   my $width = $helpers->[$count]->[2];
                   my $height = $helpers->[$count]->[3];
                   if ($width eq '') {
                       $width = 500;
                   }
                   if ($height eq '') {
                       $height = 600;
                   }
    my $href = &HTML::Entities::encode("javascript:openMyModal('/adm/help/$filename',$width,$height,'yes');");
  push @{$help_structure}, [$href, $title, undef];   push @{$help_structure}, [$href, $title, undef];
  }   }
   
Line 1739  sub convert_for_js { Line 1951  sub convert_for_js {
     return $return;      return $return;
 }  }
   
   sub do_ltipassback {
       if (@Apache::lonhomework::ltipassback) {
           foreach my $item (@Apache::lonhomework::ltipassback) {
               if (ref($item) eq 'HASH') {
                   if ((ref($item->{'lti'}) eq 'HASH') && ($item->{'cid'} =~ /^($match_domain)_($match_courseid)$/)) {
                       my ($cdom,$cnum) = ($1,$2);
                       my $msgformat = $item->{'lti'}->{'passbackformat'};
                       my $sigmethod = 'HMAC-SHA1';
                       my $ltinum = $item->{'ltinum'};
                       my $id = $item->{'pbid'};
                       my $url = $item->{'pburl'};
                       my $type = $item->{'pbtype'};
                       my $scope = $item->{'scope'};
                       my $map = $item->{'ltimap'};
                       my $symb = $item->{'ltisymb'};
                       my $uname = $item->{'uname'};
                       my $udom = $item->{'udom'};
                       my $keynum = $item->{'lti'}->{'cipher'};
                       my $crsdef = $item->{'crsdef'};
                       my $scoretype = $item->{'format'};
                       my ($total,$possible);
                       if ($scope eq 'resource') {
                           $total = $item->{'total'};
                           $possible = $item->{'possible'};
                       } elsif ($scope eq 'map') {
                           ($total,$possible) = &get_lti_score($uname,$udom,$map);
                       } elsif ($scope eq 'course') {
                           ($total,$possible) = &get_lti_score($uname,$udom);
                       }
                       if (($id ne '') && ($url ne '') && ($possible)) {
                           &LONCAPA::ltiutils::send_grade($cdom,$cnum,$crsdef,$type,$ltinum,$keynum,$id,$url,$scoretype,$sigmethod,$msgformat,$total,$possible);
                       }
                   }
               }
           }
           undef(@Apache::lonhomework::ltipassback);
       }
   }
   
   sub get_lti_score {
       my ($uname,$udom,$mapurl) = @_;
       my $navmap = Apache::lonnavmaps::navmap->new($uname,$udom);
       if (ref($navmap)) {
           my $iterator;
           if ($mapurl ne '') {
               my $map = $navmap->getResourceByUrl($mapurl);
               my $firstres = $map->map_start();
               my $finishres = $map->map_finish();
               $iterator = $navmap->getIterator($firstres,$finishres,undef,1);
           } else {
               $iterator = $navmap->getIterator(undef,undef,undef,1);
           }
           if (ref($iterator)) {
               my $depth = 1;
               my $total = 0;
               my $possible = 0;
               $iterator->next(); # ignore first BEGIN_MAP
               my $curRes = $iterator->next();
               while ( $depth > 0 ) {
                   if ($curRes == $iterator->BEGIN_MAP()) {$depth++;}
                   if ($curRes == $iterator->END_MAP()) { $depth--; }
                   if (ref($curRes) && $curRes->is_gradable() && !$curRes->randomout) {
                       my $parts = $curRes->parts();
                       foreach my $part (@{$parts}) {
                           next if ($curRes->solved($part) eq 'excused');
                           $total += $curRes->weight($part) * $curRes->awarded($part);
                           $possible += $curRes->weight($part);
                       }
                   }
                   $curRes = $iterator->next();
               }
               if ($total > $possible) {
                   $total = $possible;
               }
               return ($total,$possible);
           }
       }
       return;
   }
   
 1;  1;
 __END__  __END__

Removed from v.1.362  
changed lines
  Added in v.1.378


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