File:  [LON-CAPA] / loncom / homework / edit.pm
Revision 1.34: download - view: text, annotated - select for diffs
Fri Apr 26 13:30:08 2002 UTC (22 years ago) by harris41
Branches: MAIN
CVS tags: version_0_4, stable_2002_july, stable_2002_april, STABLE, HEAD
fixing POD glitch

    1: # The LearningOnline Network with CAPA 
    2: # edit mode helpers
    3: #
    4: # $Id: edit.pm,v 1.34 2002/04/26 13:30:08 harris41 Exp $
    5: #
    6: # Copyright Michigan State University Board of Trustees
    7: #
    8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
    9: #
   10: # LON-CAPA is free software; you can redistribute it and/or modify
   11: # it under the terms of the GNU General Public License as published by
   12: # the Free Software Foundation; either version 2 of the License, or
   13: # (at your option) any later version.
   14: #
   15: # LON-CAPA is distributed in the hope that it will be useful,
   16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
   17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   18: # GNU General Public License for more details.
   19: #
   20: # You should have received a copy of the GNU General Public License
   21: # along with LON-CAPA; if not, write to the Free Software
   22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
   23: #
   24: # /home/httpd/html/adm/gpl.txt
   25: #
   26: # http://www.lon-capa.org/
   27: #
   28: # 3/20 Guy
   29: # 01/10/02 Matthew
   30: # 03/06/02 Matthew
   31: package Apache::edit; 
   32: 
   33: use strict;
   34: use Apache::lonnet();
   35: use HTML::Entities();
   36: 
   37: # Global Vars
   38: # default list of colors to use in editing
   39: @Apache::edit::colorlist=('#ffffff','#ff0000','#00ff00','#0000ff','#0ff000','#000ff0','#f0000f');
   40: # depth of nesting of edit
   41: $Apache::edit::colordepth=0;
   42: 
   43: sub initialize_edit {
   44:   $Apache::edit::colordepth=0;
   45: }
   46: 
   47: sub tag_start {
   48:   my ($target,$token,$description) = @_;
   49:   my $result='';
   50:   if ($target eq "edit") {
   51:     my $tag=$token->[1];
   52:     if (!$description) {
   53:       $description=&Apache::lonxml::description($token);
   54:       if (!$description) { $description="<$tag>"; }
   55:     }
   56:     $result.= &start_table($token)."<tr><td>$description</td>
   57: <td>Delete".
   58:   &deletelist($target,$token)
   59:   ."</td>
   60: <td>".
   61:     &insertlist($target,$token).&end_row().&start_spanning_row();
   62: #<td>". 
   63: #  &movebuttons($target,$token).
   64: #    "</tr><tr><td colspan=\"3\">\n";
   65:   }
   66:   return $result;
   67: }
   68: 
   69: sub tag_end {
   70:   my ($target,$token,$description) = @_;
   71:   my $result='';
   72:   if ($target eq 'edit') {
   73:     my $tag=$token->[1];
   74:     if (!defined($description)) {
   75:       $result.="</td></tr><tr><td>&lt;/$tag&gt;</td><td colspan=\"2\">&nbsp;</td>";
   76:     } else {
   77:       if ($description ne '') { $result.="</td></tr><tr><td>$description</td><td colspan=\"2\">&nbsp;</td>"; }
   78:     }
   79:     $result.="</tr>".&end_table()."\n";
   80:   }
   81:   return $result;
   82: }
   83: 
   84: sub start_table {
   85:   my ($token)=@_;
   86:   my $tag = $token->[1];
   87:   my $tagnum;
   88:   foreach my $namespace (reverse @Apache::lonxml::namespace) {
   89:     my $testtag=$namespace.'::'.$tag;
   90:     $tagnum=$Apache::lonxml::insertlist{"$testtag.num"};
   91:     if (defined($tagnum)) { last; }
   92:   }
   93:   if (!defined ($tagnum)) { $tagnum=$Apache::lonxml::insertlist{"$tag.num"}; }
   94:   my $color = $Apache::lonxml::insertlist{"$tagnum.color"};
   95:   if (!defined($color)) {
   96:     $color = $Apache::edit::colorlist[$Apache::edit::colordepth];
   97:   }
   98:   $Apache::edit::colordepth++;
   99:   my $result="<table bgcolor=\"$color\" width=\"100%\" border=\"5\">";
  100:   return $result;
  101: }
  102: 
  103: sub end_table {
  104:   $Apache::edit::colordepth--;
  105:   my $result="</table>";
  106:   return $result;
  107: }
  108: 
  109: sub start_spanning_row { return '<tr><td colspan="3">';}
  110: sub start_row          { return '<tr><td>';            }
  111: sub end_row            { return '</td></tr>';          }
  112: 
  113: sub movebuttons {
  114:   my ($target,$token) = @_;
  115:   my $result='<input type="submit" name="moveup.'.
  116:     $Apache::lonxml::curdepth.'" value="Move Up" />';
  117:   $result.='<input type="submit" name="movedown.'.
  118:     $Apache::lonxml::curdepth.'" value="Move Down" />';
  119:   return $result;
  120: }
  121: 
  122: sub deletelist {
  123:   my ($target,$token) = @_;
  124:   my $result = "<select name=\"delete_$Apache::lonxml::curdepth\">
  125: <option></option>
  126: <option>Yes</option>
  127: </select>";
  128:   return $result;
  129: }
  130: 
  131: sub handle_delete {
  132:   if (!$ENV{"form.delete_$Apache::lonxml::curdepth"}) { return ''; }
  133:   my ($space,$target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
  134:   my $result=0;
  135:   if ($space) {
  136:     my $sub1="$space\:\:delete_$token->[1]";
  137:     {
  138:       no strict 'refs';
  139:       if (defined &$sub1) {
  140: 	$result=&$sub1($target,$token,$tagstack,$parstack,$parser,$safeeval,$style);
  141:       }
  142:     }
  143:   }
  144:   if (!$result) {
  145:     my $endtag='/'.$token->[1];
  146:     my $bodytext=&Apache::lonxml::get_all_text($endtag,$$parser[$#$parser]);
  147:     $$parser['-1']->get_token();
  148:     &Apache::lonxml::debug("Deleting :$bodytext: for $token->[1]");
  149:     &Apache::lonxml::end_tag($tagstack,$parstack,$token);
  150:   }
  151:   return 1;
  152: }
  153: 
  154: sub get_insert_list {
  155:   my ($token) = @_;
  156:   my $result='';
  157:   my @tagnums= ();
  158:   #&Apache::lonxml::debug("keys ".join("\n",sort(keys(%Apache::lonxml::insertlist))));
  159:   if ($Apache::lonxml::insertlist{"$token->[1].which"}) {
  160:     push (@tagnums, @{ $Apache::lonxml::insertlist{"$token->[1].which"} });
  161:   }
  162:   foreach my $namespace (@Apache::lonxml::namespace) {
  163:     if ($Apache::lonxml::insertlist{"$namespace".'::'."$token->[1].which"}) {
  164:       push (@tagnums, @{ $Apache::lonxml::insertlist{"$namespace".'::'."$token->[1].which"} });
  165:     }
  166:   }
  167:   if (@tagnums) {
  168:     my %options;
  169:     foreach my $tagnum (@tagnums) {
  170:       my $descrip=$Apache::lonxml::insertlist{"$tagnum.description"};
  171:       $options{$descrip} ="<option value=\"$tagnum\">".$descrip."</option>\n";
  172:     }
  173:     foreach my $option (sort(keys(%options))) { $result.=$options{$option}; }
  174:     if ($result) { $result='<option selected="on"></option>'.$result; }
  175:   }
  176:   return $result;
  177: }
  178: 
  179: sub insertlist {
  180:   my ($target,$token) = @_;
  181:   my $result;
  182:   if ($target eq 'edit') {
  183:     my $optionlist= &get_insert_list($token);
  184:     if ($optionlist) {
  185:       $result = "Insert:
  186: <select name=\"insert_$Apache::lonxml::curdepth\">
  187: $optionlist
  188: </select>"
  189:     } else {
  190:       $result="&nbsp;";
  191:     }
  192:   }
  193:   return $result;
  194: }
  195: 
  196: sub handle_insert {
  197:   if ($ENV{"form.insert_$Apache::lonxml::curdepth"} eq '') { return ''; }
  198:   my $result;
  199:   my $tagnum = $ENV{"form.insert_$Apache::lonxml::curdepth"};
  200:   my $func=$Apache::lonxml::insertlist{"$tagnum.function"};
  201:   if ($func eq 'default') {
  202:     my $newtag=$Apache::lonxml::insertlist{"$tagnum.tag"};
  203:     my $namespace;
  204:     if ($newtag =~ /::/) { ($namespace,$newtag) = split(/::/,$newtag); }
  205:     $result.="\n<$newtag>\n</$newtag>";
  206:   } else {
  207:     if (defined(&$func)) {
  208:       {
  209: 	no strict 'refs';
  210: 	$result.=&$func();
  211:       }
  212:     } else {
  213:       my $newtag=$Apache::lonxml::insertlist{"$tagnum.tag"};
  214:       &Apache::lonxml::error("Unable to insert tag $newtag, $func was not defined.");
  215:     }
  216:   }
  217:   return $result;
  218: }
  219: 
  220: sub insert_responseparam {
  221:   return '
  222:     <responseparam />';
  223: }
  224: 
  225: sub insert_formularesponse {
  226:   return '
  227: <formularesponse answer="" samples="">
  228:     <textline />
  229:     <hintgroup>
  230:     </hintgroup>
  231: </formularesponse>';
  232: }
  233: 
  234: sub insert_numericalresponse {
  235:   return '
  236: <numericalresponse answer="">
  237:     <textline />
  238:     <hintgroup>
  239:     </hintgroup>
  240: </numericalresponse>';
  241: }
  242: 
  243: sub insert_stringresponse {
  244:   return '
  245: <stringresponse answer="" type="">
  246:     <textline />
  247:     <hintgroup>
  248:     </hintgroup>
  249: </stringresponse>';
  250: }
  251: 
  252: sub insert_optionresponse {
  253:   return '
  254: <optionresponse max="10">
  255:     <foilgroup options="">
  256:     </foilgroup>
  257:     <hintgroup>
  258:     </hintgroup>
  259: </optionresponse>';
  260: }
  261: 
  262: sub insert_radiobuttonresponse {
  263:   return '
  264: <radiobuttonresponse max="10">
  265:     <foilgroup>
  266:     </foilgroup>
  267:     <hintgroup>
  268:     </hintgroup>
  269: </radiobuttonresponse>';
  270: }
  271: 
  272: sub insert_displayduedate { return '<displayduedate />'; }
  273: sub insert_displaytitle   { return '<displaytitle />'; }
  274: sub insert_hintpart {
  275:   return '
  276: <hintpart on="default">
  277:     <startouttext/>
  278:     <endouttext />
  279: </hintpart>';
  280: }
  281: 
  282: sub insert_numericalhint {
  283:   return '
  284: <numericalhint>
  285: </numericalhint>';
  286: }
  287: 
  288: sub insert_startouttext {
  289:   return "<startouttext />\n<endouttext />";
  290: }
  291: 
  292: sub insert_script {
  293:   return "\n<script type=\"loncapa/perl\">\n</script>";
  294: }
  295: 
  296: sub textarea_sizes {
  297:   my ($data)=@_;
  298:   my $count=0;
  299:   my $maxlength=-1;
  300:   foreach (split ("\n", $$data)) { $count++;
  301: 	if (length($_) > $maxlength) { $maxlength = length($_); }
  302:       }
  303:   my $rows = $count;
  304:   my $cols = $maxlength;
  305:   return ($rows,$cols);
  306: }
  307: 
  308: sub editline {
  309:     my ($tag,$data,$description,$size)=@_;
  310:     $data=&HTML::Entities::encode($data);
  311:     if ($description) { $description="<br />".$description."<br />"; }
  312:     my $result = <<"END";
  313: $description
  314: <input type="text" name="homework_edit_$Apache::lonxml::curdepth" 
  315:        value="$data" size="$size" />
  316: END
  317:     return $result;
  318: }
  319: 
  320: sub editfield {
  321:   my ($tag,$data,$description,$minwidth,$minheight)=@_;
  322: 
  323:   my ($rows,$cols)=&textarea_sizes(\$data);
  324:   if ($cols > 80) { $cols = 80; }
  325:   if ($cols < $minwidth ) { $cols = $minwidth; }
  326:   if ($rows < $minheight) { $rows = $minheight; }
  327:   if ($description) { $description="<br />".$description."<br />"; }
  328:   return $description."\n".'&nbsp;&nbsp;&nbsp;<textarea rows="'.$rows.
  329:     '" cols="'.$cols.'" name="homework_edit_'.$Apache::lonxml::curdepth.'">'.
  330:       &HTML::Entities::encode($data).'</textarea>'."\n";
  331: }
  332: 
  333: sub modifiedfield {
  334:   my ($token) = @_;
  335:   my $result;
  336: #  foreach my $envkey (sort keys %ENV) {
  337: #    &Apache::lonxml::debug("$envkey ---- $ENV{$envkey}");
  338: #  }
  339: #  &Apache::lonxml::debug("I want homework_edit_$Apache::lonxml::curdepth");
  340: #  &Apache::lonxml::debug($ENV{"form.homework_edit_$Apache::lonxml::curdepth"});
  341:   $result=$ENV{"form.homework_edit_$Apache::lonxml::curdepth"};
  342:   return $result;
  343: }
  344: 
  345: # Returns a 1 if the token has been modified and you should rebuild the tag
  346: # side-effects, will modify the $token if new values are found
  347: sub get_new_args {
  348:   my ($token,$parstack,$safeeval,@args)=@_;
  349:   my $rebuild=0;
  350:   foreach my $arg (@args) {
  351:     #just want the string that it was set to
  352:     my $value=$token->[2]->{$arg};
  353:     my $newvalue=$ENV{"form.$Apache::lonxml::curdepth.$arg"};
  354:     &Apache::lonxml::debug(" for:$arg: cur is :$value: new is :$newvalue:");
  355:     if ($value ne $newvalue) {
  356:       $token->[2]->{$arg}=$newvalue;
  357:       $rebuild=1;
  358:     }
  359:   }
  360:   return $rebuild;
  361: }
  362: 
  363: # looks for /> on start tags
  364: sub rebuild_tag {
  365:   my ($token) = @_;
  366:   my $result;
  367:   if ($token->[0] eq 'S') {
  368:     $result = '<'.$token->[1];
  369:     while (my ($key,$val)= each(%{$token->[2]})) {
  370:       $val=~s:^\s+|\s+$::g;
  371:       $val=~s:"::g; #"
  372:       &Apache::lonxml::debug("setting :$key: to  :$val:");
  373:       $result.=' '.$key.'="'.$val.'"';
  374:     }
  375:     if ($token->[4] =~ m:/>$:) {
  376:       $result.=' />';
  377:     } else {
  378:       $result.='>';
  379:     }
  380:   } elsif ( $token->[0] eq 'E' ) {
  381:     $result = '</'.$token->[1].'>';
  382:   }
  383:   return $result;
  384: }
  385: 
  386: sub text_arg {
  387:   my ($description,$name,$token,$size) = @_;
  388:   my $result;
  389:   if (!defined $size) { $size=20; }
  390:   my $arg=$token->[2]{$name};
  391:   $result=$description.'<input name="'."$Apache::lonxml::curdepth.$name".
  392:     '" type="text" value="'.$arg.'" size="'.$size.'" />';
  393:   return $result;
  394: }
  395: 
  396: sub select_arg {
  397:   my ($description,$name,$list,$token) = @_;
  398:   my $result;
  399:   my $optionlist="";
  400:   my $selected=$token->[2]{$name};
  401:   foreach my $option (@$list) {
  402:     if ( $selected eq $option ) {
  403:       $optionlist.="<option selected=\"on\">$option</option>\n";
  404:     } else {
  405:       $optionlist.="<option>$option</option>\n";
  406:     }
  407:   }
  408:   $result.=$description.'<select name="'."$Apache::lonxml::curdepth.$name".'">
  409:        '.$optionlist.'
  410:       </select>';
  411:   return $result;
  412: }
  413: 
  414: sub select_or_text_arg {
  415:   my ($description,$name,$list,$token,$size) = @_;
  416:   my $result;
  417:   my $optionlist="";
  418:   my $found=0;
  419:   my $selected=$token->[2]{$name};
  420:   foreach my $option (@$list) {
  421:     if ( $selected eq $option ) {
  422:       $optionlist.="<option selected=\"on\">$option</option>\n";
  423:       $found=1;
  424:     } else {
  425:       $optionlist.="<option>$option</option>\n";
  426:     }
  427:   }
  428:   $optionlist.="<option value=\"TYPEDINVALUE\">Type in value</option>\n";
  429:   if ($found) {
  430:     $result.=$description.'<select name="'."$Apache::lonxml::curdepth.$name".'">
  431:        '.$optionlist.'
  432:       </select>';
  433:   } else {
  434:     $result.=&text_arg($description,$name,$token,$size);
  435:   }
  436:   return $result;
  437: }
  438: 
  439: #----------------------------------------------------- browse
  440: sub browse {
  441:     # insert a link to call up the filesystem browser (lonindexer)
  442:     $_ = shift;
  443:     my $form    = 'lonhomework';
  444:     my $element = &Apache::lonnet::escape("$Apache::lonxml::curdepth.$_");
  445:     my $result = <<"ENDBUTTON";
  446: <a href=\"javascript:openbrowser('$form','$element')\"\>Browse</a>
  447: ENDBUTTON
  448:     return $result;
  449: }
  450: 
  451: #----------------------------------------------------- browse
  452: sub search {
  453:     # insert a link to call up the filesystem browser (lonindexer)
  454:     $_ = shift;
  455:     my $form    = 'lonhomework';
  456:     my $element = &Apache::lonnet::escape("$Apache::lonxml::curdepth.$_");
  457:     my $result = <<"ENDBUTTON";
  458: <a href=\"javascript:opensearcher('$form','$element')\"\>Search</a>
  459: ENDBUTTON
  460:     return $result;
  461: }
  462: 
  463: 
  464: 1;
  465: __END__
  466: 
  467: =head1 NAME
  468: 
  469: Apache::edit - edit mode helpers
  470: 
  471: =head1 SYNOPSIS
  472: 
  473: Invoked by many homework and xml related modules.
  474: 
  475:  &Apache::edit::SUBROUTINENAME(ARGUMENTS);
  476: 
  477: =head1 INTRODUCTION
  478: 
  479: This module outputs HTML syntax helpful for the rendering of edit
  480: mode interfaces.
  481: 
  482: This is part of the LearningOnline Network with CAPA project
  483: described at http://www.lon-capa.org.
  484: 
  485: =head1 HANDLER SUBROUTINE
  486: 
  487: There is no handler subroutine.
  488: 
  489: =head1 OTHER SUBROUTINES
  490: 
  491: =over 4
  492: 
  493: =item *
  494: 
  495: initialize_edit() : initialize edit (set colordepth to zero)
  496: 
  497: =item *
  498: 
  499: tag_start($target,$token,$description) : provide deletion and insertion lists
  500: for the manipulation of a start tag; return a scalar string
  501: 
  502: =item *
  503: 
  504: tag_end($target,$token,$description) : ending syntax corresponding to
  505: &tag_start. return a scalar string.
  506: 
  507: =item *
  508: 
  509: start_table($token) : start table; update colordepth; return scalar string.
  510: 
  511: =item *
  512: 
  513: end_table() : reduce color depth; end table; return scalar string
  514: 
  515: =item *
  516: 
  517: start_spanning_row() : start a new table row spanning the 'edit' environment.
  518: 
  519: =item *
  520: 
  521: start_row() : start a new table row and element. 
  522: 
  523: =item *
  524: 
  525: end_row() : end current table element and row.
  526: 
  527: =item *
  528: 
  529: movebuttons($target,$token) : move-up and move-down buttons; return scalar
  530: string
  531: 
  532: =item *
  533: 
  534: deletelist($target,$token) : provide a yes option in an HTML select element;
  535: return scalar string
  536: 
  537: =item *
  538: 
  539: handle_delete($space,$target,$token,$tagstack,$parstack,$parser,$safeeval,
  540: $style) : respond to a user delete request by passing relevant stack
  541: and array information to various rendering functions; return a scalar string
  542: 
  543: =item *
  544: 
  545: get_insert_list($token) : provide an insertion list based on possibilities
  546: from lonxml; return a scalar string
  547: 
  548: =item *
  549: 
  550: insertlist($target,$token) : api that uses get_insert_list;
  551: return a scalar string
  552: 
  553: =item *
  554: 
  555: handleinsert($token) : provide an insertion list based on possibilities
  556: from lonxml; return a scalar string
  557: 
  558: =item *
  559: 
  560: get_insert_list($token) : provide an insertion list based on possibilities
  561: from lonxml; return a scalar string
  562: 
  563: =item *
  564: browse($elementname) : provide a link which will open up the filesystem
  565: browser (lonindexer) and, once a file is selected, place the result in
  566: the form element $elementname.
  567: 
  568: =item *
  569: search($elementname) : provide a link which will open up the filesystem
  570: searcher (lonsearchcat) and, once a file is selected, place the result in
  571: the form element $elementname.
  572: 
  573: =item *
  574: editline(tag,data,description,size): Provide a <input type="text" ../> for
  575: single-line text entry.  This is to be used for text enclosed by tags, not
  576: arguements/parameters associated with a tag.
  577: 
  578: =back
  579: 
  580: incomplete...
  581: 
  582: =cut

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