Annotation of loncom/publisher/testbankimport.pm, revision 1.20

1.3       albertel    1: # Handler for parsing text upload problem descriptions into .problems
1.20    ! bisitz      2: # $Id: testbankimport.pm,v 1.19 2008/12/19 03:53:00 raeburn Exp $
1.3       albertel    3: #
                      4: # Copyright Michigan State University Board of Trustees
                      5: #
                      6: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
                      7: #
                      8: # LON-CAPA is free software; you can redistribute it and/or modify
                      9: # it under the terms of the GNU General Public License as published by
                     10: # the Free Software Foundation; either version 2 of the License, or
                     11: # (at your option) any later version.
                     12: #
                     13: # LON-CAPA is distributed in the hope that it will be useful,
                     14: # but WITHOUT ANY WARRANTY; without even the implied warranty of
                     15: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     16: # GNU General Public License for more details.
                     17: #
                     18: # You should have received a copy of the GNU General Public License
                     19: # along with LON-CAPA; if not, write to the Free Software
                     20: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
                     21: #
                     22: # /home/httpd/html/adm/gpl.txt
                     23: #
                     24: # http://www.lon-capa.org/
                     25: #
                     26: 
1.1       raeburn    27: package Apache::testbankimport;
                     28: 
1.3       albertel   29: use strict;
                     30: use Apache::Constants qw(:common :http :methods);
                     31: use Apache::loncacc;
                     32: use Apache::loncommon();
                     33: use Apache::lonnet;
                     34: use HTML::Entities();
                     35: use Apache::lonlocal;
                     36: use Apache::lonupload;
1.15      raeburn    37: use Apache::londocs;
1.3       albertel   38: use File::Basename();
1.11      albertel   39: use LONCAPA();
1.15      raeburn    40: use File::MMagic;
                     41: use XML::DOM;
                     42: use RTF::HTMLConverter;
                     43: use HTML::TokeParser;
1.1       raeburn    44: 
                     45: # ---------------------------------------------------------------- Display Control
                     46: sub display_control {
                     47: # figure out what page we're on and where we're heading.
1.6       albertel   48:     my $page = $env{'form.page'};
                     49:     my $command = $env{'form.go'};
1.1       raeburn    50:     my $current_page = &calculate_page($page,$command);
                     51:     return $current_page;
                     52: }
                     53: 
                     54: # CALCULATE THE CURRENT PAGE
                     55: sub calculate_page($$) {
                     56:     my ($prev,$dir) = @_;
                     57:     return 0 if $prev eq '';    # start with first page
                     58:     return $prev + 1 if $dir eq 'NextPage';
                     59:     return $prev - 1 if $dir eq 'PreviousPage';
                     60:     return $prev     if $dir eq 'ExitPage';
                     61:     return 0 if $dir eq 'BackToStart';
                     62: }
                     63: 
1.15      raeburn    64: sub jscript_zero {
                     65:     my ($webpath,$jsref) = @_;
                     66:     my $start_page =
                     67:         &Apache::loncommon::start_page('Create Testbank directory',undef,
                     68:                                        {'only_body'   => 1,
                     69:                                         'js_ready'    => 1,});
                     70:     my $end_page =
                     71:         &Apache::loncommon::end_page({'js_ready' => 1,});
                     72:     my %lt = &Apache::lonlocal::texthash(
                     73:                                          loca => 'Location',
                     74:                                          newd => 'New Directory',
                     75:                                          ente => 'Enter the name of the new directory where you will save the converted testbank questions',
                     76:                                          go  => 'Go',
                     77:                                         );
                     78:     $$jsref = <<"END_SCRIPT";
                     79: function createWin() {
                     80:     document.info.newdir.value = "";
                     81:     newWindow = window.open("","CreateDir","HEIGHT=400,WIDTH=750,scrollbars=yes")
                     82:     newWindow.document.open()
                     83:     newWindow.document.write('$start_page')
                     84:     newWindow.document.write("<img border='0' src='/adm/lonInterFace/author.jpg' alt='[Author Header]'>\\n")
                     85:     newWindow.document.write("<h3>$lt{'loca'}: <tt>$webpath</tt></h3><h3>$lt{'newd'}</h3>\\n")
                     86:     newWindow.document.write("<form name='fileaction' action='/adm/cfile' method='post'>\\n")
                     87:     newWindow.document.write("$lt{'ente'}.<br /><br />")
                     88:     newWindow.document.write("<input type='hidden' name='filename' value='$webpath'>")
                     89:     newWindow.document.write("<input type='hidden' name='action' value='newdir'>")
                     90:     newWindow.document.write("<input type='hidden' name='callingmode' value='testbank'>")
                     91:     newWindow.document.write("$webpath<input type='text' name='newfilename' value=''/>")
                     92:     newWindow.document.write("<input type='button' value='$lt{'go'}' onClick='document.fileaction.submit();' /></form>")
                     93:     newWindow.document.write('$end_page')
                     94:     newWindow.document.close()
                     95:     newWindow.focus()
                     96: }
                     97: 
                     98: END_SCRIPT
                     99:     return;
                    100: }
                    101: 
                    102: 
