Diff for /loncom/homework/lonhomework.pm between versions 1.344.2.10.4.2 and 1.349.2.3

version 1.344.2.10.4.2, 2023/07/05 16:58:52 version 1.349.2.3, 2015/05/26 13:25:52
Line 51  use Apache::functionplotresponse(); Line 51  use Apache::functionplotresponse();
 use Apache::drawimage();  use Apache::drawimage();
 use Apache::Constants qw(:common);  use Apache::Constants qw(:common);
 use Apache::loncommon();  use Apache::loncommon();
 use Apache::lonparmset();  use Apace::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 193  sub proctor_checked_in { Line 189  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') || ($type eq 'tool')) {      } elsif ($type eq 'problem') {
  $key ='resource.0.checkedin';   $key ='resource.0.checkedin';
     }      }
     # backward compatability, used to be username@domain,       # backward compatability, used to be username@domain, 
Line 208  sub proctor_checked_in { Line 204  sub proctor_checked_in {
     return 1;      return 1;
  }   }
     }      }
       
     return 0;      return 0;
 }  }
   
 sub check_slot_access {  sub check_slot_access {
     my ($id,$type,$symb,$partlist)=@_;      my ($id,$type)=@_;
   
     # does it pass normal muster      # does it pass normal muster
     my ($status,$datemsg)=&check_access($id,$symb);      my ($status,$datemsg)=&check_access($id);
       
     my $useslots = &Apache::lonnet::EXT("resource.0.useslots",$symb);      my $useslots = &Apache::lonnet::EXT("resource.0.useslots");
     if ($useslots ne 'resource' && $useslots ne 'map'       if ($useslots ne 'resource' && $useslots ne 'map' 
  && $useslots ne 'map_map') {   && $useslots ne 'map_map') {
  return ($status,$datemsg);   return ($status,$datemsg);
Line 239  sub check_slot_access { Line 236  sub check_slot_access {
     $Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass') {      $Apache::lonhomework::history{"resource.$version.0.status"} eq 'pass') {
     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");
     my $available = &Apache::lonnet::EXT("resource.0.available",$symb);      my $available = &Apache::lonnet::EXT("resource.0.available");
     my @slots= (split(':',$availablestudent),split(':',$available));      my @slots= (split(':',$availablestudent),split(':',$available));
   
 #    if (!@slots) {  #    if (!@slots) {
Line 258  sub check_slot_access { Line 249  sub check_slot_access {
     my ($returned_slot,$slot_name);      my ($returned_slot,$slot_name);
     my $now = time;      my $now = time;
     my $num_usable_slots = 0;      my $num_usable_slots = 0;
     unless ($symb) {  
         ($symb) = &Apache::lonnet::whichuser();  
     }  
     foreach my $slot (@slots) {      foreach my $slot (@slots) {
  $slot =~ s/(^\s*|\s*$)//g;   $slot =~ s/(^\s*|\s*$)//g;
  &Apache::lonxml::debug("getting $slot");   &Apache::lonxml::debug("getting $slot");
Line 284  sub check_slot_access { Line 272  sub check_slot_access {
  $slotstatus=$status;   $slotstatus=$status;
     }      }
   
     my ($is_correct,$got_grade,$checkin,$checkinslot,$checkedin,$consumed_uniq);      my ($is_correct,$got_grade,$checkedin);
     if ($type eq 'Task') {      if ($type eq 'Task') {
  my $version=$Apache::lonhomework::history{'resource.0.version'};   my $version=$Apache::lonhomework::history{'resource.0.version'};
         $checkin = "resource.$version.0.checkedin";  
  $got_grade =    $got_grade = 
     ($Apache::lonhomework::history{"resource.$version.0.status"}       ($Apache::lonhomework::history{"resource.$version.0.status"} 
      =~ /^(?:pass|fail)$/);       =~ /^(?:pass|fail)$/);
Line 296  sub check_slot_access { Line 283  sub check_slot_access {
      || $Apache::lonhomework::history{"resource.0.solved"} =~ /^correct_/ );       || $Apache::lonhomework::history{"resource.0.solved"} =~ /^correct_/ );
  $checkedin =   $checkedin =
     $Apache::lonhomework::history{"resource.$version.0.checkedin"};      $Apache::lonhomework::history{"resource.$version.0.checkedin"};
     } elsif (($type eq 'problem') || ($type eq 'tool')) {      } elsif ($type eq 'problem') {
         $checkin = 'resource.0.checkedin';   $got_grade  = 1;
  $checkedin  = $Apache::lonhomework::history{$checkin};   $checkedin  = $Apache::lonhomework::history{"resource.0.checkedin"};
     }   $is_correct =
     if ($checkedin) {      ($Apache::lonhomework::history{"resource.0.solved"} =~/^correct_/);
         $checkinslot = $Apache::lonhomework::history{"$checkin.slot"};  
         my %slot=&Apache::lonnet::get_slot($checkinslot);  
         $consumed_uniq = $slot{'uniqueperiod'};  
     }  
     if (($type eq 'problem') || ($type eq 'tool')) {  
         if ((ref($partlist) eq 'ARRAY') && (@{$partlist} > 0)) {  
             my ($numcorrect,$numgraded) = (0,0);  
             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 364  sub check_slot_access { Line 304  sub check_slot_access {
     # However, the problem is not closed, and potentially, another slot might be      # However, the problem is not closed, and potentially, another slot might be
     # 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'}) {                  if ($env{'request.course.id'}) {
                     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 ($symb)=&Apache::lonnet::whichuser();
                     $slotstatus = 'NOTRESERVABLE';                      $slotstatus = 'NOTRESERVABLE';
                     my ($reservable_now_order,$reservable_now,$reservable_future_order,                      my ($reservable_now_order,$reservable_now,$reservable_future_order,
                         $reservable_future) =                           $reservable_future) = 
                         &Apache::loncommon::get_future_slots($cnum,$cdom,$now,$symb);                          &Apache::loncommon::get_future_slots($cnum,$cdom,$now,$symb);
                     if ((ref($reservable_now_order) eq 'ARRAY') && (ref($reservable_now) eq 'HASH')) {                      if ((ref($reservable_now_order) eq 'ARRAY') && (ref($reservable_now) eq 'HASH')) {
                         if (@{$reservable_now_order} > 0) {                          if (@{$reservable_now_order} > 0) {
                             if ((!$checkedin) || (ref($consumed_uniq) ne 'ARRAY')) {                              $slotstatus = 'RESERVABLE';
                                 $slotstatus = 'RESERVABLE';                              $datemsg = $reservable_now->{$reservable_now_order->[-1]}{'endreserve'};
                                 $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') {                      unless ($slotstatus eq 'RESERVABLE') {
                         if ((ref($reservable_future_order) eq 'ARRAY') && (ref($reservable_future) eq 'HASH')) {                          if ((ref($reservable_future_order) eq 'ARRAY') && (ref($reservable_future) eq 'HASH')) {
                             if (@{$reservable_future_order} > 0) {                              if (@{$reservable_future_order} > 0) {
                                 if ((!$checkedin) || (ref($consumed_uniq) ne 'ARRAY')) {                                  $slotstatus = 'RESERVABLE_LATER';
                                     $slotstatus = 'RESERVABLE_LATER';                                  $datemsg = $reservable_future->{$reservable_future_order->[0]}{'startreserve'};
                                     $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'};  
                                     }  
                                 }  
                             }                              }
                         }                          }
                     }                      }
Line 450  sub check_slot_access { Line 348  sub check_slot_access {
     }      }
   
     if ( $is_correct) {      if ( $is_correct) {
  if (($type eq 'problem') || ($type eq 'tool')) {   if ($type eq 'problem') {
     return ($status);      return ($status);
  }   }
  return ('SHOW_ANSWER');   return ('SHOW_ANSWER');
Line 467  sub check_slot_access { Line 365  sub check_slot_access {
 # 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 {
     my ($id,$symb) = @_;      my ($id) = @_;
     my $date ='';      my $date ='';
     my $status;      my $status;
     my $datemsg = '';      my $datemsg = '';
Line 497  sub check_access { Line 395  sub check_access {
     &Apache::lonxml::debug("checking for part :$id:");      &Apache::lonxml::debug("checking for part :$id:");
     &Apache::lonxml::debug("time:".time);      &Apache::lonxml::debug("time:".time);
   
     unless ($symb) {      my ($symb)=&Apache::lonnet::whichuser();
         ($symb)=&Apache::lonnet::whichuser();  
     }  
     &Apache::lonxml::debug("symb:".$symb);      &Apache::lonxml::debug("symb:".$symb);
     #if ($env{'request.state'} ne "construct" && $symb ne '') {      #if ($env{'request.state'} ne "construct" && $symb ne '') {
     if ($env{'request.state'} ne "construct") {      if ($env{'request.state'} ne "construct") {
         my $idacc = &Apache::lonnet::EXT("resource.$id.acc",$symb);          my $idacc = &Apache::lonnet::EXT("resource.$id.acc");
  my $allowed=&Apache::loncommon::check_ip_acc($idacc);   my $allowed=&Apache::loncommon::check_ip_acc($idacc);
  if (!$allowed && ($Apache::lonhomework::browse ne 'F')) {   if (!$allowed && ($Apache::lonhomework::browse ne 'F')) {
     $status='INVALID_ACCESS';      $status='INVALID_ACCESS';
Line 519  sub check_access { Line 415  sub check_access {
  foreach my $temp ("opendate","duedate","answerdate") {   foreach my $temp ("opendate","duedate","answerdate") {
     $lastdate = $date;      $lastdate = $date;
     if ($temp eq 'duedate') {      if ($temp eq 'duedate') {
  $date = &due_date($id,$symb);   $date = &due_date($id);
     } else {      } else {
  $date = &Apache::lonnet::EXT("resource.$id.$temp",$symb);   $date = &Apache::lonnet::EXT("resource.$id.$temp");
     }      }
           
     my $thistype = &Apache::lonnet::EXT("resource.$id.$temp.type",$symb);      my $thistype = &Apache::lonnet::EXT("resource.$id.$temp.type");
     if ($thistype =~ /^(con_lost|no_such_host)/ ||      if ($thistype =~ /^(con_lost|no_such_host)/ ||
  $date     =~ /^(con_lost|no_such_host)/) {   $date     =~ /^(con_lost|no_such_host)/) {
  $status='UNAVAILABLE';   $status='UNAVAILABLE';
Line 533  sub check_access { Line 429  sub check_access {
     }      }
     if ($thistype eq 'date_interval') {      if ($thistype eq 'date_interval') {
  if ($temp eq 'opendate') {   if ($temp eq 'opendate') {
     $date=&Apache::lonnet::EXT("resource.$id.duedate",$symb)-$date;      $date=&Apache::lonnet::EXT("resource.$id.duedate")-$date;
  }   }
  if ($temp eq 'answerdate') {   if ($temp eq 'answerdate') {
     $date=&Apache::lonnet::EXT("resource.$id.duedate",$symb)+$date;      $date=&Apache::lonnet::EXT("resource.$id.duedate")+$date;
  }   }
     }      }
     &Apache::lonxml::debug("found :$date: for :$temp:");      &Apache::lonxml::debug("found :$date: for :$temp:");
Line 570  sub check_access { Line 466  sub check_access {
  (($Apache::lonhomework::browse eq 'F') && ($status eq 'CLOSED'))) {   (($Apache::lonhomework::browse eq 'F') && ($status eq 'CLOSED'))) {
  #check #tries, and if correct.   #check #tries, and if correct.
  my $tries = $Apache::lonhomework::history{"resource.$id.tries"};   my $tries = $Apache::lonhomework::history{"resource.$id.tries"};
  my $maxtries = &Apache::lonnet::EXT("resource.$id.maxtries",$symb);   my $maxtries = &Apache::lonnet::EXT("resource.$id.maxtries");
  if ( $tries eq '' ) { $tries = '0'; }   if ( $tries eq '' ) { $tries = '0'; }
  if ( $maxtries eq '' &&    if ( $maxtries eq '' && 
      $env{'request.state'} ne 'construct') { $maxtries = '2'; }        $env{'request.state'} ne 'construct') { $maxtries = '2'; } 
Line 579  sub check_access { Line 475  sub check_access {
  if ( ($Apache::lonhomework::history{"resource.$id.solved"}=~/^correct/)   if ( ($Apache::lonhomework::history{"resource.$id.solved"}=~/^correct/)
       && (&show_problem_status()) ) {        && (&show_problem_status()) ) {
             if (($Apache::lonhomework::history{"resource.$id.awarded"} >= 1) ||              if (($Apache::lonhomework::history{"resource.$id.awarded"} >= 1) ||
                 (&Apache::lonnet::EXT("resource.$id.retrypartial",$symb) !~/^1|on|yes$/i)) {                  (&Apache::lonnet::EXT("resource.$id.retrypartial") !~/^1|on|yes$/i)) {
         $status = 'CANNOT_ANSWER';          $status = 'CANNOT_ANSWER';
             }              }
         } elsif ($Apache::lonhomework::history{"resource.$id.solved"}=~/^excused/) {          } elsif ($Apache::lonhomework::history{"resource.$id.solved"}=~/^excused/) {
Line 591  sub check_access { Line 487  sub check_access {
  }   }
     }      }
     if ($status eq 'CAN_ANSWER' || $status eq 'CANNOT_ANSWER') {      if ($status eq 'CAN_ANSWER' || $status eq 'CANNOT_ANSWER') {
  my @interval=&Apache::lonnet::EXT("resource.$id.interval",$symb);   my @interval=&Apache::lonnet::EXT("resource.$id.interval");
  &Apache::lonxml::debug("looking for interval @interval");   &Apache::lonxml::debug("looking for interval @interval");
  if ($interval[0]) {   if ($interval[0]) {
     my $first_access=&Apache::lonnet::get_first_access($interval[1],$symb);      my $first_access=&Apache::lonnet::get_first_access($interval[1]);
     &Apache::lonxml::debug("looking for accesstime $first_access");      &Apache::lonxml::debug("looking for accesstime $first_access");
     if (!$first_access) {      if (!$first_access) {
  $status='NOT_YET_VIEWED';   $status='NOT_YET_VIEWED';
  my $due_date = &due_date($id,$symb);   my $due_date = &due_date($id);
  my $seconds_left = $due_date - time;   my $seconds_left = $due_date - time;
  if ($seconds_left > $interval[0] || $due_date eq '') {   if ($seconds_left > $interval[0] || $due_date eq '') {
     $seconds_left = $interval[0];      $seconds_left = $interval[0];
Line 872  sub analyze_header { Line 768  sub analyze_header {
        .&Apache::loncommon::head_subbox(         .&Apache::loncommon::head_subbox(
                 &Apache::loncommon::CSTR_pageheader());                  &Apache::loncommon::CSTR_pageheader());
     $result .=       $result .= 
             '<form name="lonhomework" method="post" action="'.      '<form name="lonhomework" method="post" action="'.
     &HTML::Entities::encode($env{'request.uri'},'<>&"').'">'.      &HTML::Entities::encode($env{'request.uri'},'<>&"').'">'.
             '<input type="hidden" name="problemmode" value="'.              '<input type="hidden" name="problemmode" value="'.
             $env{'form.problemmode'}.'" />'.              $env{'form.problemmode'}.'" />'.
Line 887  sub analyze_header { Line 783  sub analyze_header {
             'onclick="javascript:setmode(this.form,'."'view'".')" />              'onclick="javascript:setmode(this.form,'."'view'".')" />
             <hr />              <hr />
             </div>'              </div>'
             .&Apache::lonxml::message_location().'              .&Apache::lonxml::message_location().
             </form>';              '</form>';
     &Apache::lonxml::add_messages(\$result);      &Apache::lonxml::add_messages(\$result);
     $request->print($result);      $request->print($result);
     $request->rflush();      $request->rflush();
Line 1117  sub editxmlmode { Line 1013  sub editxmlmode {
             <div class="LC_edit_problem_header">              <div class="LC_edit_problem_header">
               <div class="LC_edit_problem_header_title">'.                <div class="LC_edit_problem_header_title">'.
                 &mt('Problem Editing').' '.&Apache::loncommon::help_open_topic('Problem_Editor_XML_Index').                  &mt('Problem Editing').' '.&Apache::loncommon::help_open_topic('Problem_Editor_XML_Index').
                 '</div><div class="LC_edit_actionbar" id="actionbar">';                '</div><div class="LC_edit_actionbar" id="actionbar">';
   
     $result.='<input type="hidden" name="problemmode" value="saveedit" />'.          $result.='<input type="hidden" name="problemmode" value="saveedit" />'.
                   &Apache::structuretags::problem_edit_buttons('editxml');                    &Apache::structuretags::problem_edit_buttons('editxml');
     $result.='<div>';          $result.='<div class="LC_edit_problem_discards">';
   
     $result .= '<ol class="LC_primary_menu" style="display:inline-block;font-size:90%;vertical-align:middle;">';  
   
     unless ($env{'environment.nocodemirror'}) {  
         # dropdown menus  
         $result .= Apache::lonmenu::create_submenu("#", "",  
             &mt("Problem Templates"), template_dropdown_datastructure());  
   
         $result .= Apache::lonmenu::create_submenu("#", "",  
             &mt("Response Types"), responseblock_dropdown_datastructure());  
   
         $result .= Apache::lonmenu::create_submenu("#", "",  
             &mt("Conditional Blocks"), conditional_scripting_datastructure());  
   
         $result .= Apache::lonmenu::create_submenu("#", "",  
             &mt("Miscellaneous"), misc_datastructure());  
     }  
   
     $result .= Apache::lonmenu::create_submenu("#", "",  
         &mt("Help") . ' <img src="/adm/help/help.png" alt="' . &mt("Help") .  
         '" style="vertical-align:text-bottom; height: auto; margin:0; "/>',  
         helpmenu_datastructure(),"");  
   
     $result.="</ol></div>";  
   
     $result .= '</div></div>' .  
         &Apache::lonxml::message_location() .  
         &Apache::loncommon::xmleditor_js() .  
         '<textarea ' . &Apache::edit::element_change_detection() .  
         ' rows="'.$rows.'" cols="'.$cols.'" style="width:100%" ' .  
         ' name="editxmltext" id="LC_editxmltext">' .  
         &HTML::Entities::encode($problem,'<>&"') .  
         '</textarea> <div id="LC_aftertextarea"> </div> </form>';  
   
     my $resource = $env{'request.ambiguous'};  
     unless($env{'environment.nocodemirror'}){  
         $result .= '<link rel="stylesheet" href="/adm/codemirror/codemirror-combined-xml.css">  
         <script src="/adm/codemirror/codemirror-compressed-xml.js"></script>  
         <script>  
             CodeMirror.defineMode("mixedmode", function(config) {  
                 return CodeMirror.multiplexingMode(  
                     CodeMirror.getMode(config, "xml"),  
                     {  
                         open: "\<script type=\"loncapa/perl\"\>", close: "\</script\>",  
                         mode: CodeMirror.getMode(config, "perl"),  
                         delimStyle: "tag",  
                     }  
               );  
             });  
             var cm = CodeMirror.fromTextArea(document.getElementById("LC_editxmltext"),  
             {  
                 mode: "mixedmode",  
                 lineWrapping: true,  
                 lineNumbers: true,  
                 tabSize: 4,  
                 indentUnit: 4,  
   
                 autoCloseTags: true,  
                 autoCloseBrackets: true,  
                 height: "auto",  
                 styleActiveLine: true,  
   
                 extraKeys: {  
                     "Tab": "indentMore",  
                     "Shift-Tab": "indentLess",  
                 }  
             });  
             restoreScrollPosition("'.$resource.'");  
         </script>';  
     }  
   
     $result .= &Apache::loncommon::end_page();   unless ($env{'environment.nocodemirror'}) {
     &Apache::lonxml::add_messages(\$result);   # dropdown menues
     $request->print($result);      $result .= '<ol class="LC_primary_menu LC_floatleft">'.
       &Apache::lonmenu::create_submenu("#", "", &mt("Insert Menu"), &Apache::structuretags::insert_menu_datastructure(),"").'</ol>';
    }
       $result .= '<ol class="LC_primary_menu LC_floatleft">'.
       Apache::lonmenu::create_submenu("#", "", &mt("Help"), &Apache::structuretags::helpmenu_datastructure(),"").'</ol>';
       $result.="</div>";
            
            $result.='<hr style="clear:both;visibility:hidden" /></div></div>'.&Apache::lonxml::message_location().
                     &Apache::loncommon::xmleditor_js().
     '<textarea '.&Apache::edit::element_change_detection().
                 ' rows="'.$rows.'" cols="'.$cols.'" style="width:100%" '.
         ' name="editxmltext" id="LC_editxmltext">'.
         &HTML::Entities::encode($problem,'<>&"').'</textarea>
               <div id="LC_aftertextarea">
               </div>
           </form>';
       my $resource = $env{'request.ambiguous'};
       unless($env{'environment.nocodemirror'}){
           
           $result .= '<link rel="stylesheet" href="/adm/codemirror/codemirror-combined-xml.css">
           <script src="/adm/codemirror/codemirror-compressed-xml.js"></script>
           <script>
               CodeMirror.defineMode("mixedmode", function(config) {
                   return CodeMirror.multiplexingMode(
                       CodeMirror.getMode(config, "xml"),
                       {
                           open: "\<script type=\"loncapa/perl\"\>", close: "\</script\>",
                           mode: CodeMirror.getMode(config, "perl"),
                           delimStyle: "tag",
                       }
                 );
               });
               var cm = CodeMirror.fromTextArea(document.getElementById("LC_editxmltext"),
               {
                   mode: "mixedmode",
                   lineWrapping: true,
                   lineNumbers: true,
                   tabSize: 4,
                   indentUnit: 4,
   
                   autoCloseTags: true,
                   autoCloseBrackets: true,
                   height: "auto",
                   styleActiveLine: true,
                   
                   extraKeys: {
                       "Tab": "indentMore",
                       "Shift-Tab": "indentLess",
                   }
               });
               restoreScrollPosition("'.$resource.'");
           </script>';
       }
           $result .= &Apache::loncommon::end_page();
           &Apache::lonxml::add_messages(\$result);
           $request->print($result);
     }      }
     return '';      return '';
 }  }
Line 1204  sub editxmlmode { Line 1085  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) = @_;
   
     my @targets = @{$targets || [&get_target()]};      my @targets = @{$targets || [&get_target()]};
     &Apache::lonhomework::showhashsubset(\%env,'form.');      &Apache::lonhomework::showhashsubset(\%env,'form.');
Line 1230  sub renderpage { Line 1111  sub renderpage {
     $problem='';      $problem='';
     my $filename=(split('/',$file))[-1];      my $filename=(split('/',$file))[-1];
     my $error =      my $error =
                 &mt('Unable to find [_1]',   '<p class="LC_error">'
    '<span class="LC_filename">'.$filename.'</span>');                 .&mt('Unable to find [_1]',
      '<span class="LC_filename">'.$filename.'</span>')
    ."</p>";
     $result.=      $result.=
  &Apache::loncommon::simple_error_page($request,'Not available',   &Apache::loncommon::simple_error_page($request,'Not available',
       $error,{'no_auto_mt_msg' => 1});        $error,{'no_auto_mt_msg' => 1});
Line 1255  sub renderpage { Line 1138  sub renderpage {
     if ($target eq 'analyze') {      if ($target eq 'analyze') {
  $result=&Apache::lonnet::hashref2str(\%Apache::lonhomework::analyze);   $result=&Apache::lonnet::hashref2str(\%Apache::lonhomework::analyze);
  undef(%Apache::lonhomework::analyze);   undef(%Apache::lonhomework::analyze);
             } elsif ($target eq 'web') {  
                 if ($donebuttonmsg) {  
                     $result =~ s{</body>}{};  
                     $result.= &Apache::loncommon::confirmwrapper(&Apache::lonhtmlcommon::confirm_success($donebuttonmsg,1))."\n</body>";  
                 }  
     }      }
     #my $td=&tv_interval($t0);      #my $td=&tv_interval($t0);
     #if ( $Apache::lonxml::debug) {      #if ( $Apache::lonxml::debug) {
Line 1293  sub finished_parsing { Line 1171  sub finished_parsing {
 # value 1: title  # value 1: title
 # value 2: category  # value 2: category
 # 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')),
                       (&Apache::lonnet::metadata($_, 'category')?&mt(&Apache::lonnet::metadata($_, 'category')):&mt('Miscellaneous')),                        (&Apache::lonnet::metadata($_, 'category')?&mt(&Apache::lonnet::metadata($_, 'category')):&mt('Miscellaneous')),
                      &mt(&Apache::lonnet::metadata($_, 'help'))]} (@files);                        &mt(&Apache::lonnet::metadata($_, 'help'))]} (@files);
     @files = sort {$a->[2].$a->[1] cmp $b->[2].$b->[1]} (@files);      @files = sort {$a->[2].$a->[1] cmp $b->[2].$b->[1]} (@files);
     return @files;      return @files;
 }  }
Line 1312  sub get_template_html { Line 1190  sub get_template_html {
     &Apache::lonxml::debug("Looking for :$extension:");      &Apache::lonxml::debug("Looking for :$extension:");
     my $glob_extension  = $extension;      my $glob_extension  = $extension;
     if ($extension eq 'survey' || $extension eq 'exam') {      if ($extension eq 'survey' || $extension eq 'exam') {
         $glob_extension = 'problem';   $glob_extension = 'problem';
     }      }
     my @files = &get_template_list($extension);      my @files = &get_template_list($extension);
     my ($midpoint,$seconddiv,$numfiles);      my ($midpoint,$seconddiv,$numfiles);
Line 1376  sub get_template_html { Line 1254  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);
         my $templatefilename =          &File::Copy::copy('/home/httpd/html/res/adm/includes/templates/blank.problem',$dest);
             $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;   my $file = $env{'form.template'};
         my ($extension) = ($env{'form.template'} =~ /\.(\w+)$/);   my $dest = &Apache::lonnet::filelocation("",$request->uri);
         if ($extension) {   &File::Copy::copy($file,$dest);
             my @files = &get_template_list($extension);   &renderpage($request,$dest);
             foreach my $poss (@files) {   return;
                 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 1425  sub newproblem { Line 1282  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 $brcrum = [{'href' => &Apache::loncommon::authorspace($request->uri),
                        'text' => 'Authoring Space'},                         'text' => 'Authoring Space'},
Line 1475  sub update_construct_style { Line 1333  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");
 #  #
 sub zero_timer {  sub zero_timer {
     my ($symb) = @_;      my ($symb) = @_;
     my ($hastimeleft,$first_access,$now);      my ($hastimeleft,$first_access,$now);
     my @interval=&Apache::lonnet::EXT("resource.0.interval",$symb);      my @interval=&Apache::lonnet::EXT("resource.0.interval");
     if (@interval > 1) {      if (@interval > 1) {
         if ($interval[1] eq 'course') {          if ($interval[1] eq 'course') {
             return ('fail',&mt('Ending of timed events not supported for intervals set course-wide'));              return;
         } else {          } else {
             my $now = time;              my $now = time;
             my $first_access=&Apache::lonnet::get_first_access($interval[1],$symb);              my $first_access=&Apache::lonnet::get_first_access($interval[1],$symb);
             if ($first_access > 0) {              if ($first_access > 0) {
                 my ($timelimit,$donesuffix) = split(/_/,$interval[0],2);                  if ($first_access+$interval[0] > $now) {
                 if ($donesuffix =~ /^done(?:|\:[^\:]+\:)(.*)$/) {                      my $done_time = $now - $first_access;
                     my ($dummy,$proctor,$secret) = split(/_/,$1);                      my $snum = 1;
                     if (($proctor) && ($secret ne '')) {                      if ($interval[1] eq 'map') {
                         my $key = $env{'form.LC_interval_done_proctorpass'};                          $snum = 2;
                         $key =~ s/^\s+//;  
                         $key =~ s/\s+$//;  
                         if ($env{'form.LC_interval_done_proctorpass'} ne $secret) {  
                             return ('fail',  
                                    &mt('Incorrect key entered by proctor'));  
                         }  
                     }                      }
                     if ($first_access+$timelimit > $now) {                      my $result = 
                         my $done_time = $now - $first_access;                          &Apache::lonparmset::storeparm_by_symb_inner($symb,'0_interval',
                         my $snum = 1;                                                                       $snum,$done_time,
                         if ($interval[1] eq 'map') {                                                                       'date_interval',
                             $snum = 2;                                                                       $env{'user.name'},
                         }                                                                       $env{'user.domain'});
                         my $result =                      return $result;
                             &Apache::lonparmset::storeparm_by_symb_inner($symb,'0_interval',  
                                                                          $snum,$done_time,  
                                                                          'date_interval',  
                                                                          $env{'user.name'},  
                                                                          $env{'user.domain'});  
                         if ($result eq '') {  
                             # Record action in "User Notes"  
                             &Apache::lonmsg::store_instructor_comment(  
                                 'Pressed Done button for symb:<br />'.$symb,  
                                 $env{'user.name'}, $env{'user.domain'});  
                             return ('ok');  
                         } else {  
                             return ('fail',&mt('Error ending timed event: [_1]',$result));  
                         }  
                     } else {  
                         return ('fail',&mt('Timed event already ended'));  
                     }  
                 } else {  
                     return ('fail',&mt('Timed event can not be ended before the time limit'));  
                 }                  }
             } else {  
                 return ('fail',&mt('Timer not yet started for this timed event'));  
             }              }
         }          }
     } else {  
         return ('fail',&mt('No timer in use'));  
     }      }
     return();      return;
 }  }
   
 sub handler {  sub handler {
     #my $t0 = [&gettimeofday()];      #my $t0 = [&gettimeofday()];
     my $request=$_[0];      my $request=$_[0];
   
     $Apache::lonxml::request=$request;      $Apache::lonxml::request=$request;
     $Apache::lonxml::debug=$env{'user.debug'};      $Apache::lonxml::debug=$env{'user.debug'};
     $env{'request.uri'}=$request->uri;      $env{'request.uri'}=$request->uri;
Line 1550  sub handler { Line 1380  sub handler {
     my $file=&Apache::lonnet::filelocation("",$request->uri);      my $file=&Apache::lonnet::filelocation("",$request->uri);
   
     #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()) { 
  # 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 1589  sub handler { Line 1419  sub handler {
  &renderpage($request,$file);   &renderpage($request,$file);
     }      }
  } else {   } else {
             &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},   &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
                                                     ['mode']);      ['mode']);
     # requested file doesn't exist in contruction space      # requested file doesn't exist in contruction space
     &newproblem($request);      &newproblem($request);
  }   }
     } 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);          if ($symb && $env{'form.LC_interval_done'} eq 'true') {  
         if ($symb && $env{'form.LC_interval_done'} eq 'true') {              &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'});  
         }          }
  # 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");
         undef(@Apache::lonhomework::ltipassback);   &renderpage($request,$file);
  &renderpage($request,$file,undef,undef,$donemsg);  
         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 1624  sub handler { Line 1444  sub handler {
   
 }  }
   
 sub template_dropdown_datastructure {  
     # gathering the all templates and their path, title, category and help topic  
     my @templates = get_template_list('problem');  
     # template category => title  
     my %tmplthash = ();  
     # template title => path  
     my %tmpltcontent = ();  
   
     foreach my $template (@templates){  
         # put in hash if the template is not empty  
         unless ($template->[1] eq ''){  
             push(@{$tmplthash{$template->[2]}}, $template->[1]);  
             push(@{$tmpltcontent{$template->[1]}},$template->[0]);  
         }  
     }  
   
         my $catList = [];  
     foreach my $cat (sort keys %tmplthash) {  
                 my $catItems = [];  
         foreach my $title (sort @{$tmplthash{$cat}}) {  
             my $path = $tmpltcontent{$title}->[0];  
             my $code;  
             open(FH, "<$path");  
             while(<FH>){  
                 $code.= $_ unless $_ =~ /(<problem>)|(<\/problem>)/;  
             }  
             close(FH);  
   
                         if ($code ne '') {  
                 my $href = 'javascript:insertText(\'' . &convert_for_js(&HTML::Entities::encode($code,'<>&"')) . '\')';  
                                 my $currItem = [$href, $title, undef];  
                                 push @{$catItems}, $currItem;  
                         }  
         }  
                 push @{$catList}, [$catItems, $cat, undef];  
     }  
   
     return $catList;  
 }  
   
 sub responseblock_dropdown_datastructure {  
   
     my $mathCat = [  
                     [  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_formularesponse())) . "\')", &mt("Formula Response"), undef],  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_functionplotresponse())) . "\')", &mt("Function Plot Response"), undef],  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_mathresponse())) . "\')", &mt("Math Response"), undef],  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_numericalresponse())) . "\')", &mt("Numerical Response"), undef]  
                     ],  
                     &mt("Math"),  
                     undef  
         ];  
   
     my $miscCat = [  
                     [  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_imageresponse())) . "\')", &mt("Click on Image"), undef],  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_customresponse())) . "\')", &mt("Custom Response"), undef],  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_externalresponse())) . "\')", &mt("External Response"), undef],  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_matchresponse())) . "\')", &mt("Match Two Lists"), undef],  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_radiobuttonresponse())) . "\')", &mt("One out of N statements"), undef],  
             ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_optionresponse())) . "\')", &mt("Select from Options"), undef],  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_rankresponse())) . "\')", &mt("Rank Values"), undef]  
                 ],  
                 &mt("Miscellaneous"),  
                 undef  
         ];  
   
     my $chemCat = [  
                     [  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_reactionresponse())) . "\')", &mt("Chemical Reaction"), undef],  
                         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_organicresponse())) . "\')", &mt("Organic Chemical Structure"), undef]  
                      ],  
                      &mt("Chemistry"),  
                      undef  
                    ];  
   
     my $textCat = [  
                     [  
                       ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_stringresponse())) . "\')", &mt("String Response"), undef],  
                       ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_essayresponse())) . "\')", &mt("Essay"), undef]  
                      ],  
                      &mt("Text"),  
                      undef  
                    ];  
   
     return [$mathCat, $miscCat, $chemCat, $textCat];  
 }  
   
   
 sub conditional_scripting_datastructure {  
 # TODO: corresponding routines should be used for the javascript:insertText parts  
 # instead of the placeholder routine default_xml_tag with the tags  
 # e.g. &default_xml_tag("postanswerdate") should be replaced with a routine which  
 # returns the corresponding content for this case  
   
 #TODO translated is currently temporarily here, another solution should be found where the  
 # needed string can be retrieved  
   
     my $translatedTag = '  
 <translated>  
     <lang which="en"></lang>  
     <lang which="default"></lang>  
 </translated>';  
     return [  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode($translatedTag)) . "\')", &mt("Translated Block"), undef],  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("block"))) . "\')", &mt("Conditional Block"), undef],  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("postanswerdate"))) . "\')", &mt("After Answer Date Block"), undef],  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("preduedate"))) . "\')", &mt("Before Due Date Block"), undef],  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("solved"))) . "\')", &mt("Block For After Solved"), undef],  
              ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("notsolved"))) . "\')", &mt("Block For When Not Solved"), undef]  
         ];  
 }  
   
 sub misc_datastructure {  
     return [  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_img())) . "\')", &mt("Image"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::lonplot::insert_gnuplot())) . "\')", &mt("GNU Plot"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_organicstructure())) . "\')", &mt("Organic Structure"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::edit::insert_script())) . "\')", &mt("Script Block"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("allow"))) . "\')", &mt("File Dependencies"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("import"))) . "\')", &mt("Import a File"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&Apache::londefdef::insert_meta())) . "\')", &mt("Custom Metadata"), undef],  
         ["javascript:insertText(\'" . &convert_for_js(&HTML::Entities::encode(&default_xml_tag("part"))) . "\')", &mt("Problem Part"), undef]  
     ];  
 }  
   
 # helper routine for the datastructure building subroutines  
 sub default_xml_tag {  
     my ($tag) = @_;  
     return "\n<$tag></$tag>";  
 }  
   
 sub helpmenu_datastructure {  
   
     # filename, title, width, height  
     my $helpers = [  
                     ['Problem_LON-CAPA_Functions.hlp', &mt('Script Functions'), 800, 600],  
                     ['Greek_Symbols.hlp', &mt('Greek Symbols'), 500, 600],  
                     ['Other_Symbols.hlp', &mt('Other Symbols'), 500, 600],  
                     ['Authoring_Output_Tags.hlp', &mt('Output Tags'), 800, 600],  
                     ['Authoring_Multilingual_Problems.hlp', &mt('Languages'), 800, 600],  
                    ];  
   
     my $help_structure = [];  
   
     foreach my $count (0..(scalar(@{$helpers})-1)) {  
         my $filename = $helpers->[$count]->[0];  
         my $title = $helpers->[$count]->[1];  
         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];  
     }  
   
     return $help_structure;  
 }  
   
 # we need substitution to not break javascript code  
 sub convert_for_js {  
     my $return = shift;  
     $return =~ s|script|ESCAPEDSCRIPT|g;  
     $return =~ s|\\|\\\\|g;  
     $return =~ s|\n|\\r\\n|g;  
     $return =~ s|'|\\'|g;  
     $return =~ s|&#39;|\\&#39;|g;  
     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.344.2.10.4.2  
changed lines
  Added in v.1.349.2.3


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