Annotation of loncom/homework/edit.pm, revision 1.109

1.1       albertel    1: # The LearningOnline Network with CAPA 
                      2: # edit mode helpers
1.25      albertel    3: #
1.109   ! albertel    4: # $Id: edit.pm,v 1.108 2007/06/19 20:24:42 banghart Exp $
1.25      albertel    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: #
1.82      www        28: 
1.1       albertel   29: package Apache::edit; 
                     30: 
                     31: use strict;
1.92      albertel   32: use Apache::lonnet;
1.32      albertel   33: use HTML::Entities();
1.71      www        34: use Apache::lonlocal;
1.102     www        35: use lib '/home/httpd/lib/perl/';
                     36: use LONCAPA;
                     37:  
1.1       albertel   38: 
1.10      albertel   39: # Global Vars
                     40: # default list of colors to use in editing
                     41: @Apache::edit::colorlist=('#ffffff','#ff0000','#00ff00','#0000ff','#0ff000','#000ff0','#f0000f');
                     42: # depth of nesting of edit
                     43: $Apache::edit::colordepth=0;
1.38      www        44: @Apache::edit::inserttag=();
1.49      www        45: # image-type responses: active background image and curdepth at definition
                     46: $Apache::edit::bgimgsrc='';
                     47: $Apache::edit::bgimgsrccurdepth='';
1.10      albertel   48: 
                     49: sub initialize_edit {
1.63      albertel   50:     $Apache::edit::colordepth=0;
                     51:     @Apache::edit::inserttag=();
1.10      albertel   52: }
                     53: 
1.1       albertel   54: sub tag_start {
1.63      albertel   55:     my ($target,$token,$description) = @_;
                     56:     my $result='';
                     57:     if ($target eq "edit") {
                     58: 	my $tag=$token->[1];
                     59: 	if (!$description) {
                     60: 	    $description=&Apache::lonxml::description($token);
                     61: 	    if (!$description) { $description="<$tag>"; }
                     62: 	}
                     63: 	$result.= &start_table($token)."<tr><td>$description</td>
                     64:                       <td>Delete".
                     65: 		      &deletelist($target,$token)
                     66: 		      ."</td>
                     67:                        <td>".
                     68: 		       &insertlist($target,$token);
1.27      matthew    69: #<td>". 
1.22      albertel   70: #  &movebuttons($target,$token).
                     71: #    "</tr><tr><td colspan=\"3\">\n";
1.66      matthew    72: 	my @help = Apache::lonxml::helpinfo($token);
1.63      albertel   73: 	if ($help[0]) {
1.88      albertel   74: 	    $result .= '</td><td align="right" valign="top">' .
                     75: 		Apache::loncommon::help_open_topic(@help);
                     76: 	} else { $result .= "</td><td>&nbsp;"; }
1.63      albertel   77: 	$result .= &end_row().&start_spanning_row();
                     78:     }
                     79:     return $result;
1.1       albertel   80: }
                     81: 
                     82: sub tag_end {
1.63      albertel   83:     my ($target,$token,$description) = @_;
                     84:     my $result='';
                     85:     if ($target eq 'edit') {
                     86: 	$result.="</td></tr>".&end_table()."\n";
                     87:     }
                     88:     return $result;
1.4       albertel   89: }
1.1       albertel   90: 
1.10      albertel   91: sub start_table {
1.63      albertel   92:     my ($token)=@_;
1.104     albertel   93:     my $tag = &Apache::lonxml::get_tag($token);
                     94:     
                     95:     my $color = $Apache::lonxml::insertlist{"$tag.color"};
                     96:     &Apache::lonxml::debug(" $tag -- $color");
1.63      albertel   97:     if (!defined($color)) {
                     98: 	$color = $Apache::edit::colorlist[$Apache::edit::colordepth];
                     99:     }
                    100:     $Apache::edit::colordepth++;
                    101:     push(@Apache::edit::inserttag,$token->[1]);
                    102:     my $result='<div align="right">';
                    103:     $result.='<table bgcolor="'.$color.'" width="97%" border="0" cellspacing="5" cellpadding="3">';
                    104:     return $result;
1.10      albertel  105: }
                    106: 
                    107: sub end_table {
1.63      albertel  108:     $Apache::edit::colordepth--;
                    109:     my $result='</table></div>';
1.94      albertel  110:     $result.='<div align="left"><table><tr><td>';
1.63      albertel  111: 
                    112:     my ($tagname,$closingtag);
                    113:     if (defined($Apache::edit::inserttag[-2])) {
                    114: 	$tagname=$Apache::edit::inserttag[-2];
1.105     albertel  115:     } else {
                    116: 	if ($Apache::lonhomework::parsing_a_task) {
                    117: 	    $tagname='Task';
                    118: 	} else {
                    119: 	    $tagname='problem';
                    120: 	}
                    121:     }
1.63      albertel  122:     if (defined($Apache::edit::inserttag[-1])) {
                    123: 	$closingtag=$Apache::edit::inserttag[-1];
                    124:     }
                    125:     $result.=&innerinsertlist('edit',$tagname,$closingtag).
1.94      albertel  126: 	"</td></tr></table></div>";
1.105     albertel  127:     my $last = pop(@Apache::edit::inserttag);
1.63      albertel  128:     return $result;
1.10      albertel  129: }
                    130: 
1.58      bowersj2  131: sub start_spanning_row { return '<tr><td colspan="4" bgcolor="#DDDDDD">';}
1.41      www       132: sub start_row          { return '<tr><td bgcolor="#DDDDDD">';            }
1.27      matthew   133: sub end_row            { return '</td></tr>';          }
                    134: 
1.22      albertel  135: sub movebuttons {
1.63      albertel  136:     my ($target,$token) = @_;
                    137:     my $result='<input type="submit" name="moveup.'.
                    138: 	$Apache::lonxml::curdepth.'" value="Move Up" />';
                    139:     $result.='<input type="submit" name="movedown.'.
                    140: 	$Apache::lonxml::curdepth.'" value="Move Down" />';
                    141:     return $result;
1.22      albertel  142: }
                    143: 