1.1       raeburn   103: # ---------------------------------------------------------------- Jscript One
                    104: 
                    105: sub jscript_one {
                    106:     my $jsref = shift;
                    107:     $$jsref = <<"END_SCRIPT";
                    108: function verify() {
                    109:     if ((document.forms.display.blocks.value == "") || (!document.forms.display.blocks.value) || (document.forms.display.blocks.value == "0")) {
                    110:         alert("You must enter the number of blocks of questions of a given question type.  This number must be 1 or more.")
                    111:         return false
                    112:     }
                    113:     if (document.forms.display.qnumformat.options[document.forms.display.qnumformat.selectedIndex].value == "-1") {
                    114:         alert("You must select the format used for the question number, e.g., (1), 1., (1, or 1).")
                    115:         return false
                    116:     }
                    117:     return true
                    118: }
                    119: function nextPage() {
                    120:     if (verify()) {
                    121:         document.forms.display.go.value="NextPage"
                    122:         document.forms.display.submit()
                    123:     }
                    124: }
                    125: function backPage() {
                    126:     document.forms.display.go.value="PreviousPage"
                    127:     document.forms.display.submit()
                    128: }
                    129: function setElements() {
                    130:     var iter = 0
                    131:     var selParam = 0
                    132: END_SCRIPT
1.6       albertel  133:     if (exists($env{'form.blocks'}) ) {
1.1       raeburn   134:         $$jsref .= qq|
1.6       albertel  135:     document.forms.display.blocks.value = $env{'form.blocks'}\n|;
1.15      raeburn   136:     }
                    137:     if (exists($env{'form.qnumformat'}) ) {
1.1       raeburn   138:         $$jsref .= <<"TO_HERE";
                    139:     for (iter=0; iter<document.forms.display.qnumformat.length; iter++) {
1.6       albertel  140:         if(document.forms.display.qnumformat.options[iter].value == "$env{'form.qnumformat'}") {
1.1       raeburn   141:             selParam = iter
                    142:         }
                    143:     }
                    144:     document.forms.display.qnumformat.selectedIndex = selParam
                    145: TO_HERE
                    146:     }
                    147:     $$jsref .= qq|
                    148: }
                    149:     |;
                    150: }
                    151: 
                    152: # ---------------------------------------------------------------- Jscript Two
                    153: sub jscript_two {
                    154:     my ($jsref,$qcount) = @_;
                    155:     my $blocks = 0;
1.6       albertel  156:     if ( exists( $env{'form.blocks'}) ) {
                    157:         $blocks = $env{'form.blocks'};
1.1       raeburn   158:     }
                    159:     $$jsref = <<"END_SCRIPT";
                    160: function verify() {
                    161:     var poolForm = document.forms.display
                    162:     var curmax = 0
                    163:     var curmin = 0
                    164:     for (var i=0; i<$blocks; i++) {
                    165:         var iter = i+1
                    166:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "MC") {
                    167:             if (poolForm.elements[5*i+4].selectedIndex == 0) {
                    168:                 alert ("You must choose the foil labelling format in Multiple Choice questions")
                    169:                 return false
                    170:             }
                    171:         }
                    172:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "MA") {
                    173:             if (poolForm.elements[5*i+4].selectedIndex == 0) {
                    174:                 alert ("You must choose the foil labelling format in Multiple Answer questions")
                    175:                 return false
                    176:             }
                    177:             if (poolForm.elements[5*i+5].selectedIndex == 0) {
                    178:                 alert ("You must choose the answer format in Multiple Answer questions") 
                    179:                 return false
                    180:             }
                    181:         }
                    182:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "FIB") {
                    183:             if (poolForm.elements[5*i+5].selectedIndex == 0) {
                    184:                 alert ("You must choose the answer format in Fill-in-the-blank questions") 
                    185:                 return false
                    186:             }
                    187:         }
                    188:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "TF") {
                    189:             if (poolForm.elements[5*i+5].selectedIndex == 0) {
                    190:                 alert ("You must choose the answer format in True/False questions") 
                    191:                 return false
                    192:             }
                    193:         }
                    194:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "Ord") {
                    195:             if (poolForm.elements[5*i+4].selectedIndex == 0) {
                    196:                 alert ("You must choose the foil labelling format in Ranking/ordering questions")
                    197:                 return false
                    198:             }
                    199:             if (poolForm.elements[5*i+5].selectedIndex == 0) {
                    200:                 alert ("You must choose the answer format in Ranking/ordering questions")
                    201:                 return false
                    202:             }
                    203:         }
                    204:         if (poolForm.elements[5*i+3].options[poolForm.elements[5*i+3].selectedIndex].value == "-1") {
                    205:             alert ("You must choose the question type for block "+iter)
                    206:             return false
                    207:         }
                    208:         if ((poolForm.elements[5*i+1].value == "") || !(poolForm.elements[5*i+1].value)) {
                    209:             alert ("You must choose the start number for block "+iter)
                    210:             return false
                    211:         }
                    212:         if ((poolForm.elements[5*i+2].value == "") || !(poolForm.elements[5*i+2].value)) {
                    213:             alert ("You must choose the end number for block "+iter)
                    214:             return false
                    215:         }
                    216:         if (poolForm.elements[5*i+2].value - poolForm.elements[5*i+1].value < 0) {
                    217:             alert ("In block: "+iter+" the end number must be the same or greater than the start number")
                    218:             return false
                    219:         }
                    220:         if (i == 0) {
                    221:             curmin = parseInt(poolForm.elements[5*i+1].value)
                    222:             curmax = parseInt(poolForm.elements[5*i+2].value)
                    223:         }
                    224:         else {
                    225:             if (parseInt(poolForm.elements[5*i+1].value) < curmin) {
                    226:                 if (parseInt(poolForm.elements[5*i+2].value) >= curmin ) {
                    227:                     alert("The question number range for block "+iter+" overlaps with the question number range for one of the previous blocks - this is not permitted.")
                    228:                     return false
                    229:                 }
                    230:             }
                    231:             else {
                    232:                 if (parseInt(poolForm.elements[5*i+1].value) <= curmax) {
                    233:                     for (var j=parseInt(poolForm.elements[5*i+1].value); j<=parseInt(poolForm.elements[5*i+2].value); j++) {
                    234:                         for (var k=0; k<i; k++) {
                    235:                             if ((j >= parseInt(poolForm.elements[5*k+1].value)) && (j <= parseInt(poolForm.elements[5*k+2].value))) {
                    236:                                 var overlap = k+1
                    237:                                 alert("The question number range for block "+iter+" overlaps with the question number range for block "+overlap+" - this is not permitted.")
                    238:                                 return false
                    239:                             }
                    240:                         }
                    241:                     }
                    242:                 }
                    243:             }
                    244:             if (parseInt(poolForm.elements[5*i+1].value) < curmin) {
                    245:                 curmin = parseInt(poolForm.elements[5*i+1].value)
                    246:             }
                    247:             if (parseInt(poolForm.elements[5*i+2].value) > curmax) {
                    248:                 curmax = parseInt(poolForm.elements[5*i+2].value)
                    249:             }
                    250:         }
                    251:     }
                    252:     if (curmax >$qcount+curmin) {
                    253:         alert("The last # for one or more of the blocks is too large -  the last number of the last block can not be greater than $qcount: the total number of questions in the uploaded file.")
                    254:         return false
                    255:     }
                    256:     var endpt = $qcount + curmin
                    257:     for (var n=curmin; n<endpt; n++) {
                    258:         var warnFlag = true
                    259:         for (var m=0; m<$blocks; m++) {
                    260:             if ((n >= parseInt(poolForm.elements[5*m+1].value)) && (n <= parseInt(poolForm.elements[5*m+2].value))) {
                    261:                 warnFlag = false
                    262:             }
                    263:         }
                    264:         if (warnFlag) {
                    265:             alert("The question type for question "+n+" could not be identified because it does not fall within the number ranges you have provided for any of the $blocks block(s)")
                    266:             return false
                    267:         }
                    268:     } 
                    269:     return true 
                    270: }
                    271:  
                    272: function nextPage() {
                    273:     if (verify()) {
                    274:         document.forms.display.go.value="NextPage"
                    275:         document.forms.display.submit()
                    276:     }
                    277: }
                    278: function backPage() {
                    279:     document.forms.display.go.value="PreviousPage"
                    280:     document.forms.display.submit()
                    281: }
                    282: function colSet(caller) {
                    283:     var poolForm = document.forms.display
                    284:     var curVal = poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value  
                    285:     poolForm.elements[caller*5+4].length = 0
                    286:     if (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "-1") {
                    287:         poolForm.elements[caller*5+4].options[0] = new Option("<--- Set type ","-1",true,true)
                    288:     }
                    289:     else {
                    290:         if ((poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "MC") || (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "MA") || (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "Ord")) {
1.15      raeburn   291:             poolForm.elements[caller*5+4].options[0] = new Option("Select","-1",true,true)
1.1       raeburn   292:             poolForm.elements[caller*5+4].options[1] = new Option("a.","lcperiod",false,false)
                    293:             poolForm.elements[caller*5+4].options[2] = new Option("A.","ucperiod",false,false)
                    294:             poolForm.elements[caller*5+4].options[3] = new Option("(a)","lcparen",false,false)
                    295:             poolForm.elements[caller*5+4].options[4] = new Option("(A)","ucparen",false,false)
1.5       raeburn   296:             poolForm.elements[caller*5+4].options[5] = new Option("a)","lconeparen",false,false)
                    297:             poolForm.elements[caller*5+4].options[6] = new Option("A)","uconeparen",false,false)
                    298:             poolForm.elements[caller*5+4].options[7] = new Option("a.)","lcdotparen",false,false)
                    299:             poolForm.elements[caller*5+4].options[8] = new Option("A.)","ucdotparen",false,false)
                    300:             poolForm.elements[caller*5+4].options[9] = new Option("(i)","romparen",false,false)
                    301:             poolForm.elements[caller*5+4].options[10] = new Option("i)","romoneparen",false,false)
                    302:             poolForm.elements[caller*5+4].options[11] = new Option("i.)","romdotparen",false,false)
                    303:             poolForm.elements[caller*5+4].options[12] = new Option("i.","romperiod",false,false)
1.1       raeburn   304:             poolForm.elements[caller*5+4].selectedIndex = 0
                    305:         }
                    306:         else {
                    307:             poolForm.elements[caller*5+4].options[0] = new Option("Not required","0",true,true)
                    308:         }
                    309:     }
                    310:     poolForm.elements[caller*5+5].length = 0
                    311:     if (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "-1") {
                    312:         poolForm.elements[caller*5+5].options[0] = new Option("<--- Set type ","-1",true,true)
                    313:     }
                    314:     else {
                    315:         if ((poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "MA") || (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "FIB"))  {
1.15      raeburn   316:             poolForm.elements[caller*5+5].options[0] = new Option("Select","-1",true,true)
1.1       raeburn   317:             poolForm.elements[caller*5+5].options[1] = new Option("single answer","single",false,false)
                    318:             poolForm.elements[caller*5+5].options[2] = new Option("comma","comma",false,false)
                    319:             poolForm.elements[caller*5+5].options[3] = new Option("space","space",false,false)
                    320:             poolForm.elements[caller*5+5].options[4] = new Option("new line","line",false,false)
                    321:             poolForm.elements[caller*5+5].options[5] = new Option("tab","tab",false,false)
                    322:         }
                    323:         else {
                    324:             if (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "Ord") {
1.15      raeburn   325:                 poolForm.elements[caller*5+5].options[0] = new Option("Select","-1",true,true)
1.1       raeburn   326:                 poolForm.elements[caller*5+5].options[1] = new Option("comma","comma",false,false)
                    327:                 poolForm.elements[caller*5+5].options[2] = new Option("space","space",false,false)
                    328:                 poolForm.elements[caller*5+5].options[3] = new Option("new line","line",false,false)
                    329:                 poolForm.elements[caller*5+5].options[4] = new Option("tab","tab",false,false)
                    330:             }
                    331:             else { 
                    332:                 if (poolForm.elements[caller*5+3].options[poolForm.elements[caller*5+3].selectedIndex].value == "TF") {
1.15      raeburn   333:                     poolForm.elements[caller*5+5].options[0] = new Option("Select","-1",true,true)
1.1       raeburn   334:                     poolForm.elements[caller*5+5].options[1] = new Option("True or False","word",false,false)
1.5       raeburn   335:                     poolForm.elements[caller*5+5].options[2] = new Option("true or false","word",false,false)
                    336:                     poolForm.elements[caller*5+5].options[3] = new Option("TRUE or FALSE","word",false,false)
                    337:                     poolForm.elements[caller*5+5].options[4] = new Option("T or F","lett",false,false)
                    338:                     poolForm.elements[caller*5+5].options[5] = new Option("t or f","lett",false,false)
1.1       raeburn   339:                 }
                    340:                 else {
                    341:                     poolForm.elements[caller*5+5].options[0] = new Option("Not required","0",true,true)
                    342:                 }
                    343:             }
                    344:         }
                    345:     }
                    346: }
                    347: 
                    348: function setElements() {
                    349:     var iter = 0
                    350:     var selParam = 0
                    351: END_SCRIPT
                    352:     my @names = ("start_","end_","qtype_","foilformat_","ansr_");
                    353:     for (my $x=0; $x<$blocks; $x++) {
                    354:         foreach my $name (@names) {
                    355:             my $parname = $name.$x;
1.6       albertel  356:             my $value = $env{"form.$parname"};
1.1       raeburn   357:             if ($value ne "") {
                    358:                 if (($name eq "start_")  || ($name eq "end_")) {
                    359:                     $$jsref .= qq|
                    360:     document.forms.display.$parname.value = $value\n|;
                    361:                 } elsif ($name eq "qtype_") {
                    362:                     $$jsref .= qq|
                    363:     for (iter=0; iter<document.forms.display.$parname.length; iter++) {
                    364:         if (document.forms.display.$parname.options[iter].value == "$value") {
                    365:             selParam = iter
                    366:         }
                    367:     }
                    368:     document.forms.display.$parname.selectedIndex = selParam
                    369:     colSet($x)
                    370:                     |;
                    371:                 } elsif (($name eq "foilformat_") || ($name eq "ansr_")) {
                    372:                     $$jsref .= <<"TO_HERE";
                    373:     for (iter=0; iter<document.forms.display.$parname.length; iter++) {
                    374:         if (document.forms.display.$parname.options[iter].value == "$value") {
                    375:             selParam = iter
                    376:         }
                    377:     }
                    378:     document.forms.display.$parname.selectedIndex = selParam
                    379: TO_HERE
                    380:                 } 
                    381:             }
                    382:         }
                    383:     }
                    384:     $$jsref .= qq|
                    385: }
                    386:     |;
                    387: } 
                    388: # ---------------------------------------------------------------- Jscript Three
                    389: 
                    390: sub jscript_three {
1.15      raeburn   391:     my ($webpath,$jsref) = @_;
1.1       raeburn   392:     my $source = '';
1.6       albertel  393:     if (exists($env{'form.go'}) ) {
                    394:         $source = $env{'form.go'};
1.1       raeburn   395:     }
1.8       albertel  396: 
1.1       raeburn   397:     $$jsref = <<"END_OF_ONE";
                    398: function nextPage() {
                    399:     if (verify()) {
                    400:         document.forms.dataForm.go.value="NextPage"
1.15      raeburn   401:         document.forms.dataForm.submit();
1.1       raeburn   402:     }
                    403: }
1.15      raeburn   404: 
1.1       raeburn   405: function backPage() {
                    406:     document.forms.dataForm.go.value="PreviousPage"
                    407:     document.forms.dataForm.submit()
                    408: }
                    409: 
                    410: END_OF_ONE
                    411:     if ($source eq "PreviousPage") { 
                    412:         $$jsref .= qq|  
                    413: function setElements() {
                    414:     var iter = 0
                    415:     var selParam = 0
                    416:         |;
1.15      raeburn   417:         foreach my $item (keys(%env)) {
                    418:             if ($item =~ m/^form\.(probfile_\d+)$/) {
1.1       raeburn   419:                 my $name = $1; 
1.6       albertel  420:                 my $value = $env{"form.$name"};
1.15      raeburn   421:                 if ($value ne '') {
                    422:              	    $$jsref .= qq(    document.dataForm.$name.value = "$value"\n);
1.1       raeburn   423:                 }
                    424:             }
                    425:         }
                    426:         $$jsref .= "}";
                    427:     }
1.15      raeburn   428:     $$jsref .= '
                    429: 
                    430: function verify() {
                    431: ';
                    432:     my $blocks = 0;
                    433:     if ( exists( $env{'form.blocks'}) ) {
                    434:         $blocks = $env{'form.blocks'};
                    435:     }
                    436:     my $numitems = 0;
                    437:     for (my $i=0; $i<$blocks; $i++) {
                    438:         my $count = 0;
                    439:         if (($env{"form.start_$i"} ne '') && ($env{"form.end_$i"} ne '')) {
                    440:             $count = $env{"form.end_$i"} - $env{"form.start_$i"} +1;
                    441:         }
                    442:         $numitems += $count;
                    443:     }
                    444:     if ($numitems > 0) {
                    445:         my $maxnum = $numitems - 1;
                    446:         my %lt = &Apache::lonlocal::texthash(
                    447:                                               fnmb => 'File names must be unique',
                    448:                                               isum => 'is used more than once',
                    449:                                             );
                    450:         $$jsref .= qq|
                    451:     for (var j=$maxnum; j>0;  j--) {
                    452:         var currname = document.dataForm.elements['probfile_'+j].value;
                    453:         for (var k=j-1; k>=0; k--) {
                    454:             var comparename = document.dataForm.elements['probfile_'+k].value;
                    455:             if (currname == comparename) {
                    456:                 alert("$lt{fnmb} - "+currname+" $lt{isum}");
                    457:                 return false;
                    458:             }
                    459:         }
                    460:     }
                    461: |;
                    462:     }
                    463:     $$jsref .= '
                    464:     return true;
                    465: }
                    466: ';
                    467:     $$jsref .= &Apache::loncommon::check_uncheck_jscript();
                    468:     return;
1.1       raeburn   469: }
                    470: 
                    471: # ---------------------------------------------------------------- Jscript Four
                    472: sub jscript_four {
1.15      raeburn   473:     my ($jsref,$webpath) = @_;
1.1       raeburn   474:     $$jsref = qq|   
                    475: function backtoStart() {
1.15      raeburn   476:     document.location.href="$webpath"
1.1       raeburn   477: }
1.15      raeburn   478: function backPage() {
1.1       raeburn   479:     document.forms.verify.go.value="PreviousPage"
1.15      raeburn   480:     document.forms.verify.submit();
1.1       raeburn   481: }
                    482:     |;
                    483: }
                    484: 
                    485: # ---------------------------------------------------------------- Display Zero
                    486: sub display_zero {
1.15      raeburn   487:     my ($r,$uname,$fn,$page,$webpath) = @_;
                    488:     my $go_default = 'NextPage'; 
                    489:     if ($fn eq '') {
                    490:         $r->print('<b>'.&mt('Incomplete file upload').'</b> '.&mt('Return to the [_1]construction space menu[_2] to upload a file','<a href="'.$webpath.'">','</a>'));
                    491:     }
                    492:     $r->print(&mt('The <b>Testbank Upload</b> utility can be used by LON-CAPA authors to generate LON-CAPA problem files from a testbank file of questions/answers.').'<br />'.
                    493:               &mt('The following question types can be converted:').'
                    494:               <ul>
                    495:                 <li>'.&mt('multiple choice').'</li>
                    496:                 <li>'.&mt('multiple answer correct').'</li>
                    497:                 <li>'.&mt('fill-in-the-blank').'</li>
                    498:                 <li>'.&mt('ordering/ranking').'</li>
                    499:                 <li>'.&mt('true/false').'</li>
                    500:                 <li>'.&mt('essay').'</li>
                    501:               </ul>
                    502:               '.&mt('The file of questions (in plain text, RTF or HTML format) must meet certain requirements for the conversion process to generate functioning LON-CAPA problems.').&Apache::loncommon::help_open_topic('Testbank_Formatting').'<br />'.
                    503:               &mt('Five steps are involved in the conversion process.').'
1.1       raeburn   504:         <ol>
1.15      raeburn   505:          <li>'.&mt('Optionally create a new sub-directory where the converted testbank questions will be saved.').'</li>
                    506:          <li>'.&mt('Provide information about the question format - i.e.,  question numbering style, and the number of blocks of questions of each question type.').'</li>
                    507:          <li>'.&mt('Provide information about the questions in each block, including question type, start and end question numbers for each block, and foil labelling style and answer format where required.').'</li>
                    508:          <li>'.&mt('Review the identified questions, choose which to convert, and (optionally) override the default filename to be used for each problem file.').'</li> 
                    509:          <li>'.&mt('Complete the import of questions.').'</li>
                    510:         </ol><form name="info" method="post" action="/adm/testbank">'.
                    511:         &topic_bar(1,&mt('Optional: create a sub-directory in which the testbank questions will be saved')).
                    512:         &mt('By default, LON-CAPA problems generated from the testbank file will be stored in the current directory.').' '.&mt('To store them in a new sub-directory:'). 
                    513:        ' <input type="button" name="createdir" value="'.&mt('Create sub-directory').'" onClick="javascript:createWin()">'.
                    514:        &page_footer($env{'form.newdir'},$uname,$fn,$page,$webpath).'
                    515:        </form>');
1.1       raeburn   516: }
                    517: 
                    518: # ---------------------------------------------------------------- Display One
                    519: 
                    520: sub display_one {
1.15      raeburn   521:     my ($r,$uname,$fn,$page,$textref,$header) = @_;
                    522:     my %topics;
                    523:     $topics{2} = &mt('Select the format of the question number - e.g., 1,  1., 1), (1 or (1) - ').'&nbsp;
                    524:                <select name="qnumformat">
                    525:                   <option value = "-1" selected>'.&mt('Select').'</option>
                    526:                   <option value="number">1</option>
                    527:                   <option value="period">1.</option>
                    528:                   <option value="paren">(1)</option>
                    529:                   <option value="leadparen">(1</option>
                    530:                   <option value="trailparen">1)</option>
                    531:                  </select>'."\n";
                    532:     $topics{3} = &mt('Indicate the number of blocks of different question types in the testbank file.').'&nbsp;&nbsp;<input type="text" name="blocks" value="" size="5" />';
                    533:     $r->print('<h3>'.&mt('Identification of blocks of questions').'</h3>'."\n".
                    534:               '<form method="post" name="display" action="/adm/testbank">'."\n".
                    535:               &show_uploaded_data($textref,$header)."\n".
                    536:               &topic_bar(2,$topics{2}).'<p>'.
                    537:               &mt('A number in the specified format should appear at the start of each question.').'<br />'.
                    538:               &mt('For multiple choice questions, the question number must begin the line that contains the question text; foils (starting (a), (i) etc.) should occur on subsequent lines.').'<br />'."\n".
                    539:               &mt('Correct answers should be numbered in the same way as the questions and should appear after <b>all</b> the questions (including question text and possible foils for all questions).').'<br />'."\n".
                    540:               &mt('Each numbered question must have a corresponding numbered answer, although the answer itself may be blank for essay questions.').'<br /><br />'."\n".
                    541:               &mt('For example, you would select <b>1.</b> if your testbank file contained the following questions:').'<br /><blockquote>'.
                    542: '<pre>
                    543:  1. '.&mt('The capital of the USA is ...').'
                    544:  (a) Washington D.C.
                    545:  (b) New York
                    546:  (c) Los Angeles
                    547: 
                    548:  2. '.&mt('The capital of Canada is ...').'
                    549:  (a) Toronto
                    550:  (b) Vancouver
                    551:  (c) Ottawa
                    552: 
                    553:  3. '.&mt('Describe an experiment you could conduct to measure c, the speed of light in a vacuum.').'
                    554:  1. (a)
                    555:  2. (c)
                    556:  3.
                    557: </pre>'.
                    558:              '</blockquote></p>'.
                    559:              &topic_bar(3,$topics{3}).'<p>'.
                    560:              &mt('For example, you would enter <b>6</b> if your testbank file contained the following sequence of questions:').'</p><blockquote>'.
                    561:              &mt('10 multiple choice questions').'<br />'.
                    562:              &mt('5 essay questions').'<br />'.
                    563:              &mt('5 fill-in-the-blank questions').'<br />'.
                    564:              &mt('5 multiple answer questions').'<br />'.
                    565:              &mt('4 multiple choice questions').'<br />'.
                    566:              &mt('3 essay questions').'</blockquote></p><p>'.
                    567:              &mt('You will indicate the question type and the question number range for each of the blocks on the next page.').'</p><br />'.
                    568:              &page_footer($env{'form.newdir'},$uname,$fn,$page).'
                    569:  </form>');
                    570:     return;
1.1       raeburn   571: }
                    572: 
                    573: # ---------------------------------------------------------------- Display Two
                    574: 
                    575: sub display_two {
1.15      raeburn   576:     my ($r,$uname,$fn,$page,$textref,$header,$qcount) = @_;
1.6       albertel  577:     my $blocks = $env{'form.blocks'};
                    578:     my $qnumformat = $env{'form.qnumformat'};
1.1       raeburn   579:     my @types = ("MC","MA","TF","Ess","FIB","Ord");
                    580:     my %typenames = (
                    581:              MC => "Multiple Choice",
                    582:              TF => "True/False",
                    583:              MA => "Multiple Answer",
                    584:              Ess => "Essay",
                    585:              FIB => "Fill-in-the-blank",
                    586:              Ord => "Ranking/ordering",
                    587:              );
                    588:     my %qnumtypes = (
                    589:              number => "1",
                    590:              period => "1.",
                    591:              paren => "(1)",
                    592:              leadparen => "(1",
                    593:              trailparen => "1)",
                    594:              );
                    595:     my $bl1st = '';
                    596:     my $bl1end = '';
                    597:     if ($blocks == 1) {
                    598:         $bl1st = '1';
                    599:         $bl1end = $qcount;
                    600:     }
1.15      raeburn   601:     my $steptitle = &mt('Information about question types and formats in each block.');
                    602:     $r->print('<h3>'.&mt('Classification of blocks').'</h3>'.
                    603:               '<form method="post" name="display" action="/adm/testbank"><p>'.
                    604:               &mt('You indicated that <b>all</b> questions (and the corresponding answer(s) for each question) begin with a number in the following format: [_1].','<b>'.$qnumtypes{$qnumformat}.'</b>').'</p><p>'.
                    605:               &mt('A total of <b>[quant,_1,question]</b> and <b>[quant,_2,answer]</b> were found in the file you uploaded.',$qcount,$qcount).' '.
                    606:               &mt('If this total does not match the number you expect, examine your original testbank file to verify that each question <i>and</i> each answer begins with a number in the specified format.').' '.
                    607:               &mt('If necessary use an editor to edit your testbank file of questions, and click "Previous Page" on this page and the "Exit Now" on the preceding page, so you can upload your file again.').'</p><p>'.
                    608:               &mt('You also indicated that the <b>[quant,_1,question]</b> can be divided into <b>[quant,_2,block]</b> of questions of a particular question type.',$qcount,$blocks).'</p><p>'.
                    609:               &mt('Provide additional information below, about the types of questions you have uploaded, and, if applicable, the format of answers and "foils" for specific types of questions.').'</p>'.
                    610:               &show_uploaded_data($textref,$header).
                    611:               &topic_bar(4,$steptitle).'<p>'.
                    612:               &mt('For <i>each</i> of the [_1] question blocks, specify the question numbers of the first and last questions in the block (e.g., 1 and 10), and the question type of the questions in the block.','<b>'.$blocks.'</b>').' '.
                    613:               &mt('If required, provide additional information about foil formats and answer formats for the question types you select.').'</p><p>'.
                    614:               &Apache::loncommon::start_data_table().
                    615:               &Apache::loncommon::start_data_table_header_row().
                    616:               '<th>'.&mt('Block').'</th>'."\n".
                    617:               '<th>'.&mt('First number').'</th>'."\n".
                    618:               '<th>'.&mt('Last number').'</th>'."\n".
                    619:               '<th>'.&mt('Question type').'</th>'."\n".
                    620:               '<th>'.&mt('Foil format').'</th>'."\n".
                    621:               '<th>'.&mt('Answer format').'</th>'."\n".
                    622:               &Apache::loncommon::end_data_table_header_row());
1.1       raeburn   623:     for (my $i=0; $i<$blocks; $i++) {
                    624:         my $iter = $i+1;
1.15      raeburn   625:         $r->print(&Apache::loncommon::start_data_table_row().
                    626:                  '<td valign="top">&nbsp;'.$iter.'&nbsp;</td>'."\n".
                    627:                  '<td valign="top">&nbsp;<input type="text" name="start_'.$i.'" value="'.$bl1st.'" size="5" />&nbsp;</td>'."\n".
                    628:                  '<td valign="top">&nbsp;<input type="text" name="end_'.$i.'" value="'.$bl1end.'" size="5">&nbsp;</td>'."\n".
                    629:                  '<td valign="top">
                    630:    <select name="qtype_'.$i.'" onChange="colSet('.$i.')">
                    631:     <option value= "-1" selected>'.&mt('Select').'</option>'."\n");
1.1       raeburn   632:         foreach my $qtype (@types) {
1.15      raeburn   633:             $r->print('<option value="'.$qtype.'">'.$typenames{$qtype}.'</option>'."\n");
1.1       raeburn   634:         }
1.15      raeburn   635:         $r->print('   </select>
1.1       raeburn   636:   </td>
1.15      raeburn   637:   <td align="left" valign="top">&nbsp;
                    638:     <select name="foilformat_'.$i.'">
1.1       raeburn   639:      <option value="-1">&lt;--- Set type&nbsp; 
                    640:     </select>&nbsp;
                    641:   </td>
1.15      raeburn   642:   <td align="left" valign="top">&nbsp;
                    643:     <select name="ansr_'.$i.'">
1.1       raeburn   644:      <option value="-1">&lt;--- Set type&nbsp;
                    645:     </select>
1.15      raeburn   646:   </td>'.
                    647:                      &Apache::loncommon::end_data_table_row()); 
1.1       raeburn   648:     }
1.15      raeburn   649:     $r->print(&Apache::loncommon::end_data_table().'</p><ul><li>'.
                    650:               &mt('For <i>multiple choice</i>, <i>multiple correct answer</i> and <i>ranking</i> type questions, you must use the <b>Foil format</b> column to choose the format of the identifier used for each of the possible answers (e.g., (a), a, a., i, (i) etc.) provided for a given question stem.').'</li><li>'.
                    651:              &mt('For <i>multiple correct answer</i> and <i>fill-in-the-blank</i> questions with more than one correct answer you must use the <b>Answer format</b> column to choose the separator used between the answers, e.g., if the correct answers for question 28. were listed as:[_1] you would choose "comma", or if they were listed as:[_2] you would choose "new line".','<blockquote><pre>28. (a),(d),(e)</pre></blockquote>','<blockquote><pre>
                    652: 28. (a)
                    653:     (d)
                    654:     (e)
                    655: </pre></blockquote>').'</li><li>'.
                    656:              &mt('For <i>true/false</i> questions you must use the <b>Answer format</b> column to choose how the correct answer - True or False, is displayed in the text file (e.g., T or F, true or false etc.).').'</li><li>'.
                    657:             &mt('For <i>ranking</i> questions you must use the <b>Answer format</b> column to choose the separator used between the (ranked) answers.').'</li></ul>
                    658: <input type="hidden" name="blocks" value="'.$blocks.'" />
                    659: <input type="hidden" name="qnumformat" value="'.$qnumformat.'" />'.
                    660:            &page_footer($env{'form.newdir'},$uname,$fn,$page).'
                    661: </form>');
                    662:     return;
                    663: }
                    664: 
1.1       raeburn   665: # ---------------------------------------------------------------- Display Three
1.15      raeburn   666: sub display_three {
                    667:     my ($r,$uname,$fn,$page,$textref,$res,$header,$urlpath,$qcount) = @_;
1.6       albertel  668:     my $qnumformat = $env{'form.qnumformat'};
                    669:     my $filename = $env{'form.filename'};
                    670:     my $source = $env{'form.go'};
                    671:     my $blocks = $env{'form.blocks'};
1.15      raeburn   672:     my ($alphabet,$romans) = &get_constants();
1.1       raeburn   673:     my @start = ();
                    674:     my @end = ();
                    675:     my @nums = ();
                    676:     my @qtype = ();
                    677:     my @foilformats = ();
                    678:     my @ansrtypes = ();
                    679:     my %multparts = ();
                    680:     my $numitems = 0;
1.15      raeburn   681:     my %lt = &Apache::lonlocal::texthash (
                    682:                                           crt  => 'Create?',
                    683:                                           typ  => 'Type',
                    684:                                           fnam => 'File Name',
                    685:                                           ques => 'Question',
                    686:                                           answ => 'Answer',
                    687:                                           chka => 'check all',
                    688:                                           unch => 'uncheck all',
                    689:                                          );
1.1       raeburn   690:     for (my $i=0; $i<$blocks; $i++) {
1.6       albertel  691:         if (($env{"form.start_$i"} ne '') && ($env{"form.end_$i"} ne '')) {
                    692:             $start[$i] = $env{"form.start_$i"};
                    693:             $end[$i] = $env{"form.end_$i"};
1.1       raeburn   694:             $nums[$i] = $end[$i]-$start[$i] +1;
1.6       albertel  695:             $qtype[$i] = $env{"form.qtype_$i"};
1.1       raeburn   696:             if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "Ord")) {
1.6       albertel  697:                 $foilformats[$i] = $env{"form.foilformat_$i"};
1.1       raeburn   698:             } else {
                    699:                 $foilformats[$i] = '';
                    700:             } 
                    701:             if (($qtype[$i] eq "MA") || ($qtype[$i] eq "FIB") || ($qtype[$i] eq "TF") || ($qtype[$i] eq "Ord")) {
1.6       albertel  702:                 $ansrtypes[$i] = $env{"form.ansr_$i"};
1.1       raeburn   703:             } else {
                    704:                 $ansrtypes[$i] = '';
                    705:             }  
                    706:         } else { 
                    707:             $nums[$i] = 0;
                    708:         }
                    709:         $numitems += $nums[$i];
                    710:     }
1.15      raeburn   711:     my ($items,$ids,$footer) = &file_split(\@start,\@end,\@nums,$qnumformat,\@foilformats,$textref,\%multparts,$numitems,\@qtype,$blocks);
                    712:     my ($showheader,$showcss);
                    713:     if ($res eq 'application/rtf' || $res eq 'text/html') {
                    714:         if ($header ne '') {
                    715:             $showheader = &HTML::Entities::decode($header);
                    716:             if ($res eq 'text/html') {
                    717:                 $showheader = &build_image_url($urlpath,$showheader);
                    718:             }
                    719:         }
                    720:     }
                    721:     $r->print('<h3>'.&mt('Review and selection of problems to convert').'</h3>'."\n".
                    722:               '<form name="dataForm" method="post" action="/adm/testbank">'."\n".
                    723:               &mt('Based on your previous responses your data have been split into a total of [quant,_1,question].',$numitems).
                    724:               &topic_bar(5,&mt('Choose which problems to convert and names to use for individual problem files')));
                    725:               if ($showheader) {
                    726:                   $r->print($showheader.'<br />');
                    727:               }
                    728:               $r->print('<input type="button" value="'.$lt{'chka'}.'" onclick="javascript:checkAll(document.dataForm.createprob)" /> &nbsp;
                    729: <input type="button" value="'.$lt{'unch'}.'" onclick="javascript:uncheckAll(document.dataForm.createprob)" /><br /><br />'.
                    730:               &Apache::loncommon::start_data_table().
                    731:               &Apache::loncommon::start_data_table_header_row(). 
                    732:               '<th>'.#'.</th>'.
                    733:               '<th>'.$lt{'crt'}.'</th>'.
                    734:               '<th>'.$lt{'typ'}.'</th>'.
                    735:               '<th>'.$lt{'fnam'}.'</th>'.
                    736:               '<th>'.$lt{'ques'}.'</th>'.
                    737:               '<th>'.$lt{'answ'}.'</th>'.
                    738:               &Apache::loncommon::end_data_table_header_row());
                    739:     my $idx;
                    740:     if ($numitems =~ /^\d+$/ && $numitems > 0) {
                    741:         $idx = int(log($numitems)/log(10));
                    742:         $idx ++;
                    743:     }
                    744:     if ($idx<3) {
                    745:         $idx = 3;
                    746:     }
1.1       raeburn   747:     for (my $j=0; $j<$numitems; $j++) {
1.15      raeburn   748:         my $qnum = $ids->[$j]; 
                    749:         my $libfile = 'question_';
                    750:         my $leading = 0;
                    751:         while (($idx - length($qnum) - $leading) > 0) {   
                    752:             $libfile .= '0';
                    753:             $leading ++;
                    754:         }
                    755:         $libfile .= $qnum.'.problem';
1.1       raeburn   756:         for (my $i=0; $i<$blocks; $i++) {
                    757:             if ($nums[$i] > 0) {
                    758:                 if (($j+1 >= $start[$i]) && ($j+1 <= $end[$i])) { 
                    759:                     if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA")) { 
                    760:                         for (my $k=0; $k<@{$multparts{$j}}; $k++) {
                    761:                             if ($k == 0) {
1.15      raeburn   762:                                 my $showqn = $multparts{$j}[$k];
                    763:                                 if (($res eq 'application/rtf') || ($res eq 'text/html')) {
                    764:                                     $showqn = &HTML::Entities::decode($showqn);
                    765:                                     if ($res eq 'text/html') {
                    766:                                         $showqn = &build_image_url($urlpath,$showqn);
                    767:                                     }
                    768:                                 }
                    769:                                 $r->print(&Apache::loncommon::start_data_table_row().
                    770:                                           '<td valign="top">'.$qnum.'.</td>'."\n".
                    771:                                           '<td valign="top"><input name="createprob" type="checkbox" "checked="checked" value="'.$j.'" /></td>'."\n".
                    772:                                           '<td valign="top"><b>'.$qtype[$i].'</b></td>'."\n".
                    773:                                           '<td valign="top"><input type="textbox" name="probfile_'.$j.'" value="'.$libfile.'" size="20" /></td>'.
                    774:                                           '<td valign="top">'.$showqn.'<br /><br />'."\n");
                    775:                             } else {
1.1       raeburn   776:                                 my $foiltag = '';
                    777:                                 if ($foilformats[$i] eq "lcperiod") {
1.15      raeburn   778:                                     $foiltag = $alphabet->[$k-1].'.'; 
1.1       raeburn   779:                                 } elsif ($foilformats[$i] eq "lcparen") {
1.15      raeburn   780:                                     $foiltag = '('.$alphabet->[$k-1].')';
1.5       raeburn   781:                                 } elsif ($foilformats[$i] eq "lconeparen") {
1.15      raeburn   782:                                     $foiltag = $alphabet->[$k-1].')';
1.5       raeburn   783:                                 } elsif ($foilformats[$i] eq "lcdotparen") {
1.15      raeburn   784:                                     $foiltag = $alphabet->[$k-1].'.)';
1.1       raeburn   785:                                 } elsif ($foilformats[$i] eq "ucperiod") {
1.15      raeburn   786:                                     $foiltag = $alphabet->[$k-1].'.';
1.1       raeburn   787:                                     $foiltag =~ tr/a-z/A-Z/;
                    788:                                 } elsif ($foilformats[$i] eq "ucparen") {
1.15      raeburn   789:                                     $foiltag = '('.$alphabet->[$k-1].')';
1.1       raeburn   790:                                     $foiltag =~ tr/a-z/A-Z/;
1.5       raeburn   791:                                 } elsif ($foilformats[$i] eq "uconeparen") {
1.15      raeburn   792:                                     $foiltag = $alphabet->[$k-1].')';
1.5       raeburn   793:                                     $foiltag =~ tr/a-z/A-Z/;
                    794:                                 } elsif ($foilformats[$i] eq "ucdotparen") {
1.15      raeburn   795:                                     $foiltag = $alphabet->[$k-1].'.)';
1.5       raeburn   796:                                     $foiltag =~ tr/a-z/A-Z/;
1.1       raeburn   797:                                 } elsif ($foilformats[$i] eq "romperiod") {
1.15      raeburn   798:                                     $foiltag = $romans->[$k-1].'.';
1.1       raeburn   799:                                 } elsif ($foilformats[$i] eq "romparen") {
1.15      raeburn   800:                                     $foiltag = '('.$romans->[$k-1].')';
1.5       raeburn   801:                                 } elsif ($foilformats[$i] eq "romoneparen") {
1.15      raeburn   802:                                     $foiltag = $romans->[$k-1].')';
1.5       raeburn   803:                                 } elsif ($foilformats[$i] eq "romdotparen") {
1.15      raeburn   804:                                     $foiltag = $romans->[$k-1].'.)';
                    805:                                 }
                    806:                                 my $showfoil = $multparts{$j}[$k];
                    807:                                 if ($res eq 'application/rtf' || $res eq 'text/html') {
                    808:                                     $showfoil = &HTML::Entities::decode($showfoil);
                    809:                                     if ($res eq 'text/html') {
                    810:                                         $showfoil = &build_image_url($urlpath,$showfoil);
                    811:                                     }
1.5       raeburn   812:                                 } 
1.15      raeburn   813:                                 $r->print("$foiltag $showfoil<br />\n");
1.1       raeburn   814:                             }
                    815:                         }
1.15      raeburn   816:                         my $showfoil = $items->[$j+$numitems];
                    817:                         if ($res eq 'application/rtf' || $res eq 'text/html') {
                    818:                             $showfoil = &HTML::Entities::decode($showfoil);
                    819:                             $showfoil =~ s/<\/?[^>]+>//g;
                    820:                         }
                    821: 
                    822:                         $r->print('<br /></td><td valign="top">'.$showfoil.'</td>'.
                    823:                                   &Apache::loncommon::end_data_table_row());
1.1       raeburn   824:                     } else {
1.15      raeburn   825:                         my $showfoil = $items->[$j+$numitems];
                    826:                         if ($res eq 'application/rtf' || $res eq 'text/html') {
                    827:                             $showfoil = &HTML::Entities::decode($showfoil);
                    828:                             $showfoil =~ s/<\/?[^>]+>//g;
                    829:                         }
                    830:                         $r->print(&Apache::loncommon::start_data_table_row().
                    831:                                   '<td valign="top">'.$qnum.'</td>'."\n".
                    832:                                   '<td valign="top"><input name="createprob" type="checkbox" "checked="checked" value="'.$j.'" /></td>'."\n".
                    833:                                   '<td valign="top"><b>'.$qtype[$i].'</b></td>'."\n".
                    834:                                   '<td valign="top"><input type="textbox" name="probfile_'.$j.'" value="'.$libfile.'" size="20" /></td>'."\n".
                    835:                                   '<td valign="top">'.$items->[$j].'</td>'."\n".
                    836:                                   '<td valign="top">'.$showfoil.'</td>'."\n".
                    837:                                   &Apache::loncommon::end_data_table_row());
1.1       raeburn   838:                     }
                    839:                     last;
                    840:                 }
                    841:             }
                    842:         }
                    843:     }
1.15      raeburn   844:     $r->print(&Apache::loncommon::end_data_table().'</p><p>'."\n".
                    845:               '<input type="hidden" name="qnumformat" value="'.$qnumformat.'" />'."\n".
                    846:               '<input type="hidden" name="blocks" value="'.$blocks.'" />');
1.1       raeburn   847:     for (my $i=0; $i<$blocks; $i++) {
1.15      raeburn   848:         $r->print('
                    849:           <input type="hidden" name="start_'.$i.'" value="'.$start[$i].'" />
                    850:           <input type="hidden" name="end_'.$i.'" value="'.$end[$i].'" />
                    851:           <input type="hidden" name="qtype_'.$i.'" value="'.$qtype[$i].'" />');
1.1       raeburn   852:         if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "Ord")) {
1.15      raeburn   853:             $r->print('
                    854:           <input type="hidden" name="foilformat_'.$i.'" value="'.$foilformats[$i].'" />');
1.1       raeburn   855:         }
                    856:         if (($qtype[$i] eq "MA") || ($qtype[$i] eq "FIB") || ($qtype[$i] eq "TF") || ($qtype[$i] eq "Ord")) {
1.15      raeburn   857:             $r->print('
                    858:           <input type="hidden" name="ansr_'.$i.'" value="'.$ansrtypes[$i].'" />');
                    859:         }
                    860:     }
                    861:     $r->print('</p>'.&page_footer($env{'form.newdir'},$uname,$fn,$page).'
                    862:               </form>');
1.1       raeburn   863: }
                    864: 
                    865: # ---------------------------------------------------------------- Final Display
                    866: sub final_display {
1.15      raeburn   867:     my ($r,$uname,$fn,$page,$textref,$res,$header,$css,$js,$webpath,$dirpath,$subdir) = @_;
1.6       albertel  868:     my $qnumformat = $env{'form.qnumformat'};
                    869:     my $blocks = $env{'form.blocks'};
1.1       raeburn   870:     my $question_id = '';
                    871:     my @question_title = ();
                    872:     my @question_status  = ();
                    873:     my @qtype = ();
                    874:     my @start = ();
                    875:     my @nums = ();
                    876:     my @end = ();
                    877:     my @foilformats = ();
                    878:     my @ansrtypes = ();
                    879:     my %multparts = ();
                    880:     my $numitems = 0;
1.15      raeburn   881:     my @createprobs = &Apache::loncommon::get_env_multiple('form.createprob');
1.1       raeburn   882:     for (my $i=0; $i<$blocks; $i++) {
1.6       albertel  883:         $start[$i] = $env{"form.start_$i"};
                    884:         $end[$i] = $env{"form.end_$i"};
1.1       raeburn   885:         if (($end[$i] - $start[$i]) >= 0) {
                    886:             $nums[$i] = $end[$i] - $start[$i]+1;
                    887:         } else {
                    888:             $nums[$i] = 0;
                    889:         }
1.6       albertel  890:         $qtype[$i] = $env{"form.qtype_$i"};
1.1       raeburn   891:         if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "Ord")) {
1.6       albertel  892:             $foilformats[$i] = $env{"form.foilformat_$i"};
1.1       raeburn   893:         } else {
                    894:             $foilformats[$i] = '';
                    895:         }
                    896:         if (($qtype[$i] eq "MA") || ($qtype[$i] eq "FIB") || ($qtype[$i] eq "TF") || ($qtype[$i] eq "Ord")) {
1.6       albertel  897:             $ansrtypes[$i] = $env{"form.ansr_$i"};
1.1       raeburn   898:         }
                    899:         $numitems += $nums[$i];
                    900:     }
                    901: 
1.15      raeburn   902:     my %answers;
                    903:     my ($items,$ids,$footer) = &file_split(\@start,\@end,\@nums,$qnumformat,\@foilformats,$textref,\%multparts,$numitems,\@qtype,$blocks);
1.1       raeburn   904: 
                    905: # Converting MC and MA answer to number, and splitting answers for FIB, and ordering for Ord.
1.15      raeburn   906:     my ($alphabet,$romans) = &get_constants();
1.1       raeburn   907:     my %patterns = (
                    908:          comma => ',',
                    909:          space => '\s+',
                    910:          line => '[\r\n\f]+',
                    911:          tab => '\t+',
                    912:        );
                    913:     for (my $i=0; $i<$blocks; $i++) {
                    914:         if ($nums[$i] > 0) {
                    915:             if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "FIB") || ($qtype[$i] eq "Ord")) {
                    916:                 for (my $k=$numitems+$start[$i]-1; $k<$numitems+$end[$i]; $k++) {
1.15      raeburn   917:                     my $qnum = $k - $numitems;
                    918:                     next if (!grep(/^$qnum$/,@createprobs));
                    919:                     if (($res eq 'application/rtf') || ($res eq 'text/html')) {
                    920:                         $items->[$k] = &HTML::Entities::decode($items->[$k]);
                    921:                     }
                    922:                     @{$answers{$qnum}} = ();
1.1       raeburn   923:                     if ($qtype[$i] eq "MC") {
1.15      raeburn   924:                         $items->[$k] =~ tr/A-Z/a-z/;
                    925:                         $items->[$k] =~ s/<\/?[^>]+>//g;
                    926:                         $items->[$k] =~ s/\W//g;
1.5       raeburn   927:                         if ($foilformats[$i] eq "lcperiod" || $foilformats[$i] eq "lcparen" || $foilformats[$i] eq "lconeparen" || $foilformats[$i] eq "lcdotparen" || $foilformats[$i] eq "ucparen" || $foilformats[$i] eq "ucperiod" || $foilformats[$i] eq "uconeparen" || $foilformats[$i] eq "ucdotparen") {
1.15      raeburn   928:                             for (my $j=0; $j<@{$alphabet}; $j++) {
                    929:                                 if ($alphabet->[$j] eq $items->[$k]) {
                    930:                                     push @{$answers{$qnum}}, $j;
1.1       raeburn   931:                                     last;
                    932:                                 }
                    933:                             }
1.5       raeburn   934:                         } elsif (($foilformats[$i] eq "romparen") || ($foilformats[$i] eq "romperiod") || ($foilformats[$i] eq "romoneparen") || ($foilformats[$i] eq "romdotparen")) {
1.15      raeburn   935:                             for (my $j=0; $j<@{$romans}; $j++) {
                    936:                                 if ($romans->[$j] eq $items->[$k]) {
                    937:                                     push @{$answers{$qnum}}, $j;
1.1       raeburn   938:                                     last;
                    939:                                 }
                    940:                             }
                    941:                         }
                    942:                     } elsif (($qtype[$i] eq "MA") || ($qtype[$i] eq "Ord")) {
1.15      raeburn   943:                         $items->[$k] =~ tr/A-Z/a-z/;
                    944:                         $items->[$k] =~ s/<\/?[^>]+>//g;
                    945:                         my @corrects = split/$patterns{$ansrtypes[$i]}/,$items->[$k];
1.1       raeburn   946:                         foreach my $correct (@corrects) {
1.14      raeburn   947:                             my @tied;
                    948:                             if ($qtype[$i] eq "Ord") {
                    949:                                 if ($correct =~ /=/) {
                    950:                                     @tied = split(/=/,$correct);
                    951:                                     for (my $j=0; $j<@tied; $j++) {
                    952:                                         $tied[$j] =~ s/\W//g;
                    953:                                     }
                    954:                                 } else {
                    955:                                     $correct =~s/\W//g;
                    956:                                 }
                    957:                             } else {
                    958:                                 $correct =~s/\W//g;
                    959:                             }
1.1       raeburn   960:                             if ($foilformats[$i] eq "lcperiod" || $foilformats[$i] eq "lcparen" || $foilformats[$i] eq "ucparen" || $foilformats[$i] eq "ucperiod") {
1.15      raeburn   961:                                 if (($qtype[$i] eq "Ord") && (@tied > 0)) {
1.14      raeburn   962:                                     my @ties;
                    963:                                     foreach my $tie (@tied) {
1.15      raeburn   964:                                         for (my $j=0; $j<@{$alphabet}; $j++) {
                    965:                                             if ($alphabet->[$j] eq $tie) {
1.14      raeburn   966:                                                 push(@ties,$j);
                    967:                                                 last;
                    968:                                             }
                    969:                                         }
                    970:                                     }
                    971:                                     my $ans = join('=',@ties);
1.15      raeburn   972:                                     push(@{$answers{$qnum}},$ans);
1.14      raeburn   973:                                 } else {
1.15      raeburn   974:                                     for (my $j=0; $j<@{$alphabet}; $j++) {
                    975:                                         if ($alphabet->[$j] eq $correct) {
                    976:                                             push @{$answers{$qnum}}, $j;
1.14      raeburn   977:                                             last;
                    978:                                         }
1.1       raeburn   979:                                     }
                    980:                                 }
1.5       raeburn   981:                             } elsif (($foilformats[$i] eq "romparen") || ($foilformats[$i] eq "romperiod") || ($foilformats[$i] eq "romoneparen") || ($foilformats[$i] eq "romdotparen")) {
1.14      raeburn   982:                                 if (($qtype[$i] eq "Ord") && (@tied > 0)) {
                    983:                                     my @ties;
                    984:                                     foreach my $tie (@tied) {
1.15      raeburn   985:                                         for (my $j=0; $j<@{$romans}; $j++) {
                    986:                                             if ($romans->[$j] eq $tie) {
1.14      raeburn   987:                                                 push(@ties,$j);
                    988:                                                 last;
                    989:                                             }
                    990:                                         }
                    991:                                     }
1.15      raeburn   992:                                     push(@{$answers{$qnum}},join('=',@ties));
1.14      raeburn   993:                                 } else {
1.15      raeburn   994:                                     for (my $j=0; $j<@{$romans}; $j++) {
                    995:                                         if ($romans->[$j] eq $correct) {
                    996:                                             push @{$answers{$qnum}}, $j;
1.14      raeburn   997:                                             last;
                    998:                                         }
1.1       raeburn   999:                                     }
                   1000:                                 }
                   1001:                             }
                   1002:                         }
                   1003:                     } elsif ($qtype[$i] eq "FIB") {
1.15      raeburn  1004:                         $items->[$k] =~ s/<\/?[^>]+>//g;
                   1005:                         @{$answers{$qnum}} = split/$patterns{$ansrtypes[$i]}/,$items->[$k];
                   1006:                         for (my $j=0; $j<@{$answers{$qnum}}; $j++) {
                   1007:                             $answers{$qnum}[$j] =~ s/^\s+//;
                   1008:                             $answers{$qnum}[$j] =~ s/\s+$//;
                   1009:                             if ($j==0) {
                   1010:                                 $answers{$qnum}[$j] =~ s/^<[^>]+>//;
                   1011:                             } elsif ($j == @{$answers{$qnum}}-1) {
                   1012:                                 $answers{$qnum}[$j] =~ s/<\/[^>]+>$//;
                   1013:                             }
1.1       raeburn  1014:                         }
                   1015:                     }
                   1016:                 }
                   1017:             }
                   1018:         }
                   1019:     }
1.15      raeburn  1020:     my $state;
                   1021: 
                   1022:     $r->print('<form name="verify" method="post" action="/adm/testbank">'."\n".
                   1023:               '<input type="hidden" name="blocks" value="'.$blocks.'" />'."\n".
                   1024:               '<input type="hidden" name="qnumformat" value="'.$qnumformat.'" />'."\n");
                   1025:     for (my $i=0; $i<$blocks; $i++) {
                   1026:        $r->print('<input type="hidden" name="start_'.$i.'" value="'.$start[$i].'" />
                   1027:            <input type="hidden" name="end_'.$i.'" value="'.$end[$i].'" />
                   1028:            <input type="hidden" name="qtype_'.$i.'" value="'.$qtype[$i].'" />
                   1029:            <input type="hidden" name="foilformat_'.$i.'" value="'.$foilformats[$i].'" />
                   1030:            <input type="hidden" name="ansr_'.$i.'" value="'.$ansrtypes[$i].'" />'."\n");
                   1031:     }
                   1032:     for (my $i=0; $i<$numitems; $i++) {
                   1033:         $r->print('<input type="hidden" name="probfile_'.$i.'" value="'.$env{'form.probfile_'.$i}.'" />'."\n");
                   1034:     }
                   1035:     $r->print(&topic_bar(6,&mt('Result of conversion of testbank questions to LON-CAPA problems')));
                   1036:     my $destdir = $dirpath;
                   1037:     if ($destdir ne '' && $subdir ne '') {
                   1038:         $subdir .= '/';
                   1039:         $destdir .= $subdir; 
                   1040:     }
                   1041:     if (@createprobs == 0) {
                   1042:         $state = 'unchecked';
                   1043:         $r->print('<p>'.&mt('No questions were selected for conversion.').'</p>'.
                   1044:                   &page_footer($env{'form.newdir'},$uname,$fn,$page,$webpath,$subdir,$state).'</form>');
                   1045:     } elsif (($destdir ne '') && (-e $destdir)) {
                   1046:         my (@qn_file,@result,@numid);
1.1       raeburn  1047:         my $qcount = 0;
1.15      raeburn  1048:         my $itemcount = 0;
1.1       raeburn  1049:         for (my $i=0; $i<$blocks; $i++) {
                   1050:             if ($nums[$i] > 0) {
                   1051:                 if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "FIB") || ($qtype[$i] eq "Ord")) {
                   1052:                     for (my $j=$start[$i]-1; $j<$end[$i]; $j++) {
1.15      raeburn  1053:                         $numid[$qcount] = $ids->[$itemcount];
                   1054:                         $itemcount ++;
                   1055:                         next if (!grep(/^$qcount$/,@createprobs));
                   1056:                         my $libfile = &probfile_name($j);
1.1       raeburn  1057:                         my $answer = $j + $numitems;
1.15      raeburn  1058:                         my $numans = scalar(@{$answers{$qcount}});
1.1       raeburn  1059:                         my $foilcount = 0;
                   1060:                         if (($qtype[$i] eq "MC") || ($qtype[$i] eq "MA") || ($qtype[$i] eq "Ord")) { 
                   1061:                             $foilcount = @{$multparts{$j}};
                   1062:                             $foilcount --;
                   1063:                         }
1.15      raeburn  1064:                         ($result[$qcount],$qn_file[$qcount]) = &create_mcq($destdir,$subdir,\@{$multparts{$j}},\@{$answers{$qcount}},$qtype[$i],$libfile,$res,$header,$footer,$js,$css);
1.1       raeburn  1065:                         $qcount ++;
                   1066:                     }
                   1067:                 } elsif ($qtype[$i] eq "TF") {
                   1068:                     for (my $j=$start[$i]-1; $j<$end[$i]; $j++) {
1.15      raeburn  1069:                         $numid[$qcount] = $ids->[$itemcount];
                   1070:                         $itemcount ++;
                   1071:                         next if (!grep(/^$qcount$/,@createprobs));
                   1072:                         my $libfile = &probfile_name($j);
1.1       raeburn  1073:                         my $answer = $j + $numitems;
1.15      raeburn  1074:                         $items->[$answer] =~ s/^\s+//;
                   1075:                         $items->[$answer] =~ s/\s+$//;
                   1076:                         $items->[$answer] =~ s/\W//g;
                   1077:                         $items->[$answer] =~ tr/A-Z/a-z/;
1.1       raeburn  1078:                         my $answer_id = '';
                   1079:                         if ($ansrtypes[$i] eq 'word' ) {
1.15      raeburn  1080:                             if ($items->[$answer] =~ m/true/) {
1.1       raeburn  1081:                                 $answer_id = 0;
                   1082:                             } else {
                   1083:                                 $answer_id = 1;
                   1084:                             }
                   1085:                         } elsif ($ansrtypes[$i] eq 'lett') {
1.15      raeburn  1086:                             if ($items->[$answer] =~ m/^t/) {
1.1       raeburn  1087:                                 $answer_id = 0;
                   1088:                             } else {
                   1089:                                 $answer_id = 1;
                   1090:                             }
                   1091:                         }
1.15      raeburn  1092:                         ($result[$qcount],$qn_file[$qcount]) = &create_ess($destdir,$subdir,$answer_id,$items->[$j],$items->[$answer],$qtype[$i],$libfile,$res,$header,$footer,$js,$css);
1.1       raeburn  1093:                         $qcount ++;
                   1094:                     }
                   1095:                 } elsif ($qtype[$i] eq "Ess") {
                   1096:                     for (my $j=$start[$i]-1; $j<$end[$i]; $j++) {
1.15      raeburn  1097:                         $numid[$qcount] = $ids->[$itemcount];
                   1098:                         $itemcount ++;
                   1099:                         next if (!grep(/^$qcount$/,@createprobs));
                   1100:                         my $libfile = &probfile_name($j);
1.1       raeburn  1101:                         my $answer = $j + $numitems;
                   1102:                         my $answer_id = '';
1.15      raeburn  1103:                         ($result[$qcount],$qn_file[$qcount]) = &create_ess($destdir,$subdir,$answer_id,$items->[$j],$items->[$answer],$qtype[$i],$libfile,$res,$header,$footer,$js,$css);
1.1       raeburn  1104:                         $qcount ++;
                   1105:                     }
                   1106:                 }
                   1107:             }
                   1108:         }
1.15      raeburn  1109:         my ($successes,$failures,$existing);
1.1       raeburn  1110:         for (my $i=0; $i<@qn_file; $i++) {
1.15      raeburn  1111:             if ($result[$i] eq 'ok') {
                   1112:                 $successes .= '<b>'.$numid[$i].':&nbsp;<a href="'.$webpath.$qn_file[$i].'">'.
                   1113:                           $qn_file[$i].'</a></b><br />'."\n";
                   1114:             } elsif ($result[$i] eq 'failed') {
                   1115:                 $failures .= $numid[$i].':&nbsp;'.$qn_file[$i].'<br />'."\n";
                   1116:             } elsif ($result[$i] eq 'exists') {
                   1117:                 $existing .= '<b>'.$numid[$i].':&nbsp;<a href="'.$webpath.$qn_file[$i].'">'.
                   1118:                           $qn_file[$i].'</a></b><br />'."\n";
                   1119:             }
                   1120:         }
                   1121:         if ($successes) {
                   1122:             $r->print('<p>'.&mt('Individual problem files have been created from the following problems included in the testbank file:').'<br />'.$successes.'</p><p>'.
                   1123:                      &mt('The problems must be published before they can be used in a course').'</p>');
                   1124:         }
                   1125:         if ($failures) {
                   1126:             $r->print('<p>'.&mt('An error occurred when opening files for the following problems, so they have not been created:').'<br />'.$failures.'</p>');
                   1127:         }
                   1128:         if ($existing) {
                   1129:             $r->print('<p>'.&mt('The following files already existed, and were not overwritten so these problems generated from the testbank have not been saved:').'<br />'.$existing.'</p>');
                   1130:             $state = 'existing';
                   1131:         }
                   1132:         $r->print(&page_footer($env{'form.newdir'},$uname,$fn,$page,$webpath,$subdir,$state).'</form>');
1.1       raeburn  1133:     } else {
1.15      raeburn  1134:         $state = 'nodir';
                   1135:         $r->print('<p>'.&mt('No destination directory was available so import of questions could not proceed.').'</p>'.
                   1136:                   &page_footer($env{'form.newdir'},$uname,$fn,$page,$webpath,$subdir,$state).'</form>');
                   1137:     }
1.1       raeburn  1138:     return;
1.15      raeburn  1139: }
                   1140: 
                   1141: sub show_uploaded_data {
                   1142:     my ($textref,$header) = @_;
                   1143:     my $output = '<p><b>'.&mt('Testbank data uploaded to the server').'</b></p><p>'."\n".
1.16      raeburn  1144:                  '<textarea name="rawdata" cols="70" rows="6" wrap="virtual" align="center" readonly>'."\n";
1.15      raeburn  1145:     if ($header ne '') {
                   1146:         $output .= $header."\n";
                   1147:     }
                   1148:     if (ref($textref) eq 'ARRAY') {
                   1149:         foreach my $line (@{$textref}) {
                   1150:            $line =~ s/\n//g;
                   1151:            if ($line ne '') {
                   1152:                $output .= $line."\n";
                   1153:            }
                   1154:         }
                   1155:     }
                   1156:     $output .= '</textarea></p>';
                   1157:     return $output;
                   1158: }
                   1159: 
                   1160: sub page_footer {
                   1161:     my ($newdir,$uname,$fn,$page,$webpath,$subdir,$state) = @_;
                   1162:     my $prevval = &mt('Previous Page');
                   1163:     my $nextval = &mt('Next Page');
                   1164:     my $prevclick = 'javascript:backPage();';
                   1165:     my $nextclick = 'javascript:nextPage();';
1.17      raeburn  1166:     my $go = '';
                   1167:     if (($page == 0) || ($state eq 'badfile')) {
1.15      raeburn  1168:         $go = 'NextPage';
                   1169:         $prevval = &mt('Exit Now');
                   1170:         $prevclick = 'javascript:location.href='."'$webpath';";
                   1171:         $nextclick = 'javascript:submit();'
                   1172:     } elsif ($page == 3) {
                   1173:         $nextval = &mt('Complete Testbank Conversion');
                   1174:     } elsif ($page == 4) {
                   1175:         if (($state ne 'existing') && ($state ne 'unchecked')) {
                   1176:             my $destdir = $webpath;
                   1177:             if ($subdir ne '') {
                   1178:                 $destdir = $webpath.$subdir;
                   1179:             }
                   1180:             $prevval = &mt('Back to Directory');
                   1181:             $prevclick = 'javascript:location.href='."'$destdir';";
                   1182:        }
                   1183:     }
                   1184:     my $output = '
                   1185:        <input type="hidden" name="newdir" value="'.&HTML::Entities::encode($newdir,'<>&"').'" />
                   1186:        <input type="hidden" name="uploaduname" value="'.$uname.'" />
                   1187:        <input type="hidden" name="filename" value="'.$fn.'" />
                   1188:        <input type="hidden" name="page" value="'.$page.'" />
                   1189:        <input type="hidden" name="phase" value="three" />
1.18      raeburn  1190:        <input type="hidden" name="go" value="'.$go.'" />
                   1191:        <input type="hidden" name="timestamp" value="'.$env{'form.timestamp'}.'" />';
1.15      raeburn  1192:     if ($page ne '') {
                   1193:         $output .= '
                   1194:        <table border="0">
                   1195:         <tr>
                   1196:          <td>
                   1197:           <input type="button" name="backpage" value="'.$prevval.'" onclick="'.$prevclick.'" />
                   1198:          </td>';
1.17      raeburn  1199:         if (($page < 4) && ($state ne 'badfile'))  {
1.15      raeburn  1200:             $output .= '
                   1201:          <td>&nbsp;</td>
                   1202:          <td>
                   1203:           <input type="button" name="nextpage" value="'.$nextval.'" onclick="'.$nextclick.'">
                   1204:          </td>';
                   1205:         }
                   1206:         $output .= '    </tr>
                   1207:        </table>
                   1208: ';
                   1209:     }
                   1210:     return $output;
1.1       raeburn  1211: }
                   1212: 
                   1213: sub question_count {
                   1214:     my ($qnumformat,$textref) = @_;
                   1215:     my $text_in = join "\n", @{$textref};
                   1216:     $text_in = "\n ".$text_in;
                   1217:     my $qpattern ='';
                   1218:     if ($qnumformat eq "period") {
                   1219:         $qpattern = '\d{1,}\.';
                   1220:     } elsif ($qnumformat eq "paren") {
                   1221:         $qpattern = '\(\d{1,}\)';
                   1222:     } elsif ($qnumformat eq "number") {
                   1223:         $qpattern = '\d{1,}';
                   1224:     } elsif ($qnumformat eq "leadparen") {
                   1225:         $qpattern = '\(\d{1,}';
                   1226:     } elsif ($qnumformat eq "trailparen") {
                   1227:         $qpattern = '\d{1,}\)';
                   1228:     }
                   1229:     my @questions = split/[\r\n\f]+\s?$qpattern\s?/,$text_in;
                   1230:     my $qcount = scalar(@questions);
                   1231:     $qcount = $qcount/2;
                   1232:     $qcount = int($qcount);
                   1233:     return $qcount;
                   1234: }
                   1235: 
