Diff for /loncom/homework/structuretags.pm between versions 1.512.2.24.2.15 and 1.512.2.25

version 1.512.2.24.2.15, 2024/03/03 22:51:31 version 1.512.2.25, 2023/09/11 13:46:11
Line 62  use Apache::lonxml; Line 62  use Apache::lonxml;
 use Apache::londefdef;  use Apache::londefdef;
 use Apache::lonenc();  use Apache::lonenc();
 use Apache::loncommon();  use Apache::loncommon();
 use Apache::lonnavmaps;  
 use Time::HiRes qw( gettimeofday tv_interval );  use Time::HiRes qw( gettimeofday tv_interval );
 use HTML::Entities();  use HTML::Entities();
 use lib '/home/httpd/lib/perl/';  use lib '/home/httpd/lib/perl/';
 use LONCAPA;  use LONCAPA;
     
 BEGIN {  BEGIN {
     &Apache::lonxml::register('Apache::structuretags',('block','languageblock','translated','instructorcomment','while','randomlist','problem','library','web','print','tex','part','preduedate','postanswerdate','solved','notsolved','problemtype','startpartmarker','startouttext','endpartmarker','endouttext','simpleeditbutton','definetag'));      &Apache::lonxml::register('Apache::structuretags',('block','languageblock','translated','instructorcomment','while','randomlist','problem','library','web','tex','part','preduedate','postanswerdate','solved','notsolved','problemtype','startpartmarker','startouttext','endpartmarker','endouttext','simpleeditbutton','definetag'));
 }  }
   
   
Line 194  sub end_web { Line 193  sub end_web {
     return '';      return '';
 }  }
   
 sub start_print {  
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;  
     if ($target ne 'edit' && $target ne 'modified') {  
         if ($target ne 'tex') {  
             my $skip = &Apache::lonxml::get_all_text("/print",$parser,$style);  
             &Apache::lonxml::debug("skipping ahead :$skip: $$parser[-1]");  
         }  
     } elsif ($target eq "edit") {  
         my $bodytext = &Apache::lonxml::get_all_text_unbalanced("/print",$parser);  
         my $result = &Apache::edit::tag_start($target,$token);  
         $result .= &Apache::edit::editfield($token->[1],$bodytext,'',80,1);  
         return $result;  
     } elsif ($target eq "modified") {  
         return $token->[4].&Apache::edit::modifiedfield("/print",$parser);  
     }  
     return '';  
 }  
   
 sub end_print {  
     return '';  
 }  
   
 sub start_tex {  sub start_tex {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;      my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result='';      my $result='';
Line 295  sub homework_js { Line 272  sub homework_js {
     }      }
     return &Apache::loncommon::resize_textarea_js().      return &Apache::loncommon::resize_textarea_js().
            &Apache::loncommon::colorfuleditor_js().             &Apache::loncommon::colorfuleditor_js().
            &Apache::lonxml::setmode_javascript().             &setmode_javascript().
  <<"JS";   <<"JS";
 <script type="text/javascript">  <script type="text/javascript">
 // <![CDATA[  // <![CDATA[
Line 407  var postsubmit = '$postsubmit'; Line 384  var postsubmit = '$postsubmit';
 JS  JS
 }  }
   
   sub setmode_javascript {
       return <<"ENDSCRIPT";
   <script type="text/javascript">
   // <![CDATA[
   function setmode(form,probmode) {
       var initial = form.problemmode.value;
       form.problemmode.value = probmode;
       form.submit();
       form.problemmode.value = initial;
   }
   // ]]>
   </script>
   ENDSCRIPT
   }
   
 sub page_start {  sub page_start {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$name,      my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$name,
  $extra_head)=@_;   $extra_head)=@_;