1.8       albertel  144: sub deletelist {
1.63      albertel  145:     my ($target,$token) = @_;
                    146:     my $result = "<select name=\"delete_$Apache::lonxml::curdepth\">
1.14      albertel  147: <option></option>
                    148: <option>Yes</option>
1.8       albertel  149: </select>";
1.63      albertel  150:     return $result;
1.8       albertel  151: }
                    152: 
1.14      albertel  153: sub handle_delete {
1.92      albertel  154:     if (!$env{"form.delete_$Apache::lonxml::curdepth"}) { return ''; }
1.63      albertel  155:     my ($space,$target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
                    156:     my $result=0;
                    157:     if ($space) {
                    158: 	my $sub1="$space\:\:delete_$token->[1]";
                    159: 	{
                    160: 	    no strict 'refs';
                    161: 	    if (defined &$sub1) {
                    162: 		$result=&$sub1($target,$token,$tagstack,$parstack,$parser,$safeeval,$style);
                    163: 	    }
                    164: 	}
                    165:     }
                    166:     if (!$result) {
                    167: 	my $endtag='/'.$token->[1];
1.97      albertel  168: 	my $bodytext=&Apache::lonxml::get_all_text($endtag,$parser,$style);
1.63      albertel  169: 	$$parser['-1']->get_token();
                    170: 	&Apache::lonxml::debug("Deleting :$bodytext: for $token->[1]");
                    171: 	&Apache::lonxml::end_tag($tagstack,$parstack,$token);
                    172:     }
                    173:     return 1;
1.14      albertel  174: }
                    175: 
1.7       albertel  176: sub get_insert_list {
1.63      albertel  177:     my ($tagname) = @_;
                    178:     my $result='';
1.104     albertel  179:     my @tags= ();
1.63      albertel  180:     #&Apache::lonxml::debug("keys ".join("\n",sort(keys(%Apache::lonxml::insertlist))));
                    181:     if ($Apache::lonxml::insertlist{"$tagname.which"}) {
1.104     albertel  182: 	push (@tags, @{ $Apache::lonxml::insertlist{"$tagname.which"} });
1.63      albertel  183:     }
                    184:     foreach my $namespace (@Apache::lonxml::namespace) {
                    185: 	if ($Apache::lonxml::insertlist{"$namespace".'::'."$tagname.which"}) {
1.104     albertel  186: 	    push (@tags, @{ $Apache::lonxml::insertlist{"$namespace".'::'."$tagname.which"} });
1.63      albertel  187: 	}
                    188:     }
1.104     albertel  189:     if (@tags) {
1.63      albertel  190: 	my %options;
1.104     albertel  191: 	foreach my $tag (@tags) {
                    192: 	    my $descrip=$Apache::lonxml::insertlist{"$tag.description"};
                    193: 	    my $tagnum =$Apache::lonxml::insertlist{"$tag.num"};
1.63      albertel  194: 	    $options{$descrip} ="<option value=\"$tagnum\">".
                    195: 		$descrip."</option>\n";
                    196: 	}
                    197: 	foreach my $option (sort(keys(%options))) {$result.=$options{$option};}
1.88      albertel  198: 	if ($result) { $result='<option selected="selected"></option>'.$result; }
1.63      albertel  199:     }
                    200:     return $result;
1.5       albertel  201: }
                    202: 
1.4       albertel  203: sub insertlist {
1.63      albertel  204:     my ($target,$token) = @_;
                    205:     return &innerinsertlist($target,$token->[1]);
1.38      www       206: }
                    207: 
                    208: sub innerinsertlist {
1.63      albertel  209:     my ($target,$tagname,$closingtag) = @_;
                    210:     my $result;
                    211:     my $after='';
                    212:     if ($closingtag) {
                    213: 	$after='_after_'.$closingtag; 
                    214:     }
                    215:     if ($target eq 'edit') {
                    216: 	my $optionlist= &get_insert_list($tagname);
                    217: 	if ($optionlist) {
                    218: 	    $result = "Insert:
                    219:             <select name=\"insert$after\_$Apache::lonxml::curdepth\">
                    220:                   $optionlist
                    221:             </select>"
                    222: 	} else {
                    223: 	    $result="&nbsp;";
                    224: 	}
1.6       albertel  225:     }
1.63      albertel  226:     return $result;
1.6       albertel  227: }
                    228: 
1.7       albertel  229: sub handle_insert {
1.92      albertel  230:     if ($env{"form.insert_$Apache::lonxml::curdepth"} eq '') { return ''; }
                    231:     my $tagnum = $env{"form.insert_$Apache::lonxml::curdepth"};
1.104     albertel  232:     return &do_insert($tagnum);
1.38      www       233: }
                    234: 
                    235: sub handle_insertafter {
1.63      albertel  236:     my $tagname=shift;
1.104     albertel  237:     if ($env{"form.insert_after_$tagname\_$Apache::lonxml::curdepth"} eq '') {
                    238: 	return '';
                    239:     }
                    240:     my $tagnum =$env{"form.insert_after_$tagname\_$Apache::lonxml::curdepth"};
1.105     albertel  241:     return &do_insert($tagnum,1);
1.104     albertel  242: }
                    243: 
                    244: sub do_insert {
1.105     albertel  245:     my ($tagnum,$after) = @_;
1.63      albertel  246:     my $result;
1.104     albertel  247: 
                    248:     my $newtag = $Apache::lonxml::insertlist{"$tagnum.tag"};
                    249:     my $func   = $Apache::lonxml::insertlist{"$newtag.function"};
1.63      albertel  250:     if ($func eq 'default') {
                    251: 	my $namespace;
                    252: 	if ($newtag =~ /::/) { ($namespace,$newtag) = split(/::/,$newtag); }
1.105     albertel  253: 	my $depth = scalar(@Apache::lonxml::depthcounter);
                    254: 	$depth -- if ($after);
                    255: 	my $inset = "\t"x$depth;
                    256: 	$result.="\n$inset<$newtag>\n$inset</$newtag>";
1.38      www       257:     } else {
1.63      albertel  258: 	if (defined(&$func)) {
                    259: 	    {
                    260: 		no strict 'refs';
                    261: 		$result.=&$func();
                    262: 	    }
                    263: 	} else {
1.104     albertel  264: 	    &Apache::lonxml::error("Unable to insert tag $newtag, $func was not defined. ($tagnum)");
1.63      albertel  265: 	}
1.5       albertel  266:     }
1.63      albertel  267:     return $result;
1.69      albertel  268: }
                    269: 
                    270: sub insert_img {
                    271:     return '
                    272:     <img />';
1.16      albertel  273: }
                    274: 
                    275: sub insert_responseparam {
1.63      albertel  276:     return '
1.16      albertel  277:     <responseparam />';
1.5       albertel  278: }
                    279: 
1.87      albertel  280: sub insert_parameter {
                    281:     return '
                    282:     <parameter />';
                    283: }
                    284: 
1.24      albertel  285: sub insert_formularesponse {
1.63      albertel  286:     return '
1.24      albertel  287: <formularesponse answer="" samples="">
1.86      albertel  288:     <responseparam description="Numerical Tolerance" type="tolerance" default="0.00001" name="tol" />
                    289:     <textline size="25"/>
1.24      albertel  290:     <hintgroup>
1.78      www       291:     <startouttext /><endouttext />
1.24      albertel  292:     </hintgroup>
                    293: </formularesponse>';
                    294: }
                    295: 
1.15      albertel  296: sub insert_numericalresponse {
1.63      albertel  297:     return '
1.15      albertel  298: <numericalresponse answer="">
1.90      www       299: <responseparam type="tolerance" default="5%" name="tol" description="Numerical Tolerance" />
                    300: <responseparam name="sig" type="int_range,0-16" default="0,15" description="Significant Figures" />
1.15      albertel  301:     <textline />
                    302:     <hintgroup>
1.78      www       303:     <startouttext /><endouttext />
1.15      albertel  304:     </hintgroup>
                    305: </numericalresponse>';
                    306: }
                    307: 
1.96      albertel  308: sub insert_customresponse {
1.95      albertel  309:     return '
1.96      albertel  310: <customresponse>
1.95      albertel  311:     <answer type="loncapa/perl">
                    312:     </answer>
                    313:     <textline />
                    314:     <hintgroup>
                    315:     <startouttext /><endouttext />
                    316:     </hintgroup>
1.96      albertel  317: </customresponse>';
1.95      albertel  318: }
                    319: 
1.96      albertel  320: sub insert_customresponse_answer {
1.95      albertel  321:     return '
                    322:     <answer type="loncapa/perl">
                    323:     </answer>
                    324: ';
                    325: }
                    326: 
1.106     albertel  327: sub insert_customhint {
                    328:     return '
                    329:         <customhint>
                    330:             <answer type="loncapa/perl">
                    331:             </answer>
                    332:         </customhint>';
                    333: }
                    334: 
                    335: sub insert_customhint_answer {
                    336:     return '
                    337:             <answer type="loncapa/perl">
                    338:             </answer>
                    339: ';
                    340: }
                    341: 
                    342: sub insert_mathresponse {
                    343:     return '
                    344: <mathresponse>
                    345:     <answer>
                    346:     </answer>
                    347:     <textline />
                    348:     <hintgroup>
                    349:         <startouttext />
                    350:         <endouttext />
                    351:     </hintgroup>
                    352: </mathresponse>';
                    353: }
                    354: 
                    355: sub insert_mathresponse_answer {
                    356:     return '
                    357:     <answer>
                    358:     </answer>
                    359: ';
                    360: }
                    361: 
                    362: sub insert_mathhint {
                    363:     return '
                    364:         <mathhint>
                    365:             <answer>
                    366:             </answer>
                    367:         </mathhint>';
                    368: }
                    369: 
                    370: sub insert_mathhint_answer {
                    371:     return '
                    372:             <answer>
                    373:             </answer>
                    374: ';
                    375: }
                    376: 
1.18      albertel  377: sub insert_stringresponse {
1.63      albertel  378:     return '
1.18      albertel  379: <stringresponse answer="" type="">
                    380:     <textline />
                    381:     <hintgroup>
1.78      www       382:     <startouttext /><endouttext />
1.18      albertel  383:     </hintgroup>
                    384: </stringresponse>';
1.36      albertel  385: }
                    386: 
                    387: sub insert_essayresponse {
1.63      albertel  388:     return '
1.36      albertel  389: <essayresponse>
                    390:     <textfield></textfield>
                    391: </essayresponse>';
1.54      albertel  392: }
                    393: 
                    394: sub insert_imageresponse {
1.63      albertel  395:     return '
1.54      albertel  396: <imageresponse max="1">
                    397:     <foilgroup>
1.89      albertel  398:       <foil>
                    399:       </foil>
1.54      albertel  400:     </foilgroup>
                    401:     <hintgroup>
1.78      www       402:     <startouttext /><endouttext />
1.54      albertel  403:     </hintgroup>
                    404: </imageresponse>';
1.18      albertel  405: }
                    406: 
1.7       albertel  407: sub insert_optionresponse {
1.63      albertel  408:     return '
1.7       albertel  409: <optionresponse max="10">
                    410:     <foilgroup options="">
1.89      albertel  411:       <foil>
                    412:          <startouttext /><endouttext />
                    413:       </foil>
1.7       albertel  414:     </foilgroup>
1.14      albertel  415:     <hintgroup>
1.78      www       416:     <startouttext /><endouttext />
1.14      albertel  417:     </hintgroup>
1.7       albertel  418: </optionresponse>';
1.1       albertel  419: }
                    420: 
1.72      albertel  421: sub insert_organicresponse {
                    422:     return '
                    423: <organicresponse>
                    424:     <textline />
                    425:     <hintgroup>
1.78      www       426:     <startouttext /><endouttext />
1.72      albertel  427:     </hintgroup>
                    428: </organicresponse>';
                    429: }
                    430: 
                    431: sub insert_organicstructure {
                    432:     return '
                    433: <organicstructure />
                    434: ';
                    435: }
                    436: 
1.23      albertel  437: sub insert_radiobuttonresponse {
1.63      albertel  438:     return '
1.23      albertel  439: <radiobuttonresponse max="10">
                    440:     <foilgroup>
1.89      albertel  441:       <foil>
                    442:          <startouttext /><endouttext />
                    443:       </foil>
1.23      albertel  444:     </foilgroup>
                    445:     <hintgroup>
1.78      www       446:     <startouttext /><endouttext />
1.23      albertel  447:     </hintgroup>
                    448: </radiobuttonresponse>';
1.72      albertel  449: }
                    450: 
                    451: sub insert_reactionresponse {
                    452:     return '
                    453: <reactionresponse>
                    454:     <textline />
                    455:     <hintgroup>
1.78      www       456:     <startouttext /><endouttext />
1.72      albertel  457:     </hintgroup>
                    458: </reactionresponse>';
1.43      albertel  459: }
                    460: 
                    461: sub insert_rankresponse {
1.63      albertel  462:     return '
1.43      albertel  463: <rankresponse max="10">
                    464:     <foilgroup options="">
1.89      albertel  465:       <foil>
                    466:          <startouttext /><endouttext />
                    467:       </foil>
1.43      albertel  468:     </foilgroup>
                    469:     <hintgroup>
1.78      www       470:     <startouttext /><endouttext />
1.43      albertel  471:     </hintgroup>
                    472: </rankresponse>';
1.23      albertel  473: }
                    474: 
1.44      albertel  475: sub insert_matchresponse {
1.63      albertel  476:     return '
1.44      albertel  477: <matchresponse max="10">
                    478:     <foilgroup options="">
                    479:       <itemgroup>
                    480:       </itemgroup>
1.89      albertel  481:       <foil>
                    482:          <startouttext /><endouttext />
                    483:       </foil>
1.44      albertel  484:     </foilgroup>
                    485:     <hintgroup>
1.78      www       486:     <startouttext /><endouttext />
1.44      albertel  487:     </hintgroup>
                    488: </matchresponse>';
                    489: }
                    490: 
1.21      albertel  491: sub insert_displayduedate { return '<displayduedate />'; }
                    492: sub insert_displaytitle   { return '<displaytitle />'; }
1.22      albertel  493: sub insert_hintpart {
1.63      albertel  494:     return '
1.22      albertel  495: <hintpart on="default">
1.90      www       496:     <startouttext/><endouttext />
1.22      albertel  497: </hintpart>';
1.67      albertel  498: }
                    499: 
                    500: sub insert_hintgroup {
                    501:   return '
                    502: <hintgroup>
1.78      www       503:     <startouttext /><endouttext />
1.67      albertel  504: </hintgroup>';
1.22      albertel  505: }
                    506: 
                    507: sub insert_numericalhint {
1.63      albertel  508:     return '
1.22      albertel  509: <numericalhint>
                    510: </numericalhint>';
1.46      albertel  511: }
                    512: 
                    513: sub insert_stringhint {
1.63      albertel  514:     return '
1.46      albertel  515: <stringhint>
                    516: </stringhint>';
                    517: }
                    518: 
                    519: sub insert_formulahint {
1.63      albertel  520:     return '
1.46      albertel  521: <formulahint>
                    522: </formulahint>';
1.37      albertel  523: }
                    524: 
                    525: sub insert_radiobuttonhint {
1.63      albertel  526:     return '
1.37      albertel  527: <radiobuttonhint>
                    528: </radiobuttonhint>';
1.50      albertel  529: }
                    530: 
                    531: sub insert_optionhint {
1.63      albertel  532:     return '
1.50      albertel  533: <optionhint>
                    534: </optionhint>';
1.22      albertel  535: }
1.21      albertel  536: 
1.23      albertel  537: sub insert_startouttext {
1.78      www       538:     return "<startouttext /><endouttext />";
1.23      albertel  539: }
                    540: 
                    541: sub insert_script {
1.78      www       542:     return "\n<script type=\"loncapa/perl\"></script>";
1.23      albertel  543: }
                    544: 
1.98      albertel  545: sub js_change_detection {
                    546:     my $unsaved=&mt("There are unsaved changes");
                    547:     return (<<SCRIPT);
                    548: <script type="text/javascript">
                    549: var clean = true;
                    550: var is_submit = false;
1.99      albertel  551: var still_ask = false;
1.98      albertel  552: function compareForm(event_) {
                    553:         if (!event_ && window.event) {
                    554:           event_ = window.event;
                    555:         }
1.99      albertel  556: 	if ((!is_submit || (is_submit && still_ask)) && !clean) {
                    557: 	    still_ask = false;
                    558: 	    is_submit = false;
1.98      albertel  559:             event_.returnValue = "$unsaved";
                    560:             return "$unsaved";
                    561:         }
                    562: }
                    563: function unClean() {
                    564:      clean=false;
                    565: }
                    566: window.onbeforeunload = compareForm;
                    567: </script>
                    568: SCRIPT
                    569: }
                    570: 
                    571: sub form_change_detection {
                    572:     return ' onsubmit="is_submit=true;" ';
                    573: }
                    574: 
                    575: sub element_change_detection {
                    576:     return ' onchange="unClean();" ';
                    577: }
                    578: 
1.99      albertel  579: sub submit_ask_anyway {
                    580:     return ' onclick="still_ask=true;" ';
                    581: }
                    582: 
1.25      albertel  583: sub textarea_sizes {
1.63      albertel  584:     my ($data)=@_;
                    585:     my $count=0;
                    586:     my $maxlength=-1;
                    587:     foreach (split ("\n", $$data)) {
                    588: 	$count+=int(length($_)/79);
                    589: 	$count++;
                    590: 	if (length($_) > $maxlength) { $maxlength = length($_); }
                    591:     }
                    592:     my $rows = $count;
                    593:     my $cols = $maxlength;
                    594:     return ($rows,$cols);
1.25      albertel  595: }
                    596: 
1.32      albertel  597: sub editline {
1.31      matthew   598:     my ($tag,$data,$description,$size)=@_;
1.81      albertel  599:     $data=&HTML::Entities::encode($data,'<>&"');
1.31      matthew   600:     if ($description) { $description="<br />".$description."<br />"; }
1.98      albertel  601:     my $change_code = &element_change_detection();
1.31      matthew   602:     my $result = <<"END";
                    603: $description
                    604: <input type="text" name="homework_edit_$Apache::lonxml::curdepth" 
1.98      albertel  605:        value="$data" size="$size" $change_code />
1.31      matthew   606: END
                    607:     return $result;
                    608: }
                    609: 
1.2       albertel  610: sub editfield {
1.82      www       611:     my ($tag,$data,$description,$minwidth,$minheight,$usehtmlarea)=@_;
1.22      albertel  612: 
1.63      albertel  613:     my ($rows,$cols)=&textarea_sizes(\$data);
1.84      www       614:     if (&Apache::lonhtmlcommon::htmlareabrowser() &&
                    615: 	!&Apache::lonhtmlcommon::htmlareablocked()) {
                    616: 	$rows+=7;      # make room for HTMLarea
                    617: 	$minheight+=7; # make room for HTMLarea
                    618:     }
1.63      albertel  619:     if ($cols > 80) { $cols = 80; }
                    620:     if ($cols < $minwidth ) { $cols = $minwidth; }
                    621:     if ($rows < $minheight) { $rows = $minheight; }
                    622:     if ($description) { $description="<br />".$description."<br />"; }
1.82      www       623:     if ($usehtmlarea) {
1.101     albertel  624: 	&Apache::lonhtmlcommon::add_htmlareafields('homework_edit_'.
                    625: 						   $Apache::lonxml::curdepth);
1.82      www       626:     }
1.105     albertel  627:     # remove typesetting whitespace from between data and the end tag
                    628:     # to make the edit look prettier
                    629:     $data =~ s/\n?[ \t]*$//;
                    630: 
1.103     albertel  631:     return $description."\n".'<textarea style="width:100%" rows="'.$rows.
1.63      albertel  632: 	'" cols="'.$cols.'" name="homework_edit_'.
1.82      www       633: 	$Apache::lonxml::curdepth.'" id="homework_edit_'.
1.98      albertel  634: 	$Apache::lonxml::curdepth.'" '.&element_change_detection().'>'.
1.85      www       635: 	&HTML::Entities::encode($data,'<>&"').'</textarea>'.
                    636: 	($usehtmlarea?&Apache::lonhtmlcommon::spelllink('lonhomework',
                    637: 			 'homework_edit_'.$Apache::lonxml::curdepth):'')."\n";
1.2       albertel  638: }
                    639: 
                    640: sub modifiedfield {
1.70      albertel  641:     my ($endtag,$parser) = @_;
1.63      albertel  642:     my $result;
1.92      albertel  643:     $result=$env{"form.homework_edit_$Apache::lonxml::curdepth"};
1.70      albertel  644:     my $bodytext=&Apache::lonxml::get_all_text($endtag,$parser);
                    645:     # textareas throw away intial \n 
1.105     albertel  646:     if ($bodytext=~/^\n/) {
                    647: 	$result="\n".$result;
                    648:     }
                    649:     # if there is typesetting whitespace from between the data and the end tag
                    650:     # restore to keep the source looking pretty
                    651:     if ($bodytext =~ /(\n?[ \t]*)$/) {
                    652: 	$result .= $1;
                    653:     }
1.63      albertel  654:     return $result;
1.2       albertel  655: }
                    656: 
1.15      albertel  657: # Returns a 1 if the token has been modified and you should rebuild the tag
1.12      albertel  658: # side-effects, will modify the $token if new values are found
                    659: sub get_new_args {
1.61      albertel  660:     my ($token,$parstack,$safeeval,@args)=@_;
                    661:     my $rebuild=0;
                    662:     foreach my $arg (@args) {
1.63      albertel  663: 	#just want the string that it was set to
                    664: 	my $value=$token->[2]->{$arg};
                    665: 	my $element=&html_element_name($arg);
1.92      albertel  666: 	my $newvalue=$env{"form.$element"};
1.63      albertel  667: 	&Apache::lonxml::debug("for:$arg: cur is :$value: new is :$newvalue:");
                    668: 	if (defined($newvalue) && $value ne $newvalue) {
                    669: 	    if (ref($newvalue) eq 'ARRAY') {
                    670: 		$token->[2]->{$arg}=join(',',@$newvalue);
                    671: 	    } else {
                    672: 		$token->[2]->{$arg}=$newvalue;
                    673: 	    }
1.79      albertel  674: 	    $rebuild=1;
                    675: 	} elsif (!defined($newvalue) && defined($value)) {
                    676: 	    delete($token->[2]->{$arg});
1.63      albertel  677: 	    $rebuild=1;
1.61      albertel  678: 	}
1.12      albertel  679:     }
1.63      albertel  680:     return $rebuild;
1.12      albertel  681: }
                    682: 
1.15      albertel  683: # looks for /> on start tags
1.12      albertel  684: sub rebuild_tag {
1.63      albertel  685:     my ($token) = @_;
                    686:     my $result;
                    687:     if ($token->[0] eq 'S') {
                    688: 	$result = '<'.$token->[1];
1.109   ! albertel  689: 	#FIXME do this in non random order
        !           690: 	foreach my $attribute (@{ $token->[3] }) {
        !           691: 	    my $value = $token->[2]{$attribute};
        !           692: 	    $value =~s/^\s+|\s+$//g;
        !           693: 	    $value =~s/\"//g;
        !           694: 	    &Apache::lonxml::debug("setting :$attribute: to  :$value:");
        !           695: 	    $result.=' '.$attribute.'="'.$value.'"';
1.63      albertel  696: 	}
                    697: 	if ($token->[4] =~ m:/>$:) {
                    698: 	    $result.=' />';
                    699: 	} else {
                    700: 	    $result.='>';
                    701: 	}
                    702:     } elsif ( $token->[0] eq 'E' ) {
                    703: 	$result = '</'.$token->[1].'>';
1.12      albertel  704:     }
1.63      albertel  705:     return $result;
1.12      albertel  706: }
1.13      albertel  707: 
1.47      matthew   708: sub html_element_name {
                    709:     my ($name) = @_;
1.48      albertel  710:     return $name.'_'.$Apache::lonxml::curdepth;
                    711: }
                    712: 
                    713: sub hidden_arg {
                    714:     my ($name,$token) = @_;
                    715:     my $result;
                    716:     my $arg=$token->[2]{$name};
                    717:     $result='<input name="'.&html_element_name($name).
                    718: 	'" type="hidden" value="'.$arg.'" />';
1.61      albertel  719:     return $result;
                    720: }
                    721: 
                    722: sub checked_arg {
                    723:     my ($description,$name,$list,$token) = @_;
                    724:     my $result;
                    725:     my $optionlist="";
                    726:     my $allselected=$token->[2]{$name};
1.71      www       727:     $result=&mt($description);
1.61      albertel  728:     foreach my $option (@$list) {
                    729: 	my ($value,$text);
                    730: 	if ( ref($option) eq 'ARRAY') {
                    731: 	    $value='value="'.$$option[0].'"';
                    732: 	    $text=$$option[1];
                    733: 	    $option=$$option[0];
                    734: 	} else {
                    735: 	    $text=$option;
                    736: 	    $value='value="'.$option.'"';
                    737: 	}
1.100     albertel  738: 	$result.="<nobr><label><input type='checkbox' $value name='".
1.61      albertel  739: 	    &html_element_name($name)."'";
                    740: 	foreach my $selected (split(/,/,$allselected)) {
                    741: 	    if ( $selected eq $option ) {
1.88      albertel  742: 		$result.=" checked='checked' ";
1.61      albertel  743: 		last;
                    744: 	    }
                    745: 	}
1.100     albertel  746: 	$result.=&element_change_detection()." />$text</label></nobr>\n";
1.61      albertel  747:     }
1.48      albertel  748:     return $result;
1.47      matthew   749: }
                    750: 
1.13      albertel  751: sub text_arg {
1.63      albertel  752:     my ($description,$name,$token,$size) = @_;
                    753:     my $result;
                    754:     if (!defined $size) { $size=20; }
                    755:     my $arg=$token->[2]{$name};
1.71      www       756:     $result=&mt($description).'&nbsp;<input name="'.&html_element_name($name).
1.98      albertel  757: 	'" type="text" value="'.$arg.'" size="'.$size.'" '.
                    758: 	&element_change_detection().'/>';
1.63      albertel  759:     return '<nobr>'.$result.'</nobr>';
1.13      albertel  760: }
                    761: 
                    762: sub select_arg {
1.39      albertel  763:     my ($description,$name,$list,$token) = @_;
                    764:     my $result;
                    765:     my $optionlist="";
                    766:     my $selected=$token->[2]{$name};
                    767:     foreach my $option (@$list) {
1.64      albertel  768: 	my ($text,$value);
1.39      albertel  769: 	if ( ref($option) eq 'ARRAY') {
1.93      albertel  770: 	    $value='value="'.&HTML::Entities::encode($$option[0]).'"';
1.64      albertel  771: 	    $text=$$option[1];
                    772: 	    $option=$$option[0];
1.39      albertel  773: 	} else {
1.64      albertel  774: 	    $text=$option;
1.93      albertel  775: 	    $value='value="'.&HTML::Entities::encode($option,'\'"&<>').'"';
1.39      albertel  776: 	}
                    777: 	if ( $selected eq $option ) {
1.88      albertel  778: 	    $optionlist.="<option $value selected=\"selected\">$text</option>\n";
1.39      albertel  779: 	} else {
1.64      albertel  780: 	    $optionlist.="<option $value >$text</option>\n";
1.39      albertel  781: 	}
1.13      albertel  782:     }
1.57      albertel  783:     $result.='<nobr>'.$description.'&nbsp;<select name="'.
1.98      albertel  784: 	&html_element_name($name).'" '.&element_change_detection().' >
1.13      albertel  785:        '.$optionlist.'
1.57      albertel  786:       </select></nobr>';
1.39      albertel  787:     return $result;
1.13      albertel  788: }
                    789: 
1.19      albertel  790: sub select_or_text_arg {
1.39      albertel  791:     my ($description,$name,$list,$token,$size) = @_;
                    792:     my $result;
                    793:     my $optionlist="";
                    794:     my $found=0;
                    795:     my $selected=$token->[2]{$name};
                    796:     foreach my $option (@$list) {
1.64      albertel  797: 	my ($text,$value);
1.39      albertel  798: 	if ( ref($option) eq 'ARRAY') {
1.93      albertel  799: 	    $value='value="'.&HTML::Entities::encode($$option[0]).'"';
1.64      albertel  800: 	    $text=$$option[1];
                    801: 	    $option=$$option[0];
1.39      albertel  802: 	} else {
1.64      albertel  803: 	    $text=$option;
1.93      albertel  804: 	    $value='value="'.&HTML::Entities::encode($option,'\'"&<>').'"';
1.39      albertel  805: 	}
                    806: 	if ( $selected eq $option ) {
1.88      albertel  807: 	    $optionlist.="<option $value selected=\"selected\">$text</option>\n";
1.39      albertel  808: 	    $found=1;
                    809: 	} else {
1.64      albertel  810: 	    $optionlist.="<option $value>$text</option>\n";
1.39      albertel  811: 	}
                    812:     }
1.60      www       813:     $optionlist.="<option value=\"TYPEDINVALUE\"".
1.88      albertel  814:  	((!$found)?' selected="selected"':'').
1.73      www       815:  	">".&mt('Type-in value')."</option>\n";
1.60      www       816: #
1.98      albertel  817:     my $change_code=&element_change_detection();
1.60      www       818:     my $element=&html_element_name($name);
                    819:     my $selectelement='select_list_'.$element;
                    820:     my $typeinelement='type_in_'.$element;
                    821:     my $typeinvalue=($found?'':$selected);
                    822: #
                    823:     my $hiddenvalue='this.form.'.$element.'.value';
                    824:     my $selectedindex='this.form.'.$selectelement.'.selectedIndex';
                    825:     my $selectedvalue='this.form.'.$selectelement.
                    826: 	     '.options['.$selectedindex.'].value';
                    827:     my $typedinvalue='this.form.'.$typeinelement.'.value';
                    828:     my $selecttypeinindex='this.form.'.$selectelement.'.options.length';
1.71      www       829:     $description=&mt($description);
1.60      www       830: #
                    831:     return (<<ENDSELECTORTYPE);
                    832: <nobr>
                    833: $description
                    834: &nbsp;<select name="$selectelement"
1.74      albertel  835: onChange="if ($selectedvalue!='TYPEDINVALUE') { $hiddenvalue=$selectedvalue; $typedinvalue=''; }" >
1.60      www       836: $optionlist
                    837: </select>
                    838: <input type="text" size="$size" name="$typeinelement"
                    839:        value="$typeinvalue" 
                    840: onChange="$hiddenvalue=$typedinvalue;"
                    841: onFocus="$selectedindex=$selecttypeinindex-1;" />
1.98      albertel  842: <input type="hidden" name="$element" value="$selected" $change_code />
1.60      www       843: </nobr>
                    844: ENDSELECTORTYPE
1.19      albertel  845: }
1.29      matthew   846: 
1.49      www       847: #----------------------------------------------------- image coordinates
                    848: # single image coordinates, x, y 
                    849: sub entercoords {
1.80      albertel  850:     my ($idx,$idy,$mode,$width,$height) = @_;
1.49      www       851:     unless ($Apache::edit::bgimgsrc) { return ''; }
                    852:     if ($idx) { $idx.='_'; }
                    853:     if ($idy) { $idy.='_'; }
1.102     www       854:     my $bgfile=&escape(&Apache::lonnet::filelocation($Apache::lonxml::pwd[-1],$Apache::edit::bgimgsrc));
1.49      www       855:     my $form    = 'lonhomework';
                    856:     my $element;
                    857:     if (! defined($mode) || $mode eq 'attribute') {
1.102     www       858:         $element = &escape("$Apache::lonxml::curdepth");
1.49      www       859:     } elsif ($mode eq 'textnode') {  # for data between <tag> ... </tag>
1.102     www       860:         $element = &escape('homework_edit_'.
1.49      www       861:                                            $Apache::lonxml::curdepth);
                    862:     }
1.80      albertel  863:     my $id=$Apache::lonxml::curdepth;
                    864:     my %data=("imagechoice.$id.type"      =>'point',
                    865: 	      "imagechoice.$id.formname"  =>$form,
                    866: 	      "imagechoice.$id.formx"     =>"$idx$element",
                    867: 	      "imagechoice.$id.formy"     =>"$idy$element",
                    868: 	      "imagechoice.$id.file"      =>$bgfile,
                    869: 	      "imagechoice.$id.formcoord" =>$element);
1.49      www       870:     if ($height) {
1.80      albertel  871: 	$data{"imagechoice.$id.formheight"}=$height.'_'.
                    872: 	    $Apache::edit::bgimgsrccurdepth;
1.49      www       873:     }
                    874:     if ($width) {
1.80      albertel  875: 	$data{"imagechoice.$id.formwidth"}=$width.'_'.
                    876: 	    $Apache::edit::bgimgsrccurdepth;
1.49      www       877:     }
1.80      albertel  878:     &Apache::lonnet::appenv(%data);
                    879:     my $text="Click Coordinates";
                    880:     my $result='<a href="/adm/imagechoice?token='.$id.'" target="imagechoice">'.$text.'</a>';
1.49      www       881:     return $result;
                    882: }
                    883: 
1.77      albertel  884: # coordinates (x1,y1)-(x2,y2)...
                    885: # mode can be either box, or polygon
                    886: sub entercoord {
                    887:     my ($idx,$mode,$width,$height,$type) = @_;
1.49      www       888:     unless ($Apache::edit::bgimgsrc) { return ''; }
1.102     www       889:     my $bgfile=&escape(&Apache::lonnet::filelocation($Apache::lonxml::pwd[-1],$Apache::edit::bgimgsrc));
1.75      albertel  890:     my $form    = 'lonhomework';
                    891:     my $element;
                    892:     if (! defined($mode) || $mode eq 'attribute') {
1.102     www       893:         $element = &escape("$idx\_$Apache::lonxml::curdepth");
1.75      albertel  894:     } elsif ($mode eq 'textnode') {  # for data between <tag> ... </tag>
1.102     www       895:         $element = &escape('homework_edit_'.
1.75      albertel  896:                                            $Apache::lonxml::curdepth);
                    897:     }
1.76      albertel  898:     my $id=$Apache::lonxml::curdepth;
1.77      albertel  899:     my %data=("imagechoice.$id.type"      =>$type,
1.76      albertel  900: 	      "imagechoice.$id.formname"  =>$form,
                    901: 	      "imagechoice.$id.file"      =>$bgfile,
                    902: 	      "imagechoice.$id.formcoord" =>$element);
1.75      albertel  903:     if ($height) {
1.76      albertel  904: 	$data{"imagechoice.$id.formheight"}=$height.'_'.
                    905: 	    $Apache::edit::bgimgsrccurdepth;
1.75      albertel  906:     }
                    907:     if ($width) {
1.76      albertel  908: 	$data{"imagechoice.$id.formwidth"}=$width.'_'.
                    909: 	    $Apache::edit::bgimgsrccurdepth;
1.75      albertel  910:     }
                    911:     &Apache::lonnet::appenv(%data);
1.77      albertel  912:     my $text="Enter Coordinates";
                    913:     if ($type eq 'polygon') { $text='Create Polygon Data'; }
                    914:     my $result='<a href="/adm/imagechoice?token='.$id.'" target="imagechoice">'.$text.'</a>';
1.49      www       915:     return $result;
                    916: }
1.76      albertel  917: 
                    918: sub deletecoorddata {
                    919:     &Apache::lonnet::delenv("imagechoice\\.");
                    920: }
                    921: 
1.29      matthew   922: #----------------------------------------------------- browse
                    923: sub browse {
                    924:     # insert a link to call up the filesystem browser (lonindexer)
1.107     banghart  925:     my ($id, $mode, $titleid, $only) = @_;
1.29      matthew   926:     my $form    = 'lonhomework';
1.42      matthew   927:     my $element;
                    928:     if (! defined($mode) || $mode eq 'attribute') {
1.102     www       929:         $element = &escape("$id\_$Apache::lonxml::curdepth");
1.42      matthew   930:     } elsif ($mode eq 'textnode') {  # for data between <tag> ... </tag>
1.102     www       931:         $element = &escape('homework_edit_'.
1.68      albertel  932:                                            $Apache::lonxml::curdepth);	
                    933:     }
                    934:     my $titleelement;
                    935:     if ($titleid) {
1.107     banghart  936: 	$titleelement=",'$only','','".&escape("$titleid\_$Apache::lonxml::curdepth")."'";
1.108     banghart  937:     } else {
                    938:         $titleelement=",'$only'";
1.42      matthew   939:     }
1.29      matthew   940:     my $result = <<"ENDBUTTON";
1.68      albertel  941: <a href=\"javascript:openbrowser('$form','$element'$titleelement)\"\>Select</a>
1.29      matthew   942: ENDBUTTON
                    943:     return $result;
                    944: }
                    945: 
1.30      matthew   946: #----------------------------------------------------- browse
                    947: sub search {
                    948:     # insert a link to call up the filesystem browser (lonindexer)
1.68      albertel  949:     my ($id, $mode, $titleid) = @_;
1.30      matthew   950:     my $form    = 'lonhomework';
1.49      www       951:     my $element;
                    952:     if (! defined($mode) || $mode eq 'attribute') {
1.102     www       953:         $element = &escape("$id\_$Apache::lonxml::curdepth");
1.49      www       954:     } elsif ($mode eq 'textnode') {  # for data between <tag> ... </tag>
1.102     www       955:         $element = &escape('homework_edit_'.
1.49      www       956:                                            $Apache::lonxml::curdepth);
                    957:     }
1.68      albertel  958:     my $titleelement;
                    959:     if ($titleid) {
1.102     www       960: 	$titleelement=",'".&escape("$titleid\_$Apache::lonxml::curdepth")."'";
1.68      albertel  961:     }
1.30      matthew   962:     my $result = <<"ENDBUTTON";
1.68      albertel  963: <a href=\"javascript:opensearcher('$form','$element'$titleelement)\"\>Search</a>
1.30      matthew   964: ENDBUTTON
                    965:     return $result;
                    966: }
                    967: 
                    968: 
1.1       albertel  969: 1;
                    970: __END__
1.26      harris41  971: 
                    972: =head1 NAME
                    973: 
                    974: Apache::edit - edit mode helpers
                    975: 
                    976: =head1 SYNOPSIS
                    977: 
                    978: Invoked by many homework and xml related modules.
                    979: 
                    980:  &Apache::edit::SUBROUTINENAME(ARGUMENTS);
                    981: 
                    982: =head1 INTRODUCTION
                    983: 
                    984: This module outputs HTML syntax helpful for the rendering of edit
                    985: mode interfaces.
                    986: 
                    987: This is part of the LearningOnline Network with CAPA project
                    988: described at http://www.lon-capa.org.
                    989: 
                    990: =head1 HANDLER SUBROUTINE
                    991: 
                    992: There is no handler subroutine.
                    993: 
                    994: =head1 OTHER SUBROUTINES
                    995: 
                    996: =over 4
                    997: 
                    998: =item *
                    999: 
                   1000: initialize_edit() : initialize edit (set colordepth to zero)
                   1001: 
                   1002: =item *
                   1003: 
                   1004: tag_start($target,$token,$description) : provide deletion and insertion lists
                   1005: for the manipulation of a start tag; return a scalar string
                   1006: 
                   1007: =item *
                   1008: 
                   1009: tag_end($target,$token,$description) : ending syntax corresponding to
                   1010: &tag_start. return a scalar string.
                   1011: 
                   1012: =item *
                   1013: 
                   1014: start_table($token) : start table; update colordepth; return scalar string.
                   1015: 
                   1016: =item *
                   1017: 
                   1018: end_table() : reduce color depth; end table; return scalar string
1.27      matthew  1019: 
                   1020: =item *
                   1021: 
                   1022: start_spanning_row() : start a new table row spanning the 'edit' environment.
                   1023: 
                   1024: =item *
                   1025: 
                   1026: start_row() : start a new table row and element. 
                   1027: 
                   1028: =item *
                   1029: 
                   1030: end_row() : end current table element and row.
1.26      harris41 1031: 
                   1032: =item *
                   1033: 
                   1034: movebuttons($target,$token) : move-up and move-down buttons; return scalar
                   1035: string
                   1036: 
                   1037: =item *
                   1038: 
                   1039: deletelist($target,$token) : provide a yes option in an HTML select element;
                   1040: return scalar string
                   1041: 
                   1042: =item *
                   1043: 
                   1044: handle_delete($space,$target,$token,$tagstack,$parstack,$parser,$safeeval,
                   1045: $style) : respond to a user delete request by passing relevant stack
                   1046: and array information to various rendering functions; return a scalar string
                   1047: 
                   1048: =item *
                   1049: 
                   1050: get_insert_list($token) : provide an insertion list based on possibilities
                   1051: from lonxml; return a scalar string
                   1052: 
                   1053: =item *
                   1054: 
                   1055: insertlist($target,$token) : api that uses get_insert_list;
                   1056: return a scalar string
                   1057: 
                   1058: =item *
                   1059: 
                   1060: handleinsert($token) : provide an insertion list based on possibilities
                   1061: from lonxml; return a scalar string
                   1062: 
                   1063: =item *
                   1064: 
                   1065: get_insert_list($token) : provide an insertion list based on possibilities
                   1066: from lonxml; return a scalar string
1.29      matthew  1067: 
                   1068: =item *
                   1069: browse($elementname) : provide a link which will open up the filesystem
                   1070: browser (lonindexer) and, once a file is selected, place the result in
1.30      matthew  1071: the form element $elementname.
                   1072: 
                   1073: =item *
                   1074: search($elementname) : provide a link which will open up the filesystem
                   1075: searcher (lonsearchcat) and, once a file is selected, place the result in
1.29      matthew  1076: the form element $elementname.
1.31      matthew  1077: 
1.34      harris41 1078: =item *
1.32      albertel 1079: editline(tag,data,description,size): Provide a <input type="text" ../> for
1.31      matthew  1080: single-line text entry.  This is to be used for text enclosed by tags, not
                   1081: arguements/parameters associated with a tag.
1.26      harris41 1082: 
                   1083: =back
                   1084: 
                   1085: incomplete...
                   1086: 
                   1087: =cut

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