1.15      raeburn  1236: sub get_constants {
                   1237:     my @alphabet = ("a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p","q","r","s","t","u","v","w","x","y","z");
                   1238:     my @romans = ("i","ii","iii","iv","v","vi","vii","viii","ix","x","xi","xii","xiii","xiv","xv","xvi","xvii","xviii","xix","xx","xxi","xxii","xxiii","xxiv","xxv","xxvi");
                   1239:     return (\@alphabet,\@romans);
                   1240: }
                   1241: 
1.1       raeburn  1242: sub file_split {
                   1243:     my ($startsref,$endsref,$numsref,$qnumformat,$foilsref,$textref,$multpartsref,$numitems,$qtyperef,$blocks) = @_;
                   1244:     my $text_in = join "\n", @{$textref};
                   1245:     $text_in = "\n ".$text_in;
                   1246:     my $dignum = length($numitems);
1.15      raeburn  1247:     my ($qpatst,$qpatend,$numpat,@questions,@qids);
                   1248:     my $numpat = '\d{1';
1.1       raeburn  1249:     if ($dignum > 1) {
1.15      raeburn  1250:         $numpat .= ','.$dignum.'}';
1.1       raeburn  1251:     } else {
1.15      raeburn  1252:         $numpat .= '}';
1.1       raeburn  1253:     }
                   1254:     if ($qnumformat eq "period") {
1.15      raeburn  1255:         $qpatend = '\.'; 
1.1       raeburn  1256:     } elsif ($qnumformat eq "paren") {
1.15      raeburn  1257:         $qpatst = '\(';
                   1258:         $qpatend = '\)';
1.1       raeburn  1259:     } elsif ($qnumformat eq "leadparen") {
1.15      raeburn  1260:         $qpatst = '\(';
1.1       raeburn  1261:     } elsif ($qnumformat eq "trailparen") {
1.15      raeburn  1262:         $qpatend = '\)';
1.1       raeburn  1263:     }
1.15      raeburn  1264:     my @lines = split/[\r\n\f]+\s*$qpatst($numpat)$qpatend\s*/,$text_in;
1.1       raeburn  1265: # my @questions = split/\n\s\d{1,3}\.\s/,$text_in;
1.15      raeburn  1266:     shift(@lines);
                   1267:     for (my $i=0; $i<@lines; $i++) {
                   1268:         if ($i%2) {
                   1269:             push(@questions,$lines[$i]);
                   1270:         } else {
                   1271:             push(@qids,$lines[$i]);
                   1272:         }
                   1273:     }
1.1       raeburn  1274:     my %multparts = ();
                   1275:     for (my $i=0; $i<$blocks; $i++) {
                   1276:         if (${$numsref}[$i] > 0) {
1.14      raeburn  1277:             if ((${$qtyperef}[$i] eq "MC") || (${$qtyperef}[$i] eq "MA") || (${$qtyperef}[$i] eq "Ord")) {
1.1       raeburn  1278:                 my $splitstr = '';
                   1279:                 if (${$foilsref}[$i] eq "lcperiod") {
                   1280:                     $splitstr = '[a-z]\.';
                   1281:                 } elsif (${$foilsref}[$i] eq "lcparen") {
                   1282:                     $splitstr = '\([a-z]\)';
1.5       raeburn  1283:                 } elsif (${$foilsref}[$i] eq "lconeparen") {
                   1284:                     $splitstr = '[a-z]\)';
                   1285:                 } elsif (${$foilsref}[$i] eq "lcdotparen") {
                   1286:                     $splitstr = '[a-z]\.\)';
1.1       raeburn  1287:                 } elsif (${$foilsref}[$i] eq "ucperiod") {
                   1288:                     $splitstr = '[A-Z]\.';
                   1289:                 } elsif (${$foilsref}[$i] eq "ucparen") {
                   1290:                     $splitstr = '\([A-Z]\)';
1.5       raeburn  1291:                 } elsif (${$foilsref}[$i] eq "uconeparen") {
                   1292:                     $splitstr = '[A-Z]\)';
                   1293:                 } elsif (${$foilsref}[$i] eq "ucdotparen") {
                   1294:                     $splitstr = '[A-Z]\.\)';
1.1       raeburn  1295:                 } elsif (${$foilsref}[$i] eq "romperiod") {
                   1296:                     $splitstr = '[ivx]+\.';
                   1297:                 } elsif (${$foilsref}[$i] eq "romparen") {
                   1298:                     $splitstr = '\([ivx]+\)';
1.5       raeburn  1299:                 } elsif (${$foilsref}[$i] eq "romoneparen") {
                   1300:                     $splitstr = '[ivx]+\)';
                   1301:                 } elsif (${$foilsref}[$i] eq "romdotparen") {
                   1302:                     $splitstr = '[ivx]+\.\)';
1.1       raeburn  1303:                 }
                   1304:                 for (my $j=${$startsref}[$i]-1; $j<${$endsref}[$i]; $j++) {
1.5       raeburn  1305:                     @{$multparts{$j}} = split/[\r\n\f]+\s*$splitstr\s*/,$questions[$j];
1.1       raeburn  1306:                     chomp(@{$multparts{$j}});
                   1307:                 }
                   1308:             } elsif (${$qtyperef}[$i] eq "FIB") { 
                   1309:                 for (my $j=${$startsref}[$i]-1; $j<${$endsref}[$i]; $j++) {
                   1310:                     @{$multparts{$j}} = ("$questions[$j]");
                   1311:                 }
                   1312:             }
                   1313:         }
1.15      raeburn  1314:     }
                   1315:     my ($lastanswer,$footer) = ($questions[-1] =~ /^([,\r\n\f\t\s().A-Za-z]+)(.+)$/);
                   1316:     if ($footer ne '') {
                   1317:         $questions[-1] = $lastanswer;
                   1318:     }
1.1       raeburn  1319:     %{$multpartsref} = %multparts;
1.15      raeburn  1320:     return (\@questions,\@qids,$footer);
1.1       raeburn  1321: }
                   1322:  
                   1323: # create_mcq builds an MC, MA, Ord or FIB question
                   1324: 
                   1325: sub create_mcq {
1.15      raeburn  1326:     my ($destdir,$subdir,$qstnref,$answerref,$qtype,$libfile,$res,$header,$footer,$js,$css) = @_;
                   1327: 
1.1       raeburn  1328:     my $qstn = ${$qstnref}[0];
                   1329:     my $numfoils = scalar(@{$qstnref}) - 1; 
                   1330:     my $datestamp = localtime;
                   1331:     my $numansrs = scalar(@{$answerref});
1.15      raeburn  1332:     my $output = '<problem>
                   1333:  <startouttext />';
                   1334:     if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1335:         if ($header ne '') {
                   1336:             $output .= &HTML::Entities::decode($header);
                   1337:         }
                   1338:         if ($js ne '') {
                   1339:             $output .= &HTML::Entities::decode($js);
                   1340:         }
                   1341:         if ($css ne '') {
                   1342:             $output .= &HTML::Entities::decode($css);
                   1343:         }
                   1344:         $qstn = &HTML::Entities::decode($qstn);
                   1345:     }
                   1346:     $output .= $qstn.'<endouttext />'."\n";