Line 428  sub page_start { Line 420  sub page_start {
     if (&Apache::lonhtmlcommon::htmlareabrowser()) {      if (&Apache::lonhtmlcommon::htmlareabrowser()) {
         my %textarea_args;          my %textarea_args;
         if (($env{'request.state'} ne 'construct') ||          if (($env{'request.state'} ne 'construct') ||
             (&Apache::loncommon::nocodemirror())) {              ($env{'environment.nocodemirror'})) {
             %textarea_args = (              %textarea_args = (
                                 dragmath => 'math',                                  dragmath => 'math',
                               );                                );
Line 536  sub page_start { Line 528  sub page_start {
     } elsif (!defined($found{'body'})       } elsif (!defined($found{'body'}) 
      && $env{'request.state'} eq 'construct') {       && $env{'request.state'} eq 'construct') {
  if ($target eq 'web' || $target eq 'edit') {   if ($target eq 'web' || $target eq 'edit') {
             unless ($env{'form.inhibitmenu'} eq 'yes') {          # Breadcrumbs for Authoring Space
                 # Breadcrumbs for Authoring Space          &Apache::lonhtmlcommon::clear_breadcrumbs();
                 &Apache::lonhtmlcommon::clear_breadcrumbs();          &Apache::lonhtmlcommon::add_breadcrumb({
                 &Apache::lonhtmlcommon::add_breadcrumb({              'text'  => 'Authoring Space',
                     'text'  => 'Authoring Space',              'href'  => &Apache::loncommon::authorspace($env{'request.uri'}),
                     'href'  => &Apache::loncommon::authorspace($env{'request.uri'}),          });
                 });          # breadcrumbs (and tools) will be created 
                 # breadcrumbs (and tools) will be created           # in start_page->bodytag->innerregister
                 # in start_page->bodytag->innerregister  
   
 # FIXME Where are we?  # FIXME Where are we?
 #                &Apache::lonhtmlcommon::add_breadcrumb({  #        &Apache::lonhtmlcommon::add_breadcrumb({
 #                    'text'  => 'Problem Editing', # 'Problem Testing'  #            'text'  => 'Problem Editing', # 'Problem Testing'
 #                    'href'  => '',  #            'href'  => '',
 #               });  #        });
                 $pageheader = &Apache::loncommon::head_subbox(          $pageheader =&Apache::loncommon::head_subbox(
                                  &Apache::loncommon::CSTR_pageheader());                  &Apache::loncommon::CSTR_pageheader());
     }   }
         }  
     } elsif (!defined($found{'body'})) {      } elsif (!defined($found{'body'})) {
  my %add_entries;   my %add_entries;
  my $background=&Apache::lonxml::get_param('background',$parstack,   my $background=&Apache::lonxml::get_param('background',$parstack,
Line 609  sub page_start { Line 599  sub page_start {
         my ($symb,$courseid,$udom,$uname)=&Apache::lonnet::whichuser();          my ($symb,$courseid,$udom,$uname)=&Apache::lonnet::whichuser();
         my ($path,$multiresp) =           my ($path,$multiresp) = 
             &Apache::loncommon::get_turnedin_filepath($symb,$uname,$udom);              &Apache::loncommon::get_turnedin_filepath($symb,$uname,$udom);
         if ($env{'request.user_in_effect'}) {          if (($is_task) || ($needs_upload)) {
             $form_tag_start .= ' onsubmit="preventDefault();"';  
         } elsif (($is_task) || ($needs_upload)) {  
             $form_tag_start .= ' onsubmit="return file_submission_check(this,'."'$path','$multiresp'".');"';              $form_tag_start .= ' onsubmit="return file_submission_check(this,'."'$path','$multiresp'".');"';
         }          }
  $form_tag_start.='>'."\n";   $form_tag_start.='>'."\n";
Line 688  sub setup_rndseed { Line 676  sub setup_rndseed {
             $reqtries = &Apache::lonnet::EXT("resource.$partfortries.randomizeontries");              $reqtries = &Apache::lonnet::EXT("resource.$partfortries.randomizeontries");
         }          }
     }      }
     if (($env{'request.state'} eq "construct")      if (($env{'request.state'} eq "construct") 
         || ($symb eq '')          || ($symb eq '')
         || ($Apache::lonhomework::type eq 'practice')          || ($Apache::lonhomework::type eq 'practice')
         || ($Apache::lonhomework::history{'resource.CODE'})          || ($Apache::lonhomework::history{'resource.CODE'}) 
         || (($env{'form.code_for_randomlist'}) && ($target eq 'analyze'))) {          || (($env{'form.code_for_randomlist'}) && ($target eq 'analyze'))) {
  &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},   &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
  ['rndseed']);   ['rndseed']);
Line 717  sub setup_rndseed { Line 705  sub setup_rndseed {
             }              }
             $env{'form.'.$Apache::inputtags::part.'.rndseed'}=$rndseed;              $env{'form.'.$Apache::inputtags::part.'.rndseed'}=$rndseed;
         }          }
  if ( ($env{'form.resetdata'} eq 'new_problem_variation'   if ( ($env{'form.resetdata'} eq &mt('New Problem Variation')
       && $env{'form.submitted'} eq 'yes')  ||        && $env{'form.submitted'} eq 'yes')  ||
     $env{'form.newrandomization'} eq &mt('New Randomization')) {      $env{'form.newrandomization'} eq &mt('New Randomization')) {
     srand(time);      srand(time);
Line 800  sub problem_edit_action_button { Line 788  sub problem_edit_action_button {
   
 sub problem_edit_buttons {  sub problem_edit_buttons {
    my ($mode)=@_;     my ($mode)=@_;
    my %editors = &Apache::loncommon::permitted_editors();  
 # Buttons that save  # Buttons that save
    my $result = '<div style="float:right">';     my $result = '<div style="float:right">';
    if ($mode eq 'editxml') {     if ($mode eq 'editxml') {
        if ($editors{'xml'}) {         $result.=&problem_edit_action_button('subsaveedit','saveeditxml','s','Save and EditXML');
            $result.=&problem_edit_action_button('subsaveedit','saveeditxml','s','Save and EditXML');  
        }  
        $result.=&problem_edit_action_button('subsaveview','saveviewxml','v','Save and View');         $result.=&problem_edit_action_button('subsaveview','saveviewxml','v','Save and View');
    } else {     } else {
        if ($editors{'edit'}) {         $result.=&problem_edit_action_button('subsaveedit','saveedit','s','Save and Edit');
            $result.=&problem_edit_action_button('subsaveedit','saveedit','s','Save and Edit');  
        }  
        $result.=&problem_edit_action_button('subsaveview','saveview','v','Save and View');         $result.=&problem_edit_action_button('subsaveview','saveview','v','Save and View');
    }     }
    $result.="\n</div>\n";     $result.="\n</div>\n";
Line 819  sub problem_edit_buttons { Line 802  sub problem_edit_buttons {
    $result .= '<div>'.     $result .= '<div>'.
               &problem_edit_action_button('subdiscview','discard','d','Discard Edits and View',1);                &problem_edit_action_button('subdiscview','discard','d','Discard Edits and View',1);
    if ($mode eq 'editxml') {     if ($mode eq 'editxml') {
        if ($editors{'edit'}) {         $result.=&problem_edit_action_button('subedit','edit','e','Edit',1);
            $result.=&problem_edit_action_button('subedit','edit','e','Edit',1);  
        }  
        if ($editors{'daxe'}) {  
            $result.=&problem_edit_action_button('subdaxe','daxe','w','Edit with Daxe',1);  
        }  
        $result.=&problem_edit_action_button('subundo','undoxml','u','Undo',1);         $result.=&problem_edit_action_button('subundo','undoxml','u','Undo',1);
        if (&Apache::loncommon::nocodemirror()) {         if ($env{'environment.nocodemirror'}) {
            $result.=&Apache::lonhtmlcommon::dragmath_button("LC_editxmltext",1);             $result.=&Apache::lonhtmlcommon::dragmath_button("LC_editxmltext",1);
        }         }
    } else {     } else {
        if ($editors{'xml'}) {         $result.=&problem_edit_action_button('subeditxml','editxml','x','EditXML',1);
            $result.=&problem_edit_action_button('subeditxml','editxml','x','EditXML',1);  
        }  
        if ($editors{'daxe'}) {  
            $result.=&problem_edit_action_button('subdaxe','daxe','w','Edit with Daxe',1);  
        }  
        $result.=&problem_edit_action_button('subundo','undo','u','Undo',1);         $result.=&problem_edit_action_button('subundo','undo','u','Undo',1);
    }     }
    $result.="\n</div>";     $result.="\n</div>";
Line 849  sub problem_edit_header { Line 822  sub problem_edit_header {
         <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').$mode.&Apache::loncommon::help_open_menu('Problem Editing','Problem_Editor_XML_Index',5,'Authoring').'          '.&mt('Problem Editing').$mode.&Apache::loncommon::help_open_menu('Problem Editing','Problem_Editor_XML_Index',5,'Authoring').'
         </div><div class="LC_edit_actionbar" id="actionbar">'.           </div><div class="LC_edit_actionbar" id="actionbar">'.
         '<input type="hidden" name="problemmode" value="saveedit" />'.           '<input type="hidden" name="problemmode" value="saveedit" />'.
         &problem_edit_buttons();           &problem_edit_buttons();
   
     $return .= '</div></div>' . &Apache::lonxml::message_location();      $return .= '</div></div>' . &Apache::lonxml::message_location();
     $return .= '<link rel="stylesheet" href="/adm/codemirror/codemirror-combined.css" />      $return .= '<link rel="stylesheet" href="/adm/codemirror/codemirror-combined.css" />
Line 908  sub option { Line 881  sub option {
   
 sub problem_web_to_edit_header {  sub problem_web_to_edit_header {
     my ($rndseed)=@_;      my ($rndseed)=@_;
     my %editors = &Apache::loncommon::permitted_editors();  
     my $result .= '<div class="LC_edit_problem_header">';      my $result .= '<div class="LC_edit_problem_header">';
   
     if (!$Apache::lonhomework::parsing_a_task) {      if (!$Apache::lonhomework::parsing_a_task) {
Line 1008  $show_all Line 980  $show_all
      </div>       </div>
      <input type="submit" name="changeproblemmode" value="'.&mt("Change View").'" />       <input type="submit" name="changeproblemmode" value="'.&mt("Change View").'" />
      <input type="submit" name="clear_style_file" accesskey="d" value="'.&mt('Show Default View').'" />       <input type="submit" name="clear_style_file" accesskey="d" value="'.&mt('Show Default View').'" />
      <button type="submit" name="resetdata" accesskey="r" value="reset_submissions">       <input type="submit" name="resetdata" accesskey="r" value="'.&mt('Reset Submissions').'" />
      '.&mt('Reset Submissions').'</button>  
    </div>     </div>
    <hr />     <hr />
    <div class="LC_edit_problem_header_randomize_row">     <div class="LC_edit_problem_header_randomize_row">
Line 1038  $show_all Line 1009  $show_all
    <hr />     <hr />
    <div>';     <div>';
     $result.='<input type="hidden" name="problemmode" value="view" />';      $result.='<input type="hidden" name="problemmode" value="view" />';
     if ($editors{'edit'}) {      $result .= '<input type="button" name="submitmode" accesskey="e" value="'.&mt('Edit').'" '.
         $result .= '<input type="button" name="submitmode" accesskey="e" value="'.&mt('Edit').'" '.                 'onclick="javascript:setmode(this.form,'."'edit'".')" />';
                    'onclick="javascript:setmode(this.form,'."'edit'".')" />';      $result .= '<input type="button" name="submitmode" accesskey="x" value="'.&mt('EditXML').'" '.
     }                 'onclick="javascript:setmode(this.form,'."'editxml'".')" />';
     if ($editors{'xml'}) {  
         $result .= '<input type="button" name="submitmode" accesskey="x" value="'.&mt('EditXML').'" '.  
                    'onclick="javascript:setmode(this.form,'."'editxml'".')" />';  
     }  
     if (($editors{'daxe'}) &&  
         ($env{'browser.type'} ne 'explorer' || $env{'browser.version'} > 9)) {  
         my $uri = $env{'request.uri'};  
         my $daxeurl = '/daxepage'.$uri;  
         $result .= '<input type="button" accesskey="w" value="'.&mt('Edit with Daxe').'" '.  
                    'onclick="javascript:setmode(this.form,'."'daxe'".')" />';  
     }  
     $result.='      $result.='
    </div>     </div>
    <hr />     <hr />
Line 1079  sub initialize_storage { Line 1039  sub initialize_storage {
  || $Apache::lonhomework::type eq 'practice') {   || $Apache::lonhomework::type eq 'practice') {
   
  my $namespace = $symb || $env{'request.uri'};   my $namespace = $symb || $env{'request.uri'};
         if ($env{'form.resetdata'} eq 'reset_submissions' ||   if ($env{'form.resetdata'} eq &mt('Reset Submissions') ||
             ($env{'form.resetdata'} eq 'new_problem_variation'      ($env{'form.resetdata'} eq &mt('New Problem Variation')
      && $env{'form.submitted'} eq 'yes') ||       && $env{'form.submitted'} eq 'yes') ||
     $env{'form.newrandomization'} eq &mt('New Randomization')) {      $env{'form.newrandomization'} eq &mt('New Randomization')) {
     &Apache::lonnet::tmpreset($namespace,'',$domain,$name);      &Apache::lonnet::tmpreset($namespace,'',$domain,$name);
Line 1132  sub initialize_storage { Line 1092  sub initialize_storage {
         for any parts with out-of-order storage (i.e., correct then incorrect,          for any parts with out-of-order storage (i.e., correct then incorrect,
         where awarded >= 1 when correct).          where awarded >= 1 when correct).
   
  Will call &store_aggregates() to increment totals for attempts,   Will call &store_aggregates() to increment totals for attempts, 
  students, and corrects, if running user has student role.   students, and corrects, if running user has student role.
   
 =cut  =cut
   
   
Line 1146  sub finalize_storage { Line 1106  sub finalize_storage {
  delete(@Apache::lonhomework::results{@remove});   delete(@Apache::lonhomework::results{@remove});
  my ($symb,$courseid,$domain,$name) =    my ($symb,$courseid,$domain,$name) = 
     &Apache::lonnet::whichuser($given_symb);      &Apache::lonnet::whichuser($given_symb);
         my ($passback,$pbscope,$pbmap,$pbsymb,$pbtype,$crsdef,$ltinum,  
             $ltiref,$total,$possible,$dopassback);  
  if ($env{'request.state'} eq 'construct'    if ($env{'request.state'} eq 'construct' 
     || $symb eq ''      || $symb eq ''
     || $Apache::lonhomework::type eq 'practice') {      || $Apache::lonhomework::type eq 'practice') {
Line 1157  sub finalize_storage { Line 1115  sub finalize_storage {
       $namespace,'',$domain,$name);        $namespace,'',$domain,$name);
     &Apache::lonxml::debug('Construct Store return message:'.$result);      &Apache::lonxml::debug('Construct Store return message:'.$result);
  } else {   } else {
             my ($laststore,$checkedparts,@parts,%postcorrect,%record);              my ($laststore,$checkedparts,@parts,%postcorrect);
             if (($env{'user.name'} eq $name) && ($env{'user.domain'} eq $domain) &&              if (($env{'user.name'} eq $name) && ($env{'user.domain'} eq $domain) &&
                 (!$Apache::lonhomework::scantronmode) && (!defined($env{'form.grade_symb'})) &&                  (!$Apache::lonhomework::scantronmode) && (!defined($env{'form.grade_symb'})) &&
                 (!defined($env{'form.grade_courseid'}))) {                  (!defined($env{'form.grade_courseid'}))) {
                 if (($env{'request.lti.login'}) || ($env{'request.deeplink.login'})) {  
                     my ($map)=&Apache::lonnet::decode_symb($symb);  
                     $map = &Apache::lonnet::clutter($map);  
                     if ($env{'request.lti.login'}) {  
                         ($passback,$pbscope,$pbmap,$pbsymb,$ltinum,$ltiref) =  
                             &needs_lti_passback($courseid,$symb,$map);  
                     } elsif ($env{'request.deeplink.login'}) {  
                         ($passback,$pbscope,$pbmap,$pbsymb,$crsdef,$ltinum,$ltiref) =  
                             &needs_linkprot_passback($courseid,$symb,$map);  
                     }  
                 }  
                 if ($Apache::lonhomework::history{'version'}) {                  if ($Apache::lonhomework::history{'version'}) {
                     $laststore = $Apache::lonhomework::history{'version'}.'='.                      $laststore = $Apache::lonhomework::history{'version'}.'='.
                                  $Apache::lonhomework::history{'timestamp'};                                   $Apache::lonhomework::history{'timestamp'};
                 } else {                  } else {
                     $laststore = '0=0';                      $laststore = '0=0';
                 }                  }
                 %record = &Apache::lonnet::restore($symb,$courseid,$domain,$name);                  my %record = &Apache::lonnet::restore($symb,$courseid,$domain,$name);
                 if ($record{'version'}) {                  if ($record{'version'}) {
                     my ($newversion,$oldversion,$oldtimestamp);                      my ($newversion,$oldversion,$oldtimestamp);
                     if ($Apache::lonhomework::history{'version'}) {                      if ($Apache::lonhomework::history{'version'}) {
Line 1254  sub finalize_storage { Line 1201  sub finalize_storage {
                     }                      }
                 }                  }
             }              }
             if ($passback) {  
                 foreach my $key (keys(%Apache::lonhomework::results)) {  
                     if ($key =~ /^resource\.([^\.]+)\.solved$/) {  
                         my $part = $1;  
                         if ((($Apache::lonhomework::results{$key} =~ /^correct_/) ||  
                             ($Apache::lonhomework::results{$key} eq 'incorrect_attempted')) &&  
                             ($Apache::lonhomework::results{"resource.$part.tries"})) {  
                             $dopassback = 1;  
                             last;  
                         }  
                     }  
                 }  
             }  
             if (($dopassback) && ($pbscope eq 'resource') && ($pbsymb eq $symb)) {  
                 $total = 0;  
                 $possible = 0;  
                 my $navmap = Apache::lonnavmaps::navmap->new();  
                 if (ref($navmap)) {  
                     my $res = $navmap->getBySymb($symb);  
                     if (ref($res)) {  
                         my $partlist = $res->parts();  
                         if (ref($partlist) eq 'ARRAY') {  
                             foreach my $part (@{$partlist}) {  
                                 unless (exists($Apache::lonhomework::results{"resource.$part.solved"})) {  
                                     next if ($Apache::lonhomework::record{"resource.$part.solved"} =~/^excused/);  
                                     my $weight = &Apache::lonnet::EXT("resource.$part.weight",$symb);  
                                     $possible += $weight;  
                                     if (($record{'version'}) && (exists($record{"resource.$part.awarded"}))) {  
                                         my $awarded = $record{"resource.$part.awarded"};  
                                         if ($awarded) {  
                                             $total += $weight * $awarded;  
                                         }  
                                     }  
                                 }  
                             }  
                         }  
                     }  
                 }  
                 foreach my $key (keys(%Apache::lonhomework::results)) {  
                     if ($key =~ /^resource\.([^\.]+)\.awarded$/) {  
                         my $part = $1;  
                         my $weight = &Apache::lonnet::EXT("resource.$part.weight",$symb);  
                         $possible += $weight;  
                         my $awarded = $Apache::lonhomework::results{$key};  
                         if ($awarded) {  
                             $total += $weight * $awarded;  
                         }  
                     }  
                 }  
             }  
     &Apache::lonxml::debug('Store return message:'.$result);      &Apache::lonxml::debug('Store return message:'.$result);
             &store_aggregates($symb,$courseid);              &store_aggregates($symb,$courseid);
             if ($dopassback) {  
                 my $scoreformat = 'decimal';  
                 if (($env{'request.lti.login'}) || ($env{'request.deeplink.login'})) {  
                     if (ref($ltiref) eq 'HASH') {  
                         if ($ltiref->{'scoreformat'} =~ /^(decimal|ratio|percentage)$/) {  
                             $scoreformat = $1;  
                         }  
                     }  
                 }  
                 my ($pbid,$pburl,$pbtype);  
                 if ($env{'request.lti.login'}) {  
                     $pbid = $env{'request.lti.passbackid'};  
                     $pburl = $env{'request.lti.passbackurl'};  
                     $pbtype = 'lti';  
                 } elsif ($env{'request.deeplink.login'}) {  
                     $pbid = $env{'request.linkprotpbid'};  
                     $pburl = $env{'request.linkprotpburl'};  
                     $pbtype = 'linkprot';  
                 }  
                 my $ltigrade = {  
                                  'ltinum'   => $ltinum,  
                                  'lti'      => $ltiref,  
                                  'crsdef'   => $crsdef,  
                                  'cid'      => $courseid,  
                                  'uname'    => $env{'user.name'},  
                                  'udom'     => $env{'user.domain'},  
                                  'pbid'     => $pbid,  
                                  'pburl'    => $pburl,  
                                  'pbtype'   => $pbtype,  
                                  'scope'    => $pbscope,  
                                  'pbmap'    => $pbmap,  
                                  'pbsymb'   => $pbsymb,  
                                  'format'   => $scoreformat,  
                                };  
                 if ($pbscope eq 'resource') {  
                     $ltigrade->{'total'} = $total;  
                     $ltigrade->{'possible'} = $possible;  
                 }  
                 push(@Apache::lonhomework::ltipassback,$ltigrade);  
             }  
  }   }
     } else {      } else {
  &Apache::lonxml::debug('Nothing to store');   &Apache::lonxml::debug('Nothing to store');
Line 1353  sub finalize_storage { Line 1210  sub finalize_storage {
     return $result;      return $result;
 }  }
   
 sub needs_lti_passback {  
     my ($courseid,$symb,$map) = @_;  
     if (($env{'request.lti.passbackid'}) && ($env{'request.lti.passbackurl'})) {  
         if ($courseid =~ /^($LONCAPA::match_domain)_($LONCAPA::match_courseid)$/) {  
             my ($cdom,$cnum) = ($1,$2);  
             my %lti = &Apache::lonnet::get_domain_lti($cdom,'provider');  
             if (ref($lti{$env{'request.lti.login'}}) eq 'HASH') {  
                 if ($lti{$env{'request.lti.login'}}{'passback'}) {  
                     my $itemnum = $env{'request.lti.login'};  
                     my ($ltiscope,$ltiuri,$ltisymb) =  
                         &LONCAPA::ltiutils::lti_provider_scope($env{'request.lti.uri'},  
                                                                $cdom,$cnum,1);  
                     my ($passback,$ltimap);  
                     if ($ltiscope eq 'resource') {  
                         if ($ltisymb eq $symb) {  
                             $passback = 1;  
                         }  
                     } elsif ($ltiscope eq 'map') {  
                         if ($ltiuri eq $map) {  
                             $passback = 1;  
                             $ltimap = $map;  
                         }  
                     } elsif ($ltiscope eq 'course') {  
                         if (($env{'request.lti.uri'} eq "/$cdom/$cnum") || ($env{'request.lti.uri'} eq '')) {  
                             $passback = 1;  
                         }  
                     }  
                     return ($passback,$ltiscope,$ltimap,$ltisymb,$itemnum,$lti{$itemnum});  
                 }  
             }  
         }  
     }  
     return;  
 }  
   
 sub needs_linkprot_passback {  
     my ($courseid,$symb,$map) = @_;  
     if (($env{'request.linkprotpbid'}) && ($env{'request.linkprotpburl'})) {  
         if ($courseid =~ /^($LONCAPA::match_domain)_($LONCAPA::match_courseid)$/) {  
             my ($cdom,$cnum) = ($1,$2);  
             my ($deeplink_symb,$deeplink_map,$deeplink,$passback);  
             $deeplink_symb = &Apache::loncommon::deeplink_login_symb($cnum,$cdom);  
             if ($deeplink_symb) {  
                 if ($deeplink_symb =~ /\.(page|sequence)$/) {  
                     $deeplink_map = &Apache::lonnet::deversion((&Apache::lonnet::decode_symb($deeplink_symb))[2]);  
                     my $navmap = Apache::lonnavmaps::navmap->new();  
                     if (ref($navmap)) {  
                         $deeplink = $navmap->get_mapparam(undef,$deeplink_map,'0.deeplink');  
                     }  
                 } else {  
                     $deeplink = &Apache::lonnet::EXT('resource.0.deeplink',$deeplink_symb);  
                     $deeplink_map = &Apache::lonnet::deversion((&Apache::lonnet::decode_symb($deeplink_symb))[0]);  
                 }  
                 if (($deeplink ne '') && ($env{'request.linkprot'} ne '')) {  
                     my ($itemid,$tinyurl) = split(/:/,$env{'request.linkprot'});  
                     if ($itemid =~ /^(\d+)(c|d)$/) {  
                         my ($itemnum,$itemtype) = ($1,$2);  
                         my ($crsdef,$lti_in_use);  
                         if ($itemtype eq 'c') {  
                             $crsdef = 1;  
                             my %crslti = &Apache::lonnet::get_course_lti($cnum,$cdom,'provider');  
                             $lti_in_use = $crslti{$itemnum};  
                         } else {  
                             my %domlti = &Apache::lonnet::get_domain_lti($cdom,'linkprot');  
                             $lti_in_use = $domlti{$itemnum};  
                         }  
                         my ($state,$others,$listed,$scope,$protect,$display,$target,$exit) = split(/,/,$deeplink);  
                         my ($passback,$pbscope);  
                         if ($scope eq 'res') {  
                             if ($deeplink_symb eq $symb) {  
                                 $passback = 1;  
                                 $pbscope = 'resource';  
                             }  
                         } elsif ($scope eq 'map') {  
                             if (&Apache::lonnet::clutter($deeplink_map) eq $map) {  
                                 $passback = 1;  
                                 $pbscope = 'nonrec';  
                             }  
                         } elsif ($scope eq 'rec') {  
                             if (&Apache::lonnet::clutter($deeplink_map) eq $map) {  
                                 $passback = 1;  
                                 $pbscope = 'map';  
                             } else {  
                                 my @recurseup = &Apache::lonnet::get_map_hierarchy($map,$env{'request.course.id'});  
                                 if (grep(/^\Q$deeplink_map\E$/,@recurseup)) {  
                                     $passback = 1;  
                                     $pbscope = 'map';  
                                 }  
                             }  
                         }  
                         return ($passback,$pbscope,$deeplink_map,$deeplink_symb,$crsdef,$itemnum,$lti_in_use);  
                     }  
                 }  
             }  
         }  
     }  
 }  
   
 =pod  =pod
   
 =item check_correctness_changes()  =item check_correctness_changes()
Line 1585  sub store_aggregates { Line 1344  sub store_aggregates {
     }      }
 }  }
   
 sub access_status_msg {  
     my ($mode,$status,$symb,$target,$ipused,$accessmsg) = @_;  
     my $msg;  
     if ($target eq 'web') {  
         if ($status eq 'UNAVAILABLE') {  
             $msg.='<p class="LC_error">'.&mt('Unable to determine if this resource is open due to network problems. Please try again later.').'</p>';  
         } elsif ($status eq 'NOT_IN_A_SLOT') {  
             $msg.='<p class="LC_warning">'.&mt('You are not currently signed up to work at this time and/or place.').'</p>';  
         } elsif (($status eq 'RESERVABLE') || ($status eq 'RESERVABLE_LATER') ||  
                  ($status eq 'NOTRESERVABLE')) {  
             $msg.='<p class="LC_warning">'.&mt('Access requires reservation to work at specific time/place.').'</p>';  
         } elsif ($status ne 'NOT_YET_VIEWED') {  
             $msg.='<p class="LC_warning">'.&mt('Not open to be viewed').'</p>';  
         }  
         if ($status eq 'CLOSED' || $status eq 'INVALID_ACCESS') {  
             $msg.=&mt('The problem ').$accessmsg;  
         } elsif ($status eq 'UNCHECKEDOUT') {  
             $msg.=&checkout_msg();  
         } elsif ($status eq 'NOT_YET_VIEWED') {  
             $msg.=&firstaccess_msg($accessmsg,$symb);  
         } elsif ($status eq 'NOT_IN_A_SLOT') {  
             $msg.=&Apache::bridgetask::add_request_another_attempt_button("Sign up for time to work");  
         } elsif ($status eq 'RESERVABLE') {  
             $msg.=&mt('Available to make a reservation.').' '.&mt('Reservation window closes [_1].',  
                       &Apache::lonnavmaps::timeToHumanString($accessmsg,'end')).  
                       '<br />'.  
                       &Apache::bridgetask::add_request_another_attempt_button("Sign up for time to work");  
         } elsif ($status eq 'RESERVABLE_LATER') {  
             $msg.=&mt('Window to make a reservation will open [_1].',  
                       &Apache::lonnavmaps::timeToHumanString($accessmsg,'start'));  
         } elsif ($status eq 'NOTRESERVABLE') {  
             $msg.=&mt('Not available to make a reservation.');  
         }  
         $msg.='<br />';  
     } elsif ($target eq 'tex') {  
         my $startminipage = ($env{'form.problem_split'}=~/yes/i)? ''  
                              : '\begin{minipage}{\textwidth}';  
   
         $msg ='\noindent \vskip 1 mm '.  
               $startminipage.'\vskip 0 mm';  
         if ($status eq 'UNAVAILABLE') {  
             $msg.=&mt('Unable to determine if this resource is open due to network problems. Please try again later.');  
         } elsif ($status eq 'CLOSED' || $status eq 'INVALID_ACCESS') {  
             $msg.=&mt('Problem is not open to be viewed. It')." $accessmsg";  
         } else {  
             $msg.=&mt('Problem is not open to be viewed.');  
         }  
         $msg .= " \\vskip 0 mm ";  
     }  
     return $msg;  
 }  
   
 sub checkin_prompt {  
     my ($target,$slot_name,$slot,$type) = @_;  
     my $result;  
     if ($target eq 'web') {  
         $result = &Apache::bridgetask::proctor_validation_screen($slot);  
     } elsif ($target eq 'grade') {  
         if (!&Apache::bridgetask::proctor_check_auth($slot_name,$slot,$type)) {  
             $result = &mt('An error occurred during check-in');  
         }  
     }  
     return $result;  
 }  
   
 sub selfcheckin_resource {  
     my ($resource_due,$slot_name,$slot,$symb) = @_;  
     if ($slot_name ne '') {  
         my $checked_in =  
             $Apache::lonhomework::history{'resource.0.checkedin'};  
         if ($checked_in eq '') {  
             # unproctored slot access, self checkin  
             my $check = &Apache::bridgetask::check_in('problem',undef,undef,  
                                                       $slot_name);  
             if ($check =~ /^error: /) {  
                 &Apache::lonnet::logthis("Error during self-checkin of problem (symb: $symb) using slot: $slot_name");  
             } else {  
                 $checked_in = $Apache::lonhomework::results{"resource.0.checkedin"};  
             }  
         }  
         if ((ref($slot) eq 'HASH') && ($checked_in ne '')) {  
             if ($slot->{'starttime'} < time()) {  
                 if (!$resource_due) {  
                     $resource_due = $slot->{'endtime'};  
                 } elsif ($slot->{'endtime'} < $resource_due) {  
                     $resource_due = $slot->{'endtime'};  
                 }  
             }  
         }  
     }  
     return $resource_due;  
 }  
   
 sub checkout_msg {  sub checkout_msg {
     my %lt=&Apache::lonlocal::texthash(       my %lt=&Apache::lonlocal::texthash( 
  'resource'=>'The resource needs to be checked out',   'resource'=>'The resource needs to be checked out',
Line 1893  sub start_problem { Line 1559  sub start_problem {
         $target eq 'tex') {          $target eq 'tex') {
         if ($env{'form.markaccess'}) {          if ($env{'form.markaccess'}) {
             my @interval=&Apache::lonnet::EXT("resource.0.interval");              my @interval=&Apache::lonnet::EXT("resource.0.interval");
             my ($timelimit) = ($interval[0] =~ /^(\d+)/);              my $is_set = &Apache::lonnet::set_first_access($interval[1],$interval[0]);
             my $is_set = &Apache::lonnet::set_first_access($interval[1],$timelimit);  
             unless (($is_set eq 'ok') || ($is_set eq 'already_set')) {              unless (($is_set eq 'ok') || ($is_set eq 'already_set')) {
                 $firstaccres = $is_set;                  $firstaccres = $is_set;
             }              }
         }          }
   
           ($status,$accessmsg,$slot_name,$slot) =
               &Apache::lonhomework::check_slot_access('0','problem');
           push (@Apache::inputtags::status,$status);
     }      }
   
     if ($target eq 'web' || $target eq 'webgrade' || $target eq 'tex'      if ($target eq 'web' || $target eq 'webgrade' || $target eq 'tex'
Line 1906  sub start_problem { Line 1575  sub start_problem {
  ($result,$form_tag_start,$probpartlist) =   ($result,$form_tag_start,$probpartlist) =
     &page_start($target,$token,$tagstack,$parstack,$parser,$safeeval,      &page_start($target,$token,$tagstack,$parstack,$parser,$safeeval,
  $name);   $name);
     } elsif ((($target eq 'grade') && ($Apache::lonhomework::type eq 'randomizetry')) ||      } elsif (($target eq 'grade') && ($Apache::lonhomework::type eq 'randomizetry')) {
              ($target eq 'answer')) {  
         my ($symb)= &Apache::lonnet::whichuser();          my ($symb)= &Apache::lonnet::whichuser();
         if ($symb ne '') {          if ($symb ne '') {
             my $navmap = Apache::lonnavmaps::navmap->new();              my $navmap = Apache::lonnavmaps::navmap->new();
Line 1920  sub start_problem { Line 1588  sub start_problem {
         }          }
     }      }
   
     if (($target eq 'web') && ($env{'request.user_in_effect'})) {  
         &Apache::lonxml::get_all_text("/problem",$parser,$style);  
         return $result;  
     }  
   
     if ($target eq 'web' || $target eq 'grade' || $target eq 'answer' ||  
         $target eq 'tex') {  
   
         my ($symb)= &Apache::lonnet::whichuser();  
         ($status,$accessmsg,$slot_name,$slot) =  
             &Apache::lonhomework::check_slot_access('0','problem',$symb,$probpartlist);  
         push (@Apache::inputtags::status,$status);  
     }   
   
     if ($target eq 'tex' and $env{'request.symb'} =~ m/\.page_/) {$result='';}      if ($target eq 'tex' and $env{'request.symb'} =~ m/\.page_/) {$result='';}
   
     if ($target eq 'analyze') { my $rndseed=&setup_rndseed($safeeval,$target); }      if ($target eq 'analyze') { my $rndseed=&setup_rndseed($safeeval,$target); }
Line 1971  sub start_problem { Line 1625  sub start_problem {
             }              }
     $form_tag_start.='<input type="hidden" name="rndseed" value="'.      $form_tag_start.='<input type="hidden" name="rndseed" value="'.
  $rndseedval.'" />'.   $rndseedval.'" />'.
     '<button type="submit" name="resetdata"      '<input type="submit" name="resetdata"
                         value="new_problem_variation">'.&mt('New Problem Variation').'</button>';                               value="'.&mt('New Problem Variation').'" />';
     if (exists($env{'form.username'})) {      if (exists($env{'form.username'})) {
  $form_tag_start.=   $form_tag_start.=
     '<input type="hidden" name="username"      '<input type="hidden" name="username"
Line 2024  sub start_problem { Line 1678  sub start_problem {
     ( $status eq 'INVALID_ACCESS')) {      ( $status eq 'INVALID_ACCESS')) {
     my $bodytext=&Apache::lonxml::get_all_text("/problem",$parser,      my $bodytext=&Apache::lonxml::get_all_text("/problem",$parser,
        $style);         $style);
             if (($status eq 'NOT_YET_VIEWED') && ($firstaccres)) {      if ( $target eq "web" ) {
                 $result .= '<p class="LC_error">'.   my $msg;
                            &mt('A problem occurred when trying to start the timer.').'</p>';   if ($status eq 'UNAVAILABLE') {
             }      $msg.='<p class="LC_error">'.&mt('Unable to determine if this resource is open due to network problems. Please try again later.').'</p>';
             $result .= &access_status_msg('problem',$status,$symb,$target,'',$accessmsg);                  } elsif ($status eq 'NOT_IN_A_SLOT') {
                       $msg.='<p class="LC_warning">'.&mt('You are not currently signed up to work at this time and/or place.').'</p>';
                   } elsif (($status eq 'RESERVABLE') || ($status eq 'RESERVABLE_LATER') ||
                            ($status eq 'NOTRESERVABLE')) {
                       $msg.='<p class="LC_warning">'.&mt('Access requires reservation to work at specific time/place.').'</p>';
    } elsif ($status ne 'NOT_YET_VIEWED') {
       $msg.='<p class="LC_warning">'.&mt('Not open to be viewed').'</p>';
                   }
    if ($status eq 'CLOSED' || $status eq 'INVALID_ACCESS') {
       $msg.=&mt('The problem ').$accessmsg;
    } elsif ($status eq 'UNCHECKEDOUT') {
       $msg.=&checkout_msg();
    } elsif ($status eq 'NOT_YET_VIEWED') {
                       if ($firstaccres) {
                           $msg .= '<p class="LC_error">'.
                                   &mt('A problem occurred when trying to start the timer.').'</p>';
                       }
       $msg.=&firstaccess_msg($accessmsg,$symb);
    } elsif ($status eq 'NOT_IN_A_SLOT') {
       $msg.=&Apache::bridgetask::add_request_another_attempt_button("Sign up for time to work");
                   } elsif ($status eq 'RESERVABLE') {
                       $msg.=&mt('Available to make a reservation.').' '.&mt('Reservation window closes [_1].',
                                 &Apache::lonnavmaps::timeToHumanString($accessmsg,'end')).
                             '<br />'.
                             &Apache::bridgetask::add_request_another_attempt_button("Sign up for time to work");
                   } elsif ($status eq 'RESERVABLE_LATER') {
                       $msg.=&mt('Window to make a reservation will open [_1].',
                                 &Apache::lonnavmaps::timeToHumanString($accessmsg,'start'));
                   } elsif ($status eq 'NOTRESERVABLE') {
                       $msg.=&mt('Not available to make a reservation.');  
    }
    $result.=$msg.'<br />';
       } elsif ($target eq 'tex') {
    my $startminipage = ($env{'form.problem_split'}=~/yes/i)? ''
                       : '\begin{minipage}{\textwidth}';
    $result.='\noindent \vskip 1 mm '.
       $startminipage.'\vskip 0 mm';
    if ($status eq 'UNAVAILABLE') {
       $result.=&mt('Unable to determine if this resource is open due to network problems. Please try again later.').'\vskip 0 mm ';
    } else {
       $result.=&mt('Problem is not open to be viewed. It')." $accessmsg \\vskip 0 mm ";
    }
       }
  } elsif ($status eq 'NEEDS_CHECKIN') {   } elsif ($status eq 'NEEDS_CHECKIN') {
     my $bodytext=&Apache::lonxml::get_all_text("/problem",$parser,      my $bodytext=&Apache::lonxml::get_all_text("/problem",$parser,
        $style);         $style);
             $result .= &checkin_prompt($target,$slot_name,$slot,'problem');      if ($target eq 'web') {
    $result .= 
       &Apache::bridgetask::proctor_validation_screen($slot);
       } elsif ($target eq 'grade') {
    &Apache::bridgetask::proctor_check_auth($slot_name,$slot,
    'problem');
       }
  } elsif ($target eq 'web') {   } elsif ($target eq 'web') {
     if ($status eq 'CAN_ANSWER') {      if ($status eq 'CAN_ANSWER') {
                 $resource_due = &Apache::lonhomework::due_date(0, $env{'request.symb'});                  $resource_due = &Apache::lonhomework::due_date(0, $env{'request.symb'});
                 if ($slot_name ne '') {                  if ($slot_name ne '') {
                     $resource_due = &selfcheckin_resource($resource_due,$slot_name,$slot,                      my $checked_in =
                                                           $env{'request.symb'});                          $Apache::lonhomework::history{'resource.0.checkedin'};
                       if ($checked_in eq '') {
                           # unproctored slot access, self checkin
                           &Apache::bridgetask::check_in('problem',undef,undef,
                                                         $slot_name);
                           $checked_in =
                               $Apache::lonhomework::results{"resource.0.checkedin"};
                       }
                       if ((ref($slot) eq 'HASH') && ($checked_in ne '')) {
                           if ($slot->{'starttime'} < time()) {
                               if (!$resource_due) {
                                   $resource_due = $slot->{'endtime'};
                               } elsif ($slot->{'endtime'} < $resource_due) {
                                   $resource_due = $slot->{'endtime'};
                               }
                           }
                       }
                 }                  }
                 if ($resource_due) {                  if ($resource_due) {
                     my $time_left = $resource_due - time();                      my $time_left = $resource_due - time();
Line 2051  sub start_problem { Line 1769  sub start_problem {
       '<input type="hidden" name="submitted" value="yes" />';        '<input type="hidden" name="submitted" value="yes" />';
     # create a page header and exit      # create a page header and exit
     if ($env{'request.state'} eq "construct") {      if ($env{'request.state'} eq "construct") {
                 if ($env{'form.inhibitmenu'} eq 'yes') {   $result.= &problem_web_to_edit_header($env{'form.rndseed'});
                     # error messages can be useful in any case  
                     $result.= &Apache::lonxml::message_location();  
                 } else {  
     $result.= &problem_web_to_edit_header($env{'form.rndseed'});  
                 }  
                 if ($Apache::lonhomework::type eq 'practice') {                  if ($Apache::lonhomework::type eq 'practice') {
                     $result.= '<button type="submit" name="resetdata" '.                      $result.= '<input type="submit" name="resetdata" '.
                               'value="new_problem_variation">'.&mt('New Problem Variation').'</button>'.                                'value="'.&mt('New Problem Variation').'" />'.
                               &practice_problem_header().'<hr />';                                &practice_problem_header().'<hr />';
                 } elsif ($Apache::lonhomework::type eq 'randomizetry') {                  } elsif ($Apache::lonhomework::type eq 'randomizetry') {
                     my $reqtries = &Apache::lonnet::EXT("resource.$Apache::inputtags::part.randomizeontries");                      my $reqtries = &Apache::lonnet::EXT("resource.$Apache::inputtags::part.randomizeontries");
Line 2086  sub start_problem { Line 1799  sub start_problem {
  } elsif ($target eq 'tex') {   } elsif ($target eq 'tex') {
     $result .= 'INSERTTEXFRONTMATTERHERE';      $result .= 'INSERTTEXFRONTMATTERHERE';
     $result .= &select_metadata_hyphenation();      $result .= &select_metadata_hyphenation();
       
   
  }   }
     } elsif ($target eq 'edit') {      } elsif ($target eq 'edit') {
  $result .= $form_tag_start.&problem_edit_header();   $result .= $form_tag_start.&problem_edit_header();
Line 2205  sub end_problem { Line 1920  sub end_problem {
     }      }
  }   }
  $result =~ s/INSERTTEXFRONTMATTERHERE/$frontmatter/;   $result =~ s/INSERTTEXFRONTMATTERHERE/$frontmatter/;
     } elsif ($target eq 'web') {  
         if ($env{'request.user_in_effect'}) {  
             &reset_problem_globals('problem');  
             $result .= &Apache::lonhtmlcommon::set_compute_end_time();  
             return $result;  
         }  
     }      }
   
     my $status=$Apache::inputtags::status['-1'];      my $status=$Apache::inputtags::status['-1'];
Line 2365  sub start_library { Line 2074  sub start_library {
   '<input type="hidden" name="submitted" value="yes" />';    '<input type="hidden" name="submitted" value="yes" />';
  $result.=&problem_web_to_edit_header($rndseed);   $result.=&problem_web_to_edit_header($rndseed);
         if ($Apache::lonhomework::type eq 'practice') {          if ($Apache::lonhomework::type eq 'practice') {
             $result.= '<button type="submit" name="resetdata" '.              $result.= '<input type="submit" name="resetdata" '.
                       'value="new_problem_variation">'.&mt('New Problem Variation').'</button>'.                        'value="'.&mt('New Problem Variation').'" />'.
                       &practice_problem_header().'<hr />';                        &practice_problem_header().'<hr />';
         }          }
     }      }
Line 3149  sub end_part { Line 2858  sub end_part {
     $gradestatus='';      $gradestatus='';
  }   }
  $result.=$gradestatus;   $result.=$gradestatus;
         if ($$tagstack[-2] eq 'td' and $target eq 'tex') {   if ($$tagstack[-2] eq 'td' and $target eq 'tex') {$result.='\end{minipage}';} 
             if (not $env{'form.problem_split'}=~/yes/) {  
                 $result.='\end{minipage}';  
             }  
         }  
     } elsif ($target eq 'edit') {      } elsif ($target eq 'edit') {
  $result.=&Apache::edit::end_table();   $result.=&Apache::edit::end_table();
     } elsif ($target eq 'modified') {      } elsif ($target eq 'modified') {
Line 3318  sub end_startouttext { Line 3023  sub end_startouttext {
                  .&Apache::edit::deletelist($target,$token)                   .&Apache::edit::deletelist($target,$token)
                  .'</span></td>'                   .'</span></td>'
          .'<td><span id="math_'.$areaid.'">';           .'<td><span id="math_'.$areaid.'">';
  if (&Apache::loncommon::nocodemirror()) {   if ($env{'environment.nocodemirror'}) {
     $result.=&Apache::lonhtmlcommon::dragmath_button($areaid,1);      $result.=&Apache::lonhtmlcommon::dragmath_button($areaid,1);
  } else {   } else {
     $result.='&nbsp;';      $result.='&nbsp;';

Removed from v.1.512.2.24.2.15  
changed lines
  Added in v.1.512.2.25


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