Diff for /loncom/interface/lonparmset.pm between versions 1.620 and 1.624

version 1.620, 2023/05/22 21:10:55 version 1.624, 2025/06/30 21:12:21
Line 329  use Apache::lonnavmaps; Line 329  use Apache::lonnavmaps;
 use Apache::longroup;  use Apache::longroup;
 use Apache::lonrss;  use Apache::lonrss;
 use HTML::Entities;  use HTML::Entities;
   use POSIX qw (floor);
 use Text::Wrap();  use Text::Wrap();
 use LONCAPA qw(:DEFAULT :match);  use LONCAPA qw(:DEFAULT :match);
   
Line 992  sub valout { Line 993  sub valout {
             $result=' ';              $result=' ';
         }          }
     } else {      } else {
         if ($type eq 'date_interval') {          if (($type eq 'date_interval') || ($type eq 'string_grace')) {
             my ($totalsecs,$donesuffix) = split(/_/,$value,2);              if ($type eq 'string_grace') {
             my ($usesdone,$donebuttontext,$proctor,$secretkey);                  my @items;
             if ($donesuffix =~ /^done\:([^\:]+)\:(.*)$/) {                  if ($value =~ /,/) {
                 $donebuttontext = $1;                      @items = split(/,/,$value);
                 (undef,$proctor,$secretkey) = split(/_/,$2);  
                 $usesdone = 'done';  
             } elsif ($donesuffix =~ /^done(|_.+)$/) {  
                 $donebuttontext = &mt('Done');  
                 ($usesdone,$proctor,$secretkey) = split(/_/,$donesuffix);  
             }  
             my ($sec,$min,$hour,$mday,$mon,$year)=gmtime($totalsecs);  
             my @timer;  
             $year=$year-70;  
             $mday--;  
             if ($year) {  
 #               $result.=&mt('[quant,_1,yr]',$year).' ';  
                 push(@timer,&mt('[quant,_1,yr]',$year));  
             }  
             if ($mon) {  
 #               $result.=&mt('[quant,_1,mth]',$mon).' ';  
                 push(@timer,&mt('[quant,_1,mth]',$mon));  
             }  
             if ($mday) {  
 #               $result.=&mt('[quant,_1,day]',$mday).' ';  
                 push(@timer,&mt('[quant,_1,day]',$mday));  
             }  
             if ($hour) {  
 #               $result.=&mt('[quant,_1,hr]',$hour).' ';  
                 push(@timer,&mt('[quant,_1,hr]',$hour));  
             }  
             if ($min) {  
 #               $result.=&mt('[quant,_1,min]',$min).' ';  
                 push(@timer,&mt('[quant,_1,min]',$min));  
             }  
             if ($sec) {  
 #               $result.=&mt('[quant,_1,sec]',$sec).' ';  
                 push(@timer,&mt('[quant,_1,sec]',$sec));  
             }  
 #           $result=~s/\s+$//;  
             if (!@timer) { # Special case: all entries 0 -> display "0 secs" intead of empty field to keep this field editable  
                 push(@timer,&mt('[quant,_1,sec]',0));  
             }  
             $result.=join(", ",@timer);  
             if ($usesdone eq 'done') {  
                 if ($secretkey) {  
                     $result .= ' '.&mt('+ "[_1]" with proctor key: [_2]',$donebuttontext,$secretkey);    
                 } else {                  } else {
                     $result .= ' + "'.$donebuttontext.'"';                      @items = ($value);
                   }
                   foreach my $item (@items) {
                       if ($item =~ /^\d+:(0|1)\.?\d*:(0|1)$/) {
                           my ($totalsecs,$fraction,$grad) = split(/:/,$item);
                           $result .= &grace_to_humanstr($totalsecs);
                           if (($fraction >=0) && ($fraction <=1)) {
                               $result .= '&nbsp;|&nbsp;'.$fraction.'&nbsp;'.&mt('pts');
                               if ($grad == 1) {
                                   $result .= '&nbsp;('.&mt('gradual').')';
                               }
                           }
                           $result .= ', ';
                       }
                   }
                   $result =~ s/, $//;
               } else {
                   my ($totalsecs,$donesuffix) = split(/_/,$value,2);
                   $result = &interval_to_humanstr($totalsecs);
                   my ($usesdone,$donebuttontext,$proctor,$secretkey);
                   if ($donesuffix =~ /^done\:([^\:]+)\:(.*)$/) {
                       $donebuttontext = $1;
                       (undef,$proctor,$secretkey) = split(/_/,$2);
                       $usesdone = 'done';
                   } elsif ($donesuffix =~ /^done(|_.+)$/) {
                       $donebuttontext = &mt('Done');
                       ($usesdone,$proctor,$secretkey) = split(/_/,$donesuffix);
                   }
                   if ($usesdone eq 'done') {
                       if ($secretkey) {
                           $result .= ' '.&mt('+ "[_1]" with proctor key: [_2]',$donebuttontext,$secretkey);
                       } else {
                           $result .= ' + "'.$donebuttontext.'"';
                       }
                 }                  }
             }              }
         } elsif (&isdateparm($type)) {          } elsif (&isdateparm($type)) {
Line 1055  sub valout { Line 1047  sub valout {
     return $result;      return $result;
 }  }
   
   sub interval_to_humanstr {
       my ($totalsecs) = @_;
       my ($sec,$min,$hour,$mday,$mon,$year)=gmtime($totalsecs);
       my @timer;
       $year=$year-70;
       $mday--;
       if ($year) {
           push(@timer,&mt('[quant,_1,yr]',$year));
       }
       if ($mon) {
           push(@timer,&mt('[quant,_1,mth]',$mon));
       }
       if ($mday) {
           push(@timer,&mt('[quant,_1,day]',$mday));
       }
       if ($hour) {
           push(@timer,&mt('[quant,_1,hr]',$hour));
       }
       if ($min) {
           push(@timer,&mt('[quant,_1,min]',$min));
       }
       if ($sec) {
           push(@timer,&mt('[quant,_1,sec]',$sec));
       }
       if (!@timer) { # Special case: all entries 0 -> display "0 secs" intead of empty field to keep this field editable
           push(@timer,&mt('[quant,_1,sec]',0));
       }
       return '<span style="white-space:nowrap">'.join('</span>, <span style="white-space:nowrap">',@timer).'</span>';
   }
   
   sub grace_to_humanstr {
       my ($totalsecs) = @_;
       my @timer;
       my $weeks = floor($totalsecs/604800);
       $totalsecs -= $weeks*604800;
       my $days = floor($totalsecs/86400);
       $totalsecs -= $days*86400;
       my $hours = floor($totalsecs/3600);
       $totalsecs -= $hours*3600;
       my $mins= floor($totalsecs/60);
       $totalsecs -= $mins*60;
       if ($weeks) {
           push(@timer,&mt('[quant,_1,wk]',$weeks));
       }
       if ($days) {
           push(@timer,&mt('[quant,_1,day]',$days));
       }
       if ($hours) {
           push(@timer,&mt('[quant,_1,hr]',$hours));
       }
       if ($mins) {
           push(@timer,&mt('[quant,_1,min]',$mins));
       }
       if (!@timer) { # Special case: all entries 0 -> display "0 mins" intead of empty field to keep this field editable
           push(@timer,&mt('[quant,_1,min]',0));
       }
       return '<span style="white-space:nowrap">'.join('</span>, <span style="white-space:nowrap">',@timer).'</span>';
   }
   
 # Returns HTML containing a link on a parameter value, for table mode.  # Returns HTML containing a link on a parameter value, for table mode.
 # The link uses the javascript function 'pjump'.  # The link uses the javascript function 'pjump'.
Line 1245  function validateParms() { Line 1295  function validateParms() {
     var ipRegExp = /^setip/;      var ipRegExp = /^setip/;
     var ipallowRegExp = /^setipallow_/;      var ipallowRegExp = /^setipallow_/;
     var ipdenyRegExp = /^setipdeny_/;       var ipdenyRegExp = /^setipdeny_/; 
       var graceRegExp = /^setgrace_/;
     var deeplinkRegExp = /^deeplink_/;      var deeplinkRegExp = /^deeplink_/;
     var dlListScopeRegExp = /^deeplink_(state|others|listing|scope)_/;       var dlListScopeRegExp = /^deeplink_(state|others|listing|scope)_/; 
     var dlLinkProtectRegExp = /^deeplink_protect_/;      var dlLinkProtectRegExp = /^deeplink_protect_/;
Line 1257  function validateParms() { Line 1308  function validateParms() {
     var dlExitRegExp = /^deeplink_exit_/;      var dlExitRegExp = /^deeplink_exit_/;
     var dlExitTextRegExp = /^deeplink_exittext_/;      var dlExitTextRegExp = /^deeplink_exittext_/;
     var patternIP = /[\[\]\*\.a-zA-Z\d\-]+/;      var patternIP = /[\[\]\*\.a-zA-Z\d\-]+/;
       var patternGrace = /^\d+:(0|1)\.?\d*:(0|1)\$/;
     var numelements = document.parmform.elements.length;      var numelements = document.parmform.elements.length;
     if ((typeof(numelements) != 'undefined') && (numelements != null)) {      if ((typeof(numelements) != 'undefined') && (numelements != null)) {
         if (numelements) {          if (numelements) {
Line 1461  function validateParms() { Line 1513  function validateParms() {
                             }                              }
                         }                          }
                     }                      }
                   } else if (graceRegExp.test(name)) {
                       var identifier = name.replace(graceRegExp,'');
                       var divElem = document.parmform.elements[i].closest('div'); 
                       var timeSels = divElem.getElementsByTagName("select");
                       var total = 0;
                       var numnotnull = 0;
                       if (timeSels.length) {
                            for (var j=0; j<timeSels.length; j++) {
                               var sname = timeSels[j].getAttribute('name');
                               var value = timeSels[j].options[timeSels[j].selectedIndex].value;
                               if ((value !== null) && (value !== '') && (value !== 'undefined')) {
                                   numnotnull ++;
                                   var poss = parseInt(value);
                                   if (sname == 'weeks_'+identifier) {
                                       if ((poss > 0) && (poss <= 52)) {
                                           total += (poss * 604800);
                                       }
                                   } else if (sname == 'days_'+identifier) {
                                       if ((poss > 0) && (poss <= 6)) {
                                           total += (poss * 86400); 
                                       }
                                   } else if (sname == 'hours_'+identifier) {
                                       if ((poss > 0) && (poss < 24)) {
                                           total += (poss * 3600);
                                       }
                                   } else if (sname == 'minutes_'+identifier) {
                                       if ((poss > 0) && (poss < 60)) {
                                           total += (poss * 60);
                                       }
                                   }
                               }
                           }
                       }
                       if (!numnotnull) {
                           total = '';
                       }
                       var inputElems = divElem.getElementsByTagName("input");
                       var frac = '';
                       var grad = '';
                       if (inputElems.length) {
                           for (var j=0; j<inputElems.length; j++) {
                               var iname = inputElems[j].getAttribute('name');
                               if (iname == 'frac_'+identifier) {
                                   var ival = inputElems[j].value;
                                   ival.trim();
                                   if ((ival != '') && (value != 'undefined')) {
                                       var poss = parseFloat(ival);
                                       if ((typeof poss === 'number') && (!isNaN(poss))) {
                                           if ((poss => 0) && (poss <= 1)) {
                                               frac = poss;
                                               numnotnull ++;
                                           }
                                       }
                                   }
                               } else if (iname == 'grad_'+identifier) {
                                   if (inputElems[j].checked) {
                                       grad = 1;
                                   } else {
                                       grad = 0;
                                   }
                               }
                           }
                       }
                       if (numnotnull) {
                           var possgrace = total+':'+frac+':'+grad;   
                           if (patternGrace.test(possgrace)) {
                               document.parmform.elements[i].value = possgrace;
                               if (document.parmform.elements['set_'+identifier].value) {
                                   document.parmform.elements['set_'+identifier].value += ',';
                               }
                               document.parmform.elements['set_'+identifier].value += document.parmform.elements[i].value;
                           } else {
                               if (frac == '') {
                                   alert('Grace Period Past-Due: enter partial credit (number between 0 and 1.0).');
                                   return false;
                               } else {
                                   alert('Grace Period Past-Due: select a number in at least one of the time past due select boxes, or delete the value for partial credit.');
                                   return false;
                               }
                           }
                       }
                 }                  }
             }              }
         }          }
Line 1513  sub ipacc_boxes_js  { Line 1646  sub ipacc_boxes_js  {
 END  END
 }  }
   
   sub grace_js {
       my %lt = &grace_titles();
       &js_escape(\%lt);
       my $overdue = '<fieldset class="LC_grace"><legend>'.$lt{'sinc'}.'</legend>';
       foreach my $which (['weeks', 604800, 52],
                          ['days', 86400, 6],
                          ['hours', 3600, 23],
                          ['minutes', 60, 59]) {
           my ($name, $factor, $max) = @{ $which };
           my %select = ((map {$_ => $_} (0..$max)),
                         'select_form_order' => [0..$max]);
           unshift(@{$select{'select_form_order'}},'');
           $select{''} = '';
           my $selector = &Apache::loncommon::select_form('',$name."_'+identifier+'",
                                                          \%select);
           $selector =~ s/([\r\n\f]+)//g;
           $overdue .= $selector.'&nbsp;'.$lt{$name}.('&nbsp;'x2).' ';
       }
       $overdue .= '</fieldset>';
       return <<"END";
   \$(document).ready(function() {
       var wrapper         = \$(".LC_string_grace_wrap");
       var add_button      = \$(".LC_add_grace_button");
       var graceRegExp     = /^LC_string_grace_/;
   
       \$(add_button).click(function(e){
           e.preventDefault();
           var identifier = \$(this).closest("div").attr("id");
           identifier = identifier.replace(graceRegExp,'');
           \$(this).closest('div').find('.LC_string_grace_inner').append('<div><input type="hidden" name="setgrace_'+identifier+'" value="" />$overdue<fieldset class="LC_grace"><legend>$lt{pcr}</legend><input type="text" size="3" name="frac_'+identifier+'" value="" />&nbsp;&nbsp;<label><input type="checkbox" value="1" name="grad_'+identifier+'" />$lt{grad}</label></fieldset><a href="#" class="LC_remove_grace">$lt{remo}</a></div>');
       });
   
       \$(wrapper).delegate(".LC_remove_grace","click", function(e){
           e.preventDefault(); \$(this).closest("div").remove();
       })
   });
   
   
   END
   }
   
 # Javascript function toggleSecret, for overview mode.  # Javascript function toggleSecret, for overview mode.
 sub done_proctor_js {  sub done_proctor_js {
     my $defaultdone = &mt('Done');      my $defaultdone = &mt('Done');
Line 2441  sub lookUpTableParameter { Line 2615  sub lookUpTableParameter {
         'opendate' => 'time_settings',          'opendate' => 'time_settings',
         'duedate' => 'time_settings',          'duedate' => 'time_settings',
         'answerdate' => 'time_settings',          'answerdate' => 'time_settings',
           'grace' => 'time_settings',
         'interval' => 'time_settings',          'interval' => 'time_settings',
         'contentopen' => 'time_settings',          'contentopen' => 'time_settings',
         'contentclose' => 'time_settings',          'contentclose' => 'time_settings',
Line 2482  sub lookUpTableParameter { Line 2657  sub lookUpTableParameter {
         'lenient' => 'grading',          'lenient' => 'grading',
         'retrypartial' => 'tries',          'retrypartial' => 'tries',
         'discussvote'  => 'misc',          'discussvote'  => 'misc',
           'texdisplay' => 'misc',
         'examcode' => 'high_level_randomization',          'examcode' => 'high_level_randomization',
     );      );
 }  }
Line 3179  sub standardkeyorder { Line 3355  sub standardkeyorder {
     return ('parameter_0_opendate' => 1,      return ('parameter_0_opendate' => 1,
         'parameter_0_duedate' => 2,          'parameter_0_duedate' => 2,
         'parameter_0_answerdate' => 3,          'parameter_0_answerdate' => 3,
         'parameter_0_interval' => 4,          'parameter_0_grace' => 4,
         'parameter_0_weight' => 5,          'parameter_0_interval' => 5,
         'parameter_0_maxtries' => 6,          'parameter_0_weight' => 6,
         'parameter_0_hinttries' => 7,          'parameter_0_maxtries' => 7,
         'parameter_0_contentopen' => 8,          'parameter_0_hinttries' => 8,
         'parameter_0_contentclose' => 9,          'parameter_0_contentopen' => 9,
         'parameter_0_type' => 10,          'parameter_0_contentclose' => 10,
         'parameter_0_problemstatus' => 11,          'parameter_0_type' => 11,
         'parameter_0_hiddenresource' => 12,          'parameter_0_problemstatus' => 12,
         'parameter_0_hiddenparts' => 13,          'parameter_0_hiddenresource' => 13,
         'parameter_0_display' => 14,          'parameter_0_hiddenparts' => 14,
         'parameter_0_ordered' => 15,          'parameter_0_display' => 15,
         'parameter_0_tol' => 16,          'parameter_0_ordered' => 16,
         'parameter_0_sig' => 17,          'parameter_0_tol' => 17,
         'parameter_0_turnoffunit' => 18,          'parameter_0_sig' => 18,
         'parameter_0_discussend' => 19,          'parameter_0_turnoffunit' => 19,
         'parameter_0_discusshide' => 20,          'parameter_0_discussend' => 20,
         'parameter_0_discussvote' => 21,          'parameter_0_discusshide' => 21,
         'parameter_0_printstartdate'  =>  22,          'parameter_0_discussvote' => 22,
         'parameter_0_printenddate' =>  23);          'parameter_0_printstartdate' => 23,
           'parameter_0_printenddate' => 24);
 }  }
   
   
Line 3711  sub assessparms { Line 3888  sub assessparms {
             'date_interval','int','float','string','string_lenient',              'date_interval','int','float','string','string_lenient',
             'string_examcode','string_deeplink','string_discussvote',              'string_examcode','string_deeplink','string_discussvote',
             'string_useslots','string_problemstatus','string_ip',              'string_useslots','string_problemstatus','string_ip',
             'string_questiontype') {              'string_questiontype','string_tex','string_grace') {
         $r->print('<input type="hidden" value="'.          $r->print('<input type="hidden" value="'.
             &HTML::Entities::encode($env{'form.recent_'.$item},'"&<>').              &HTML::Entities::encode($env{'form.recent_'.$item},'"&<>').
             '" name="recent_'.$item.'" />');              '" name="recent_'.$item.'" />');
Line 4380  sub readdata { Line 4557  sub readdata {
 # Stores parameter data, using form parameters directly.  # Stores parameter data, using form parameters directly.
 #  #
 # Uses the following form parameters. The variable part in the names is a resourcedata key (except for a modification for user data).  # Uses the following form parameters. The variable part in the names is a resourcedata key (except for a modification for user data).
 # set_* (except settext, setipallow, setipdeny, setdeeplink) - set a parameter value  # set_* (except settext, setipallow, setipdeny, setdeeplink, setgrace) - set a parameter value
 # del_* - remove a parameter  # del_* - remove a parameter
 # datepointer_* - set a date parameter (value is key_* refering to a set of other form parameters)  # datepointer_* - set a date parameter (value is key_* refering to a set of other form parameters)
 # dateinterval_* - set a date interval parameter (value refers to more form parameters)  # dateinterval_* - set a date interval parameter (value refers to more form parameters)
Line 4413  sub storedata { Line 4590  sub storedata {
             my $cmd=$1;              my $cmd=$1;
             my $thiskey=$2;              my $thiskey=$2;
             my ($altkey,$recursive,$tkey,$tkeyrec,$tkeynonrec);              my ($altkey,$recursive,$tkey,$tkeyrec,$tkeynonrec);
             next if ($cmd eq 'rec' || $cmd eq 'settext' || $cmd eq 'setipallow' || $cmd eq 'setipdeny' || $cmd eq 'setdeeplink');              next if ($cmd eq 'rec' || $cmd eq 'settext' || $cmd eq 'setipallow' || $cmd eq 'setipdeny' || $cmd eq 'setdeeplink' || $cmd eq 'setgrace');
             if ((($cmd eq 'set') || ($cmd eq 'datepointer') || ($cmd eq 'dateinterval') || ($cmd eq 'del')) &&               if ((($cmd eq 'set') || ($cmd eq 'datepointer') || ($cmd eq 'dateinterval') || ($cmd eq 'del')) && 
                  ($thiskey =~ /(?:sequence|page)\Q___(all)\E/)) {                   ($thiskey =~ /(?:sequence|page)\Q___(all)\E/)) {
                 unless ($thiskey =~ /(encrypturl|hiddenresource)$/) {                  unless ($thiskey =~ /(encrypturl|hiddenresource)$/) {
Line 4472  sub storedata { Line 4649  sub storedata {
                         if ($thiskey =~ /\.retrypartial$/) {                          if ($thiskey =~ /\.retrypartial$/) {
                             $name = 'retrypartial';                              $name = 'retrypartial';
                         }                          }
                       } elsif ($typeof eq 'string_tex') {
                           $name = 'texdisplay';
                     }                      }
                 } elsif ($cmd eq 'datepointer') {                  } elsif ($cmd eq 'datepointer') {
                     $data=&Apache::lonhtmlcommon::get_date_from_form($env{$key});                      $data=&Apache::lonhtmlcommon::get_date_from_form($env{$key});
Line 5176  sub get_date_interval_from_form { Line 5355  sub get_date_interval_from_form {
     return $seconds;      return $seconds;
 }  }
   
   
 # Returns HTML to enter a text value for a parameter.  # Returns HTML to enter a text value for a parameter.
 #  #
 # @param {string} $thiskey - parameter key  # @param {string} $thiskey - parameter key
Line 5510  sub string_deeplink_selector { Line 5688  sub string_deeplink_selector {
     return $output;      return $output;
 }  }
   
   sub string_grace_selector {
       my ($thiskey, $showval, $readonly) = @_;
       my $addmore;
       unless ($readonly) {
           $addmore = "\n".'<button class="LC_add_grace_button">'.&mt('Add more').'</button>';
       }
       my $output = '<input type="hidden" name="set_'.$thiskey.'" value="" />'.
                    '<div class="LC_string_grace_wrap" id="LC_string_grace_'.$thiskey.'">'."\n".
                    '<div class="LC_string_grace_inner">'."\n";
       if ($showval ne '') {
           my @current;
           if ($showval =~ /,/) {
               @current = split(/,/,$showval);
           } else {
               @current = ($showval);
           }
           my $num = scalar(@current);
           foreach my $item (@current) {
               my ($delta,$fraction,$gradational) = split(/:/,$item);
               if (($delta =~ /^\d+$/) && ($fraction =~ /^(0|1)\.?\d*$/) && 
                   (($gradational eq 1) || ($gradational eq '0'))) {
                   my $gradchk = '';
                   if ($gradational) {
                       $gradchk = ' checked="checked"';
                   }
                   $output .= &grace_form($thiskey,$delta,$fraction,$gradchk,
                                          $readonly);
               }
           }
       } elsif (!$readonly) {
           $output .= &grace_form($thiskey,'','','',$readonly);
       }
       $output .= '</div>'.$addmore.'</div>';
       return $output;
   }
   
   sub grace_form {
       my ($thiskey,$delta,$fraction,$gradchkon,$readonly) = @_;
       my $disabled;
       if ($readonly) {
           $disabled = ' disabled="disabled"';
       }
       my %lt = &grace_titles();
       my $output = '<div><input type="hidden" name="setgrace_'.$thiskey.'" value="" />'.
                    '<fieldset class="LC_grace"><legend>'.$lt{'sinc'}.'</legend>';
       foreach my $which (['weeks', 604800, 52],
                          ['days', 86400, 6],
                          ['hours', 3600, 23],
                          ['minutes', 60, 59]) {
           my ($name, $factor, $max) = @{ $which };
           my $amount;
           my %select = ((map {$_ => $_} (0..$max)),
                         'select_form_order' => [0..$max]);
           if ($delta eq '') {
               unshift(@{$select{'select_form_order'}},'');
               $select{''} = '';
               $amount = '';
           } else {
               $amount = int($delta/$factor);
               $delta %= $factor;
           }
           $output .= &Apache::loncommon::select_form($amount,$name.'_'.$thiskey,
                                                      \%select,'',$readonly);
           $output .= '&nbsp;'.$lt{$name}.'&nbsp;&nbsp; ';
       }
       $output .= '</fieldset>'.
                  '<fieldset class="LC_grace"><legend>'.$lt{'pcr'}.'</legend>'.
                  '<input type="text" size="3" name="frac_'.$thiskey.'" value="'.$fraction.'"'.$disabled.' />'.
                  '&nbsp;&nbsp;<label><input type="checkbox" value="1" name="grad_'.$thiskey.'"'.$gradchkon.$disabled.' />'.
                  $lt{'grad'}.'</label></fieldset>';
       unless ($readonly) {
           $output .= '<a href="#" class="LC_remove_grace">'.$lt{'remo'}.'</a>';
       }
       $output .= '</div>'."\n";
       return $output;
   }
   
   sub grace_titles {
       return &Apache::lonlocal::texthash (
                                            sinc => 'Time past due',
                                            remo => 'Remove',
                                            pcr => 'Partial credit',
                                            grad => 'gradual',
                                            weeks => 'weeks',
                                            days => 'days',
                                            hours => 'hours',
                                            minutes => 'minutes',
       );
   }
   
 { # block using some constants related to parameter types (overview mode)  { # block using some constants related to parameter types (overview mode)
   
Line 5546  my %strings = Line 5813  my %strings =
                  ['_denyfrom_','Hostname(s) or IP(s) from which access is disallowed']],                    ['_denyfrom_','Hostname(s) or IP(s) from which access is disallowed']], 
      'string_deeplink'       'string_deeplink'
              => [['on','Set choices for link protection, resource listing, access scope, shown menu items, embedding, and exit link']],               => [['on','Set choices for link protection, resource listing, access scope, shown menu items, embedding, and exit link']],
        'string_tex'
                => [['tth', 'tth (TeX to HTML)'],
                    ['mathjax', 'MathJax']],
        'string_grace'
                => [['on','Set grading scale and grace period for submissions after due date']],
     );      );
         
   
Line 5557  my %stringmatches = ( Line 5829  my %stringmatches = (
                   ['_denyfrom_','\!']],                    ['_denyfrom_','\!']],
          'string_deeplink'           'string_deeplink'
               => [['on','^(only|off|both)\,(hide|unhide)\,(full|absent|grades|details|datestatus)\,(res|map|rec)\,(none|key\:\w+|ltic\:\d+|ltid\:\d+)\,(\d+|)\,_(self|top),(yes|url|no)(|:[^:;\'",]+)$']],                => [['on','^(only|off|both)\,(hide|unhide)\,(full|absent|grades|details|datestatus)\,(res|map|rec)\,(none|key\:\w+|ltic\:\d+|ltid\:\d+)\,(\d+|)\,_(self|top),(yes|url|no)(|:[^:;\'",]+)$']],
            'string_grace'
                 => [['on','^\d+,(0|1)\.?\d*,(0|1)']],
     );      );
   
 my %stringtypes = (  my %stringtypes = (
Line 5567  my %stringtypes = ( Line 5841  my %stringtypes = (
                     examcode     => 'string_examcode',                      examcode     => 'string_examcode',
                     acc          => 'string_ip',                      acc          => 'string_ip',
                     deeplink     => 'string_deeplink',                      deeplink     => 'string_deeplink',
                       grace        => 'string_grace',
                       texdisplay   => 'string_tex',
                   );                    );
   
 # Returns the possible values and titles for a given string type, or undef if there are none.  # Returns the possible values and titles for a given string type, or undef if there are none.
Line 5627  sub string_selector { Line 5903  sub string_selector {
             ($thistype eq 'string_discussvote') ||              ($thistype eq 'string_discussvote') ||
             ($thistype eq 'string_ip') ||              ($thistype eq 'string_ip') ||
             ($thistype eq 'string_deeplink') ||              ($thistype eq 'string_deeplink') ||
               ($thistype eq 'string_tex') ||
               ($thistype eq 'string_grace') ||
             ($name eq 'retrypartial')) {              ($name eq 'retrypartial')) {
         my ($got_chostname,$chostname,$cmajor,$cminor);           my ($got_chostname,$chostname,$cmajor,$cminor); 
         foreach my $possibilities (@{ $strings{$thistype} }) {          foreach my $possibilities (@{ $strings{$thistype} }) {
Line 5664  sub string_selector { Line 5942  sub string_selector {
     }      }
   
     if ($thistype eq 'string_ip') {      if ($thistype eq 'string_ip') {
         return &string_ip_selector($thiskey,$showval,$readonly);           return &string_ip_selector($thiskey,$showval,$readonly);
       } elsif ($thistype eq 'string_grace') {
           return &string_grace_selector($thiskey,$showval,$readonly);
     } elsif ($thistype eq 'string_deeplink') {      } elsif ($thistype eq 'string_deeplink') {
         return &string_deeplink_selector($thiskey,$showval,$readonly);          return &string_deeplink_selector($thiskey,$showval,$readonly);
     }      }
Line 6103  sub newoverview { Line 6383  sub newoverview {
             &toggleparmtextbox_js()."\n".              &toggleparmtextbox_js()."\n".
             &validateparms_js()."\n".              &validateparms_js()."\n".
             &ipacc_boxes_js()."\n".              &ipacc_boxes_js()."\n".
               &grace_js()."\n".
             &done_proctor_js()."\n".              &done_proctor_js()."\n".
             &deeplink_js()."\n".              &deeplink_js()."\n".
 '// ]]>  '// ]]>
Line 6115  sub newoverview { Line 6396  sub newoverview {
     $r->print($start_page.$breadcrumbs);      $r->print($start_page.$breadcrumbs);
     &startSettingsScreen($r,'parmset',$crstype);      &startSettingsScreen($r,'parmset',$crstype);
     $r->print(<<ENDOVER);      $r->print(<<ENDOVER);
 <form method="post" action="/adm/parmset?action=newoverview" name="parmform" onsubmit="return validateParms();">  <form method="post" action="/adm/parmset?action=newoverview" name="parmform" id="newoverviewform">
   <input type="hidden" name="newoverviewsubm" value="dis" id="newoverviewsubm" />
 ENDOVER  ENDOVER
     my @ids=();      my @ids=();
     my %typep=();      my %typep=();
Line 6223  ENDOVER Line 6505  ENDOVER
     &sortmenu($r,$sortorder,'newoverview');      &sortmenu($r,$sortorder,'newoverview');
     $r->print('</div></div>');      $r->print('</div></div>');
   
     $r->print('<p><input type="submit" name="dis" value="'.&mt('Display').'" /></p>');      $r->print('<p><input type="submit" name="dis" value="'.&mt('Display').'" id="newoverviewdis" /></p>');
   
 # Build the list data hash from the specified parms  # Build the list data hash from the specified parms
   
Line 6235  ENDOVER Line 6517  ENDOVER
         &secgroup_lister($cat,$pschp,$parmlev,$listdata,\@psprt,\@selected_groups,\%defkeytype,\%allmaps,\@ids,\%symbp);          &secgroup_lister($cat,$pschp,$parmlev,$listdata,\@psprt,\@selected_groups,\%defkeytype,\%allmaps,\@ids,\%symbp);
     }      }
   
     if (($env{'form.store'}) || ($env{'form.dis'})) {      my $foundkeys;
       if ($env{'form.newoverviewsubm'}) {
   
         if ($env{'form.store'}) { &storedata($r,$crs,$dom); }          if ($env{'form.newoverviewsubm'} eq 'store') { &storedata($r,$crs,$dom); }
   
 # Read modified data  # Read modified data
   
Line 6253  ENDOVER Line 6536  ENDOVER
                 $hash_for_realm->{$symbp{$ids[$i]}} = $i;                  $hash_for_realm->{$symbp{$ids[$i]}} = $i;
             }              }
         }          }
         &listdata($r,$resourcedata,$listdata,$sortorder,'newoverview',undef,$readonly,$parmlev,$hash_for_realm,$pschp);          $foundkeys = &listdata($r,$resourcedata,$listdata,$sortorder,'newoverview',undef,$readonly,$parmlev,$hash_for_realm,$pschp);
     }      }
     $r->print(&tableend());      $r->print(&tableend());
     unless ($readonly) {      if ((!$readonly) && ($foundkeys)) {
         $r->print( ((($env{'form.store'}) || ($env{'form.dis'}))?'<p><input type="submit" name="store" value="'.&mt('Save').'" /></p>':'') );          $r->print( ($env{'form.newoverviewsubm'}? '<p><input type="submit" name="store" id="newoverviewstore" value="'.&mt('Save').'" /></p>':'') );
     }      }
     $r->print('</form>');      $r->print('</form>');
       if ($env{'form.newoverviewsubm'}) {
           $r->print(<<"END");
   <script type="text/javascript">
   const form = document.getElementById('newoverviewform');
   const storebutton = document.getElementById('newoverviewstore');
   const disbutton = document.getElementById('newoverviewdis');
   const submethod = document.getElementById('newoverviewsubm');
   if (storebutton) { 
     storebutton.addEventListener('keydown', (e) => {
       if (e.key === 'Enter') {
         if (validateParms()) {
           if (form) {
             if (submethod) {
               submethod.value='store';
             }
             form.submit();
           }
         }
         e.preventDefault();
         return;
       }
     });
     storebutton.addEventListener('click', (e) => {
       if (validateParms()) {
         if (form) {
           if (submethod) {
             submethod.value='store';
           }
           form.submit();
         }
       }
       e.preventDefault();
       return;
     });
   }
   if (disbutton) {
     disbutton.addEventListener('keydown', (e) => {
       if (e.key === 'Enter') {
         if (form) {
           if (submethod) {
             submethod.value='dis';
           }
           form.submit();
         }
         e.preventDefault();
       }
     });
     disbutton.addEventListener('click', (e) => {
       if (form) {
         if (submethod) {
           submethod.value='dis';
         }
         form.submit();
         return;
       }
       e.preventDefault();
     });
   }
   
   </script>
   
   END
       }
     &endSettingsScreen($r);      &endSettingsScreen($r);
     $r->print(&Apache::loncommon::end_page());      $r->print(&Apache::loncommon::end_page());
 }  }
Line 6331  sub overview { Line 6677  sub overview {
              &toggleparmtextbox_js()."\n".               &toggleparmtextbox_js()."\n".
              &validateparms_js()."\n".               &validateparms_js()."\n".
              &ipacc_boxes_js()."\n".               &ipacc_boxes_js()."\n".
                &grace_js()."\n".
              &done_proctor_js()."\n".               &done_proctor_js()."\n".
              &deeplink_js()."\n".               &deeplink_js()."\n".
              '// ]]>'."\n".               '// ]]>'."\n".
Line 7620  sub parm_change_log { Line 7967  sub parm_change_log {
                     } else {                      } else {
                         if (&isdateparm($istype{$parmname})) {                          if (&isdateparm($istype{$parmname})) {
                             $showvalue = &Apache::lonlocal::locallocaltime($value);                              $showvalue = &Apache::lonlocal::locallocaltime($value);
                           } elsif (($istype{$parmname} eq 'string_grace') ||
                                    ($istype{$parmname} eq 'string_ip')) {
                               $showvalue =~ s/,/, /g;
                         }                          }
                     }                      }
                     $output .= $showvalue;                      $output .= $showvalue;

Removed from v.1.620  
changed lines
  Added in v.1.624


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