1.1       raeburn  1347:     if ($qtype eq "MA") {
                   1348:         $output .= qq|
                   1349:    <optionresponse max="$numfoils" randomize="yes">
                   1350:     <foilgroup options="('True','False')">
                   1351:         |;
                   1352:         for (my $k=0; $k<@{$qstnref}-1; $k++) {
                   1353:             $output .= "   <foil name=\"foil".$k."\" value=\"";
                   1354:             if (grep/^$k$/,@{$answerref}) {
                   1355:                 $output .= "True\" location=\"random\"";
                   1356:             } else {
                   1357:                 $output .= "False\" location=\"random\"";
                   1358:             }
1.15      raeburn  1359:             my $showfoil = ${$qstnref}[$k+1];
                   1360:             if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1361:                 $showfoil = &HTML::Entities::decode($showfoil);
                   1362:             }
                   1363:             $output .= "\><startouttext />$showfoil<endouttext /></foil>\n";
1.1       raeburn  1364:         }
                   1365:         chomp($output);
                   1366:         $output .= qq|
                   1367:     </foilgroup>
1.15      raeburn  1368:    </optionresponse>|;
1.1       raeburn  1369:     }
                   1370:     if ($qtype eq "MC") {
                   1371:         $output .= qq|
                   1372:    <radiobuttonresponse max="$numfoils" randomize="yes">
                   1373:     <foilgroup>
                   1374:         |;
                   1375:         for (my $k=0; $k<@{$qstnref}-1; $k++) {
                   1376:             $output .= "   <foil name=\"foil".$k."\" value=\"";
                   1377:             if (grep/^$k$/,@{$answerref}) {
                   1378:                 $output .= "true\" location=\"";
                   1379:             } else {
                   1380:                 $output .= "false\" location=\"";
                   1381:             }
                   1382:             if (lc (${$qstnref}[$k+1]) =~ m/^\s?([Aa]ll)|([Nn]one)\sof\sthe\sabove\.?/) { 
                   1383:                 $output .= "bottom\"";
                   1384:             } else {
                   1385:                 $output .= "random\"";
                   1386:             }
1.15      raeburn  1387:             my $showfoil = ${$qstnref}[$k+1];
                   1388:             if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1389:                 $showfoil = &HTML::Entities::decode($showfoil);
                   1390:             }
                   1391:             $output .= "\><startouttext />$showfoil<endouttext /></foil>\n";
1.1       raeburn  1392:         }
                   1393:         chomp($output);
                   1394:         $output .= qq|
                   1395:     </foilgroup>
1.15      raeburn  1396:    </radiobuttonresponse>|;
1.1       raeburn  1397:     }
                   1398:     if ($qtype eq "Ord") {
                   1399:         $output .= qq|
                   1400:    <rankresponse max="$numfoils" randomize="yes">
                   1401:     <foilgroup>
                   1402:         |;
                   1403:         for (my $k=0; $k<@{$qstnref}-1; $k++) {
1.14      raeburn  1404:             my $ansval;
                   1405:             my $num = 0;
                   1406:             for (my $i=0; $i<@{$answerref}; $i++) {
                   1407:                 if ($$answerref[$i] =~ /=/) {
                   1408:                     my @tied = split(/=/,$$answerref[$i]);
                   1409:                     foreach my $tie (@tied) {
                   1410:                         if ($k == $tie) {
                   1411:                             $ansval = $num + 1;
                   1412:                             last;
                   1413:                         }
                   1414:                     }
                   1415:                     $num += scalar(@tied);
                   1416:                 } elsif ($k == $$answerref[$i]) {
                   1417:                     $ansval = $num + 1;
                   1418:                     last;
                   1419:                 } else {
                   1420:                     $num ++;
                   1421:                 }
                   1422:             }
1.15      raeburn  1423:             my $showfoil = ${$qstnref}[$k+1];
                   1424:             if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1425:                 $showfoil = &HTML::Entities::decode($showfoil);
                   1426:             }
                   1427:             $output .= "   <foil location=\"random\" name=\"foil".$k."\" value=\"".$ansval."\"><startouttext />$showfoil<endouttext /></foil>\n";
1.1       raeburn  1428:         }
                   1429:         chomp($output);
                   1430:         $output .= qq|
                   1431:     </foilgroup>
1.15      raeburn  1432:    </rankresponse>|;
1.1       raeburn  1433:     }
                   1434:     if ($qtype eq "FIB") {
                   1435:         my $numerical = 1;
                   1436:         for (my $i=0; $i<@{$answerref}; $i++) {
                   1437:             if (${$answerref}[$i] =~ m/([^\d\.]|\.\.)/) {
                   1438:                 $numerical = 0;
                   1439:             }
                   1440:         }
                   1441:         if ($numerical) {
                   1442:             my $numans;
                   1443:             my $tol;
                   1444:             if (@{$answerref} == 1) {
                   1445:                 $tol = 5;
                   1446:                 $numans = $$answerref[0];
                   1447:             } else {
1.2       raeburn  1448:                 my $min = $$answerref[0];
                   1449:                 my $max = $$answerref[0];    
                   1450:                 for (my $i=1; $i<@{$answerref}; $i++) {
                   1451:                     if ($$answerref[$i]<=$min) {
1.1       raeburn  1452:                         $min = $$answerref[$i];
1.2       raeburn  1453:                     } elsif ($$answerref[$i] >= $max) {
1.1       raeburn  1454:                         $max = $$answerref[$i];
                   1455:                     }
                   1456:                 }
                   1457:                 $numans = ($max + $min)/2;
                   1458:                 $tol = 100*($max - $min)/($numans*2); 
                   1459:             }
                   1460:             $output .= qq|
                   1461: <numericalresponse answer="$numans">
                   1462: 	<responseparam type="tolerance" default="$tol%" name="tol" description="Numerical Tolerance" />
                   1463: 	<responseparam name="sig" type="int_range,0-16" default="0,15" description="Significant Figures" />
                   1464: 	<textline />
1.15      raeburn  1465: </numericalresponse>|;
1.1       raeburn  1466:         } else {
                   1467:             if (@{$answerref} == 1) {
                   1468:                 $output .= qq|
                   1469: <stringresponse answer="$$answerref[0]" type="ci">
                   1470: <textline>
                   1471: </textline>
1.15      raeburn  1472: </stringresponse>|;
1.1       raeburn  1473:             } else {
                   1474:                 for (my $i=0; $i<@{$answerref}; $i++) {
                   1475:                     ${$answerref}[$i] =~ s/\|/\|/g;
                   1476:                 }
                   1477:                 my $regexpans = join('|',@{$answerref});
                   1478:                 $regexpans = '/('.$regexpans.')/'; 
                   1479:                 $output .= qq|
                   1480: <stringresponse answer="$regexpans" type="re">
                   1481: <textline>
                   1482: </textline>
1.15      raeburn  1483: </stringresponse>|;
1.1       raeburn  1484:             }
                   1485:         }
                   1486:     }
1.15      raeburn  1487:     if ($footer ne '') {
                   1488:         $output .= '<startouttext />'.&HTML::Entities::decode($footer).'<endouttext />';
                   1489:     }
                   1490:     $output .= qq|
                   1491:   </problem>
                   1492: |;
                   1493:     my $result;
                   1494:     if (-e $destdir.$libfile) {
                   1495:         $result = 'exists';
                   1496:     } else {
                   1497:         if (open(PROB,">$destdir$libfile")) {
                   1498:             print PROB $output;
                   1499:             close(PROB);
                   1500:             $result = 'ok';
                   1501:         } else {
                   1502:             $result = 'failed';
                   1503:         } 
                   1504:     }
                   1505:     return ($result,$subdir.$libfile);
1.1       raeburn  1506: }
                   1507: 
                   1508: # create_ess builds an essay or True/False question
                   1509: 
                   1510: sub create_ess {
1.15      raeburn  1511:     my ($destdir,$subdir,$answer_id,$qstn,$answertxt,$qtype,$libfile,$res,$header,
                   1512:         $footer,$js,$css) = @_;
                   1513:     my $output = '<problem>
                   1514:  <startouttext />';
                   1515:     if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1516:         if ($header ne '') {
                   1517:             $output .= &HTML::Entities::decode($header);
                   1518:         }
                   1519:         if ($js ne '') {
                   1520:             $output .= &HTML::Entities::decode($js);
                   1521:         }
                   1522:         if ($css ne '') {
                   1523:             $output .= &HTML::Entities::decode($css);
                   1524:         }
                   1525:         $qstn = &HTML::Entities::decode($qstn);
                   1526:         $answertxt = &HTML::Entities::decode($answertxt);
                   1527:     }
                   1528:     $output .= $qstn.'<endouttext />';
1.1       raeburn  1529:     my $answer = '';
                   1530:     my $answerlog = '';
                   1531:     if ($qtype eq "Ess") {
1.15      raeburn  1532:         $output .= '
1.1       raeburn  1533:    <essayresponse>
                   1534:    <textfield></textfield>
                   1535:    </essayresponse>
                   1536:    <postanswerdate>
1.13      raeburn  1537:     <startouttext />
1.15      raeburn  1538:    '.$answertxt
                   1539:    .'<endouttext />
                   1540:    </postanswerdate>';
1.1       raeburn  1541:     } elsif ($qtype eq "TF") {
                   1542:          $answer = $answer_id;
                   1543:          $output .= qq|
                   1544:    <radiobuttonresponse max="2" randomize="yes">
                   1545:     <foilgroup>
                   1546:          |;
                   1547:          $output .= "   <foil name=\"foil0\" value=\"true\" location=\"random\"><startouttext />";
                   1548:          if ($answer_id) {
                   1549:               $output .= "False";
                   1550:          } else {
                   1551:               $output .= "True";
                   1552:          }
                   1553:          $output .= "<endouttext /></foil>\n";
                   1554:          $output .= "   <foil name=\"foil1\" value=\"false\" location=\"random\"><startouttext />";
                   1555:          if ($answer_id) {
                   1556:               $output .= "True";
                   1557:          } else {
                   1558:               $output .= "False";
                   1559:          }
1.15      raeburn  1560:          $output .= '<endouttext /></foil>
1.1       raeburn  1561:     </foilgroup>
1.15      raeburn  1562:    </radiobuttonresponse>';
1.1       raeburn  1563:      }
1.15      raeburn  1564:      if ($footer ne '') {
                   1565:         $output .= '
                   1566: <startouttext />'.&HTML::Entities::decode($footer).'<endouttext />';
                   1567:      }
                   1568:      $output .= '
                   1569:   </problem>
                   1570: ';
                   1571:      my $result;
                   1572:      if (-e $destdir.$libfile) {
                   1573:          $result = 'exists';
                   1574:      } else {
                   1575:          if (open(PROB,">$destdir$libfile")) {
                   1576:              print PROB $output;
                   1577:              close(PROB);
                   1578:          } else {
                   1579:              $result = 'failed';
                   1580:          }
                   1581:      }
                   1582:      return ($result,$subdir.$libfile);
                   1583: }
                   1584: 
                   1585: sub probfile_name {
                   1586:     my ($j) = @_;
                   1587:     my $libfile = &HTML::Entities::decode($env{'form.probfile_'.$j});
                   1588:     my $qnum = $j + 1;
                   1589:     if ($libfile eq '') {
                   1590:         if (length($qnum) == 1) {
                   1591:             $qnum = "00".$qnum;
                   1592:         } elsif (length($qnum) == 2) {
                   1593:             $qnum = "0".$qnum;
                   1594:         }
                   1595:         $libfile = 'testbank_question_'.$qnum;
                   1596:         $libfile .= '.problem';
                   1597:     }
                   1598:     return $libfile;
1.1       raeburn  1599: }
                   1600: 
                   1601: sub file_error {
1.17      raeburn  1602:     my ($r,$uname,$fn,$current_page,$webpath,$res) = @_;
                   1603:     $r->print('<p><form name="display" method="post" action="/adm/testbank">'.&mt('The file you uploaded does not appear to be in the correct format.').
                   1604:               '</p><p>'.&mt('Extraction of questions is only possible for the following file types:').
                   1605:               '<ul><li>'.&mt('plain text').'</li><li>RTF</li><li>HTML</li></ul>'.
                   1606:               &mt('The file type identified for the file you uploaded is [_1].','<b>'.$res.'</b>').'</p>');
                   1607:     $r->print(&page_footer($env{'form.newdir'},$uname,$fn,$current_page,$webpath,undef,'badfile').
                   1608:              '</form>');
                   1609:     return;
1.15      raeburn  1610: }
                   1611: 
                   1612: sub parse_datafile {
1.18      raeburn  1613:     my ($r,$uname,$filename,$pathname,$dirpath,$urlpath,$page_name,$subdir,$timestamp) = @_;
1.15      raeburn  1614:     my ($badfile,$res,%allfiles,%codebase);
                   1615:     my $mm = new File::MMagic;
                   1616:     my ($text,$header,$css,$js);
                   1617:     if (-e "$dirpath") {
                   1618:         $res = $mm->checktype_filename($dirpath.$filename);
                   1619:         if ($env{'form.phase'} eq 'three') {          
                   1620:             if ($res eq 'text/plain') {
                   1621:                 open(TESTBANK,"<$dirpath$filename");
                   1622:                 @{$text} = <TESTBANK>;
                   1623:                 close(TESTBANK);
                   1624:             } elsif ($res eq 'application/rtf') {
                   1625:                 my $html = '';
1.18      raeburn  1626:                 my $image_uri = $timestamp;
1.15      raeburn  1627:                 if ($page_name eq 'Target') {
1.18      raeburn  1628:                     $image_uri = $urlpath.'/'.$timestamp;
1.15      raeburn  1629:                 }
                   1630:                 my $image_dir;
                   1631:                 if ($page_name eq 'Blocks') {
                   1632:                     $image_dir = $dirpath;
                   1633:                     $image_dir =~ s/\/$//;
1.18      raeburn  1634:                     $image_dir .= '/'.$timestamp;
                   1635:                     if (!-e $image_dir) {
                   1636:                         mkdir($image_dir,0755);
                   1637:                     }
1.15      raeburn  1638:                 } else {
                   1639:                     $image_dir = $r->dir_config('lonDaemons').'/tmp/'.
                   1640:                                  $env{'user.name'}.'_'.$env{'user.domain'}.
                   1641:                                  '_rtfupload_'.$filename.'_'.time.'_'.$$;
                   1642:                    if (!-e $image_dir) {
                   1643:                        mkdir($image_dir,0755);
                   1644:                    }
                   1645:                 }
                   1646:                 my $parser = RTF::HTMLConverter->new (
                   1647:                                   in                => $dirpath.$filename,
                   1648:                                   out               => \$html,
                   1649:                                   DOMImplementation => 'XML::DOM',
                   1650:                                   image_uri         => $image_uri,
                   1651:                                   image_dir         => $image_dir,
                   1652:                              );
                   1653:                 $parser->parse();
                   1654:                 utf8::decode($html);
                   1655:                 ($text,$header,$css,$js) = 
1.18      raeburn  1656:                     &parse_htmlcontent($res,$subdir,$html,undef,$page_name);
1.15      raeburn  1657:             } elsif ($res eq 'text/html') {
                   1658:                 ($text,$header,$css,$js) = 
1.18      raeburn  1659:                     &parse_htmlcontent($res,$subdir,undef,$dirpath.$filename,$page_name);
1.15      raeburn  1660:             } else {
                   1661:                 $badfile = 1;
                   1662:             }
                   1663:         }
                   1664:     }
                   1665:     return ($res,$badfile,$text,$header,$css,$js,\%allfiles,\%codebase);
                   1666: }
                   1667: 
                   1668: sub parse_htmlcontent {
1.18      raeburn  1669:     my ($res,$subdir,$html,$fullpath,$page_name) = @_;
1.15      raeburn  1670:     my ($p,$fh);
                   1671:     if ($res eq 'application/rtf') {
                   1672:         $p = HTML::TokeParser->new( \$html );
                   1673:     } elsif ($res eq 'text/html') {
                   1674:         open($fh, "<:utf8", $fullpath);
                   1675:         $p = HTML::TokeParser->new( $fh );
                   1676:     }
                   1677:     my ($current_tag,$line,@text,$header,$css,$js,$have_header,$delayed);
                   1678:     while (my $token = $p->get_token) {
                   1679:         if (ref($token) eq 'ARRAY') {
                   1680:             if ($token->[0] eq 'S') {
                   1681:                 if ($delayed ne '') {
                   1682:                     $line.= $delayed;
                   1683:                     $delayed = '';
                   1684:                 }
                   1685:                 $current_tag = $token->[1];
                   1686:                 next if ($token->[1] eq 'html' || $token->[1] eq 'head' || $token->[1] eq 'body' || $token->[1] eq 'meta' || $token->[1] eq 'title');
                   1687:                 if ($token->[1] eq 'p') {
                   1688:                     $line =~ s/^[\s\240]*(.*?)[\s\240]*$/$1/;
                   1689:                     if (!$have_header) {
                   1690:                         $header = $line;
                   1691:                         if ($header ne '') {
                   1692:                             $header =~ s/\s*[\n\r\f]+/\n/gs;
                   1693:                         }
                   1694:                         $have_header = 1;
                   1695:                     } else {
                   1696:                         push(@text,$line);
                   1697:                     }
                   1698:                     $line = '';
                   1699:                 } elsif ($current_tag eq 'style') {
                   1700:                     $css .= $token->[4];
                   1701:                 } elsif ($current_tag eq 'script') {
                   1702:                     $js .= $token->[4];
                   1703:                 } else {
                   1704:                     my $contents = $token->[4];
                   1705:                     if ($subdir ne '') {
                   1706:                         if (($token->[1] eq 'img') && ($token->[2]->{'src'} ne '')) {
1.18      raeburn  1707:                             if (($res eq 'text/html') || 
                   1708:                                 ($res eq 'application/rtf') && ($page_name ne 'Target')) {
                   1709:                                 $contents =~ s/(src=\s*["']?)/$1..\//i;
                   1710:                             }
1.15      raeburn  1711:                         }
                   1712:                     }
                   1713:                     if (($line eq '') && ($current_tag eq 'font')) {
                   1714:                         $delayed = &HTML::Entities::encode($contents,'<>&"');
                   1715:                     } else {
                   1716:                         $line .= &HTML::Entities::encode($contents,'<>&"');
                   1717:                     }
                   1718:                 }
                   1719:             } elsif ($token->[0] eq 'T') {
                   1720:                 if ($current_tag ne 'html' && $current_tag ne 'head' && $current_tag ne 'body' && $current_tag ne 'meta' && $current_tag ne 'title') {
                   1721:                     if ($current_tag eq 'style') { 
                   1722:                        $css .=  $token->[1];
                   1723:                     } elsif ($current_tag eq 'script') {
                   1724:                        $js .=  $token->[1];
                   1725:                     } else {
                   1726:                         if ($delayed ne '') {
                   1727:                             my ($id,$rest) = ($token->[1] =~ /^(\s*\(*[A-Za-z0-9]+\)*\.*\s+)(.+)$/s);
                   1728:                             if ($id ne '') {
                   1729:                                 $line .= $id.$delayed.$rest;
                   1730:                             } else {
                   1731:                                 $line .= $token->[1].$delayed;
                   1732:                             }
                   1733:                             $delayed = '';
                   1734:                         } else {
                   1735:                             $line .= $token->[1];
                   1736:                         }
                   1737:                     }
                   1738:                 }
                   1739:             } elsif ($token->[0] eq 'E') {
                   1740:                 next if ($token->[1] eq 'html' || $token->[1] eq 'head' || $token->[1] eq 'body' || $token->[1] eq 'meta' || $token->[1] eq 'title' || $token->[1] eq 'p');
                   1741:                 if ($token->[1] eq 'style') {
                   1742:                     $css .= $token->[2];
                   1743:                 } elsif ($token->[1] eq 'script') {
                   1744:                     $js .= $token->[2];
                   1745:                 } else {
                   1746:                     $line .= &HTML::Entities::encode($token->[2],'<>&"');
                   1747:                 }
                   1748:                 $current_tag = '';
                   1749:             }
                   1750:         }
                   1751:     }
                   1752:     if ($line ne '') {
                   1753:         if ($line ne '') {
                   1754:             $line =~ s/\s*[\n\r\f]+/\n/gs;
                   1755:         }
                   1756:         $line =~ s/^[\s\240]*(.*?)[\s\240]*$/$1/;
                   1757:         push(@text,$line);
                   1758:     }
                   1759:     if ($res eq 'text/html') {
                   1760:         close($fh);
                   1761:     }
                   1762:     return (\@text,$header,$css,$js);
                   1763: }
                   1764: 
                   1765: sub build_image_url {
                   1766:     my ($urlpath,$item) = @_;
                   1767:     $item =~ s/(<img[^>]+src=["']?\s*)(\.?\.?\/?)/$1$urlpath/gsi;
                   1768:     return $item; 
                   1769: }
                   1770: 
                   1771: sub print_header {
                   1772:     my ($uname,$udom,$javascript,$loadentries,$title) = @_;
                   1773:     my $output = &Apache::loncommon::start_page($title,$javascript,
                   1774:                                              {'add_entries' => $loadentries});
                   1775:     if (($uname ne $env{'user.name'}) || ($udom ne $env{'user.domain'})) {
1.20    ! bisitz   1776:         $output .= '<p class="LC_warning">'
        !          1777:                  .&mt('Co-Author [_1]:[_2]',$uname,$udom)
        !          1778:                  .'</p>';
1.15      raeburn  1779:     }
                   1780:     return $output;
                   1781: }
                   1782: 
                   1783: sub topic_bar {
                   1784:     my ($imgnum,$title) = @_;
                   1785:     my $output = '
                   1786: <div class="LC_topic_bar">
                   1787:     <img alt="'.&mt('Step [_1]',$imgnum).
                   1788:               ' "src="/res/adm/pages/bl_step'.$imgnum.'.gif" />&nbsp;'.$title.'
                   1789: </div>
                   1790: ';
                   1791:     return $output;
                   1792: }
1.1       raeburn  1793: 
                   1794: # ---------------------------------------------------------------- Main Handler
                   1795: sub handler {
                   1796:     my $r=shift;
                   1797:     my $uname;
                   1798:     my $udom;
                   1799:     my $javascript = '';
                   1800:     my $page_name = '';
                   1801:     my $current_page = '';
                   1802:     my $qcount = '';
1.15      raeburn  1803:     my $title = 'Upload testbank questions to Construction Space';
                   1804: 
1.6       albertel 1805:     if ($env{'form.uploaduname'}) {
                   1806:         $env{'form.filename'}='/priv/'.$env{'form.uploaduname'}.'/'.
                   1807:             $env{'form.filename'};
1.1       raeburn  1808:     }
                   1809:     ($uname,$udom)=
1.6       albertel 1810:         &Apache::loncacc::constructaccess($env{'form.filename'},
1.1       raeburn  1811:                                           $r->dir_config('lonDefDomain'));
                   1812:     unless (($uname) && ($udom)) {
1.15      raeburn  1813:         $r->log_reason($uname.':'.$udom.' trying to convert testbank file '.
                   1814:                        $env{'form.filename'}.' - not authorized',$r->filename);
1.1       raeburn  1815:         return HTTP_NOT_ACCEPTABLE;
                   1816:     }
1.15      raeburn  1817: 
                   1818:     my ($fn,$filename);
1.6       albertel 1819:     if ($env{'form.filename'}) {
                   1820:         $fn=$env{'form.filename'};
1.19      raeburn  1821:         $fn=~s/^https?\:\/\/[^\/]+\///;
1.1       raeburn  1822:         $fn=~s/^\///;
1.11      albertel 1823:         $fn=~s{(~|priv/)($LONCAPA::username_re)}{};
1.1       raeburn  1824:         $fn=~s/\/+/\//g;
                   1825:     } else {
1.6       albertel 1826:         $r->log_reason($env{'user.name'}.' at '.$env{'user.domain'}.
1.1       raeburn  1827:                        ' unspecified filename for upload', $r->filename);
                   1828:         return HTTP_NOT_FOUND;
                   1829:     }
                   1830: 
                   1831: # ----------------------------------------------------------- Start page output
                   1832:     &Apache::loncommon::content_type($r,'text/html');
                   1833:     $r->send_http_header;
                   1834: 
1.15      raeburn  1835:     my ($filename,$pathname) = &File::Basename::fileparse($fn);
                   1836:     my $webpath = '/priv/'.$uname.$pathname;
                   1837:     my $urlpath = '/~'.$uname.$pathname;
                   1838:     my $dirpath = '/home/'.$uname.'/public_html'.$pathname;
                   1839:     my ($res,$subdir,$badfile,$textref,$header,$css,$js,%loadentries);
                   1840: 
1.6       albertel 1841:     if ($env{'form.phase'} eq 'three') {
1.1       raeburn  1842:         $current_page = &display_control();
1.15      raeburn  1843:         my @pages = ('Welcome','Blocks','Format','Target','Confirmation');
                   1844:         $page_name = $pages[$current_page];
1.18      raeburn  1845:         if ($env{'form.timestamp'} eq '') {
                   1846:             $env{'form.timestamp'} = time; 
                   1847:         }
1.15      raeburn  1848:         if ($env{'form.newdir'} ne '') {
                   1849:             if ($env{'form.newdir'} =~ /^\Q$dirpath\E(.+)$/) {
                   1850:                 $subdir = $1;
                   1851:             }
                   1852:         }
                   1853:         ($res,$badfile,$textref,$header,$css,$js) = 
                   1854:             &parse_datafile($r,$uname,$filename,$pathname,$dirpath,$urlpath,
1.18      raeburn  1855:                             $page_name,$subdir,$env{'form.timestamp'});
1.15      raeburn  1856:         if ($page_name eq 'Welcome') {
                   1857:              &jscript_zero($webpath,\$javascript);
                   1858:         } elsif ($page_name eq 'Blocks') {
                   1859:             if ($env{'form.go'} eq "PreviousPage") {
                   1860: 	        $loadentries{'onload'} = "setElements()";
                   1861:             }
1.1       raeburn  1862:             &jscript_one(\$javascript);
1.15      raeburn  1863:         } elsif ($page_name eq 'Format') {
                   1864:             if ($env{'form.go'} eq "PreviousPage") {
                   1865:                 $loadentries{'onload'} = "setElements()";
                   1866:             }
                   1867:             $qcount = question_count($env{'form.qnumformat'},$textref);
1.1       raeburn  1868:  	    &jscript_two(\$javascript,$qcount);
1.15      raeburn  1869:         } elsif ($page_name eq 'Target') {
1.6       albertel 1870:              if ($env{'form.go'} eq "PreviousPage") {
1.10      albertel 1871:                  $loadentries{'onload'} = "setElements()";
1.1       raeburn  1872:  	     }
1.15      raeburn  1873: 	     &jscript_three($webpath,\$javascript);
1.1       raeburn  1874:         } elsif ($page_name eq 'Confirmation') {
1.15      raeburn  1875: 	     &jscript_four(\$javascript,$webpath);
                   1876:         }
                   1877:         $javascript = "<script type=\"text/javascript\">\n//<!--\n".
                   1878: 	              $javascript."\n// --></script>\n";
                   1879:         if ($res eq 'application/rtf' || $res eq 'text/html') {
                   1880:             if ($page_name eq 'Target') {
                   1881:                 $javascript .= $js.$css;
                   1882:             }
1.1       raeburn  1883:         }
1.8       albertel 1884:     }
                   1885: 
1.15      raeburn  1886:     $r->print(&print_header($uname,$udom,$javascript,\%loadentries,$title));
1.1       raeburn  1887: 
1.6       albertel 1888:     if ($env{'form.phase'} eq 'three') {
1.15      raeburn  1889:         if ($env{'form.action'} eq 'upload_embedded') {
                   1890:             $r->print(&Apache::lonupload::phasethree($r,$fn,$uname,$udom,'testbank'));
                   1891:         }
1.1       raeburn  1892:         if ($badfile) {
1.17      raeburn  1893:             &file_error($r,$uname,$fn,$current_page,$webpath,$res);
1.1       raeburn  1894:         } else {        
1.15      raeburn  1895:             &display_zero ($r,$uname,$fn,$current_page,$webpath) if $page_name eq 'Welcome';
                   1896:             &display_one ($r,$uname,$fn,$current_page,$textref,$header) if $page_name eq 'Blocks';
                   1897:             &display_two ($r,$uname,$fn,$current_page,$textref,$header,$qcount) if $page_name eq 'Format';
                   1898:             &display_three ($r,$uname,$fn,$current_page,$textref,$res,$header,$urlpath,$qcount) if $page_name eq 'Target';
                   1899:             &final_display ($r,$uname,$fn,$current_page,$textref,$res,$header,$css,$js,$webpath,$dirpath,$subdir) if $page_name eq 'Confirmation';
1.1       raeburn  1900:         }
1.6       albertel 1901:     } elsif ($env{'form.phase'} eq 'two') {
1.15      raeburn  1902:         my ($result,$flag) = &Apache::lonupload::phasetwo($r,$fn,$uname,$udom,'testbank');
                   1903:         $r->print($result);
1.1       raeburn  1904:         if ($flag eq 'ok') {
                   1905:             my $current_page = 0;
1.15      raeburn  1906:             my $js;
                   1907:             &jscript_zero($webpath,\$js);
                   1908:             $js = '<script type="text/javascript">'."\n$js\n".'</script>';
                   1909:             $r->print($js);
                   1910:             &display_zero($r,$uname,$fn,$current_page,$webpath);
                   1911:         } elsif ($flag eq 'embedded') {
                   1912:             $r->print($js.'<form name="testbankForm" method="post" action="/adm/testbank">'.
                   1913:                       &page_footer('',$uname,$fn).'</form>');
1.1       raeburn  1914:         }
                   1915:     } else {
                   1916:         &Apache::lonupload::phaseone($r,$fn,$uname,$udom,'testbank');
                   1917:     }
1.8       albertel 1918:     $r->print(&Apache::loncommon::end_page());
1.1       raeburn  1919:     return OK;
                   1920: }
1.15      raeburn  1921: 
1.1       raeburn  1922: 1;
                   1923: __END__
                   1924: 

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