Annotation of loncom/interface/lonprintout.pm, revision 1.627.2.38

1.389     foxr        1: # The LearningOnline Network
1.1       www         2: # Printout
                      3: #
1.627.2.38! raeburn     4: # $Id: lonprintout.pm,v 1.627.2.37 2024/10/16 18:47:18 raeburn Exp $
1.11      albertel    5: #
                      6: # Copyright Michigan State University Board of Trustees
                      7: #
                      8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
                      9: #
                     10: # LON-CAPA is free software; you can redistribute it and/or modify
                     11: # it under the terms of the GNU General Public License as published by
                     12: # the Free Software Foundation; either version 2 of the License, or
                     13: # (at your option) any later version.
                     14: #
                     15: # LON-CAPA is distributed in the hope that it will be useful,
                     16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
                     17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     18: # GNU General Public License for more details.
                     19: #
                     20: # You should have received a copy of the GNU General Public License
                     21: # along with LON-CAPA; if not, write to the Free Software
                     22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
                     23: #
                     24: # /home/httpd/html/adm/gpl.txt
                     25: # http://www.lon-capa.org/
                     26: #
1.3       sakharuk   27: #
1.1       www        28: package Apache::lonprintout;
                     29: use strict;
1.10      albertel   30: use Apache::Constants qw(:common :http);
1.2       sakharuk   31: use Apache::lonxml;
                     32: use Apache::lonnet;
1.54      sakharuk   33: use Apache::loncommon;
1.13      sakharuk   34: use Apache::inputtags;
1.54      sakharuk   35: use Apache::grades;
1.13      sakharuk   36: use Apache::edit;
1.5       sakharuk   37: use Apache::File();
1.68      sakharuk   38: use Apache::lonnavmaps;
1.511     foxr       39: use Apache::admannotations;
1.521     foxr       40: use Apache::lonenc;
1.531     foxr       41: use Apache::entities;
1.550     foxr       42: use Apache::londefdef;
1.612     foxr       43: # use Apache::structurelags;	# for language management.
1.550     foxr       44: 
                     45: use File::Basename;
1.531     foxr       46: 
1.515     foxr       47: use HTTP::Response;
1.491     albertel   48: use LONCAPA::map();
1.255     www        49: use Apache::lonlocal;
1.429     foxr       50: use Carp;
1.439     www        51: use LONCAPA;
1.60      sakharuk   52: 
1.588     foxr       53: 
1.397     albertel   54: my %perm;
1.454     foxr       55: my %parmhash;
1.459     foxr       56: my $resources_printed;
1.454     foxr       57: 
1.515     foxr       58: # Global variables that describe errors in ssi calls detected  by ssi_with_retries.
                     59: #
                     60: 
                     61: my $ssi_error;			# True if there was an ssi error.
                     62: my $ssi_last_error_resource;	# The resource URI that could not be fetched.
                     63: my $ssi_last_error;		# The error text from the server. (e.g. 500 Server timed out).
                     64: 
                     65: #
                     66: #  Our ssi max retry count.
                     67: #
                     68: 
                     69: my $ssi_retry_count = 5;	# Some arbitrary value.
                     70: 
                     71: 
1.556     foxr       72: #  Font size:
                     73: 
                     74: my $font_size = 'normalsize';	# Default is normalsize...
                     75: 
1.562     foxr       76: #----------------------------  Helper helpers. -------------------------
                     77: 
1.627.2.38! raeburn    78: ##
1.616     foxr       79: # Filter function to determine if a resource is a printable sequence.
                     80: #
                     81: # @param $res -Resource to check.
                     82: #
                     83: # @return 1 - printable and a resource
                     84: #         0 - either notm a sequence or not printable.
                     85: #
                     86: sub printable_sequence {
                     87:     my $res = shift;
                     88: 
                     89:     # Non-sequences are not listed:
                     90: 
                     91:     if (!$res->is_sequence()) {
                     92: 	return 0;
                     93:     }
                     94: 
                     95:     # Person with pav or pfo can always print:
                     96: 
                     97:     if ($perm{'pav'} || $perm{'pfo'}) {
                     98: 	return 1;
                     99:     }
                    100: 
                    101:     if ($res->is_sequence()) {
                    102: 	my $symb = $res->symb();
                    103: 	my $navmap   = $res->{NAV_MAP};
                    104: 
                    105: 	# Find the first resource in the map:
                    106: 
                    107: 	my $iterator = $navmap->getIterator($res, undef, undef, 1, 1);
                    108: 	my $first    = $iterator->next();
                    109: 
                    110: 	while (1) {
1.617     foxr      111: 	    if ($first == $iterator->END_ITERATOR) { last; }
1.616     foxr      112: 	    if (ref($first) && ! $first->is_sequence()) {last; }
                    113: 	    $first = $iterator->next();
                    114: 	}
                    115: 
                    116: 
                    117: 	# Might be an empty map:
                    118: 
                    119: 	if (!ref($first)) {
                    120: 	    return 0;
                    121: 	}
                    122: 	my $partsref = $first->parts();
                    123: 	my @parts    = @$partsref;
                    124: 	my ($open, $close) = $navmap->map_printdates($first, $parts[0]);
                    125: 	return &printable($open, $close);
                    126:     }
                    127:     return 0;
                    128: }
                    129: 
1.590     foxr      130: # BZ5209:
                    131: #    Create the states needed to run the helper for incomplete problems from
                    132: #    the current folder for selected students.
                    133: #    This includes:
                    134: #    -  A resource selector limited to problems (incompleteness must be
                    135: #       calculated on a student per student basis.
                    136: #    -  A student selector.
                    137: #    -  Tie in to the FORMAT of the print job.
                    138: #
                    139: # States:
                    140: #   CHOOSE_INCOMPLETE_PEOPLE_SEQ      - Resource selection.
                    141: #   CHOOSE_STUDENTS_INCOMPLETE        - Student selection.
                    142: #   CHOOSE_STUDENTS_INCOMPLETE_FORMAT - Format selection
                    143: # Parameters:
                    144: #    helper - the helper which already contains info about the current folder we can
                    145: #             purloin.
1.627.2.38! raeburn   146: #    map    - the map for which incomplete problems are to be printed
1.590     foxr      147: # Return:
                    148: #     XML that can be parsed by the helper to drive the state machine.
                    149: #
1.627.2.3  raeburn   150: sub create_incomplete_folder_selstud_helper {
1.590     foxr      151:     my ($helper, $map)  = @_;
                    152: 
                    153: 
                    154:     my $symbFilter = '$res->shown_symb()';
                    155:     my $selFilter   = '$res->is_problem()';
                    156: 
                    157: 
                    158:     my $resource_chooser = &generate_resource_chooser('CHOOSE_INCOMPLETE_PEOPLE_SEQ',
                    159: 						      'Select problem(s) to print',
1.627.2.24  raeburn   160: 						      'multichoice="1" toponly="1" addstatus="1" closeallpages="1" modallink="1"',
1.590     foxr      161: 						      'RESOURCES',
                    162: 						      'CHOOSE_STUDENTS_INCOMPLETE',
                    163: 						      $map,
                    164: 						      $selFilter,
                    165: 						      '',
1.627.2.38! raeburn   166: 						      $symbFilter,
1.590     foxr      167: 						      '');
                    168: 
                    169:     my $student_chooser = &generate_student_chooser('CHOOSE_STUDENTS_INCOMPLETE',
                    170: 						 'student_sort',
                    171: 						 'STUDENTS',
                    172: 						 'CHOOSE_STUDENTS_INCOMPLETE_FORMAT');
                    173: 
                    174:     my $format_chooser = &generate_format_selector($helper,
                    175: 						'Format of the print job',
1.598     raeburn   176: 						'CHOOSE_STUDENTS_INCOMPLETE_FORMAT'); # end state.
1.590     foxr      177: 
                    178:     return $resource_chooser . $student_chooser . $format_chooser;
1.627.2.38! raeburn   179: }
1.590     foxr      180: 
                    181: 
                    182: # BZ 5209
                    183: #     Create the states needed to run the helper for incomplete problems from
                    184: #     the current folder for selected students.
                    185: #     This includes:
                    186: #     - A resource selector limited to problems.  (incompleteness must be calculated
                    187: #       on a student per student basis.
                    188: #     - A student selector.
                    189: #     - Tie in to format for the print job.
                    190: # States:
                    191: #    INCOMPLETE_PROBLEMS_COURSE_RESOURCES - Resource selector.
                    192: #    INCOMPLETE_PROBLEMS_COURSE_STUDENTS  - Student selector.
                    193: #    INCOMPLETE_PROBLEMS_COURSE_FORMAT    - Format selection.
                    194: #
                    195: # Parameters:
                    196: #   helper   - Helper we are creating states for.
                    197: # Returns:
                    198: #   Text that can be parsed by the helper.
1.627.2.38! raeburn   199: #
1.590     foxr      200: 
                    201: sub create_incomplete_course_helper {
                    202:     my $helper = shift;
                    203: 
                    204:     my $filter = '$res->is_problem() || $res->contains_problem() || $res->is_sequence() || $res->is_practice())';
                    205:     my $symbfilter = '$res->shown_symb()';
1.627.2.38! raeburn   206: 
1.590     foxr      207:     my $resource_chooser = &generate_resource_chooser('INCOMPLETE_PROBLEMS_COURSE_RESOURCES',
                    208: 						      'Select problem(s) to print',
1.627.2.24  raeburn   209: 						      'multichoice = "1" suppressEmptySequences="0" addstatus="1" closeallpagtes="1" modallink="1"',
1.590     foxr      210: 						      'RESOURCES',
                    211: 						      'INCOMPLETE_PROBLEMS_COURSE_STUDENTS',
                    212: 						      '',
                    213: 						      $filter,
                    214: 						      '',
                    215: 						      $symbfilter,
                    216: 						      '');
                    217: 
                    218:     my $people_chooser  = &generate_student_chooser('INCOMPLETE_PROBLEMS_COURSE_STUDENTS',
                    219: 						    'student_sort',
                    220: 						    'STUDENTS',
                    221: 						    'INCOMPLETE_PROBLEMS_COURSE_FORMAT');
                    222: 
                    223:     my $format = &generate_format_selector($helper,
                    224: 					   'Format of the print job',
                    225: 					   'INCOMPLETE_PROBLEMS_COURSE_FORMAT'); # end state.
                    226: 
                    227:     return $resource_chooser . $people_chooser . $format;
                    228: 
                    229: 
                    230: }
                    231: 
1.627.2.38! raeburn   232: # BZ5209
1.590     foxr      233: #   Creates the states needed to run the print helper for a student
                    234: #   that wants to print his incomplete problems from the current folder.
                    235: # Parameters:
                    236: #   $helper - helper we are generating states for.
                    237: #   $map    - The map for which the student wants incomplete problems.
                    238: # Returns:
                    239: #   XML that defines the helper states being created.
                    240: #
                    241: # States:
                    242: #   CHOOSE_INCOMPLETE_SEQ  - Resource selector.
                    243: #
                    244: sub create_incomplete_folder_helper {
                    245:     my ($helper, $map) = @_;
                    246: 
                    247:     my $filter    = '$res->is_problem()';
                    248:     $filter      .= ' && $res->resprintable() ';
                    249:     $filter      .= ' && $res->is_incomplete() ';
                    250: 
                    251:     my $symfilter = '$res->shown_symb()';
                    252: 
                    253:     my $resource_chooser = &generate_resource_chooser('CHOOSE_INCOMPLETE_SEQ',
                    254: 						      'Select problem(s) to print',
1.627.2.24  raeburn   255: 						      'multichoice="1", toponly ="1", addstatus="1", closeallpages="1" modallink="1"',
1.590     foxr      256: 						      'RESOURCES',
                    257: 						      'PAGESIZE',
                    258: 						      $map,
1.627.2.38! raeburn   259: 						      $filter, '',
1.590     foxr      260: 						      $symfilter,
                    261: 						      '');
                    262: 
                    263:     return $resource_chooser;
                    264: }
                    265: 
                    266: 
                    267: #  Returns the text neded for a student chooser.
1.562     foxr      268: #  that text must still be parsed by the helper xml parser.
                    269: # Parameters:
                    270: #   this_state   - State name of the chooser.
                    271: #   sort_choice  - variable to hold the sorting choice.
                    272: #   variable     - Name of variable to hold students.
                    273: #   next_state   - State after chooser.
                    274: 
                    275: 
                    276: sub generate_student_chooser {
1.627.2.38! raeburn   277:     my ($this_state,
        !           278: 	$sort_choice,
        !           279: 	$variable,
1.562     foxr      280: 	$next_state) = @_;
                    281:     my $result = <<CHOOSE_STUDENTS;
                    282:   <state name="$this_state" title="Select Students and Resources">
                    283:       <message><b>Select sorting order of printout</b> </message>
                    284: 
                    285:     <choices variable="$sort_choice">
                    286:       <choice computer='0'>Sort by section then student</choice>
                    287:       <choice computer='1'>Sort by students across sections.</choice>
                    288:     </choices>
                    289: 
                    290:       <message><br /><hr /><br /> </message>
1.627.2.38! raeburn   291:       <student multichoice='1'
1.562     foxr      292:                variable="$variable" 
                    293:                nextstate="$next_state" 
                    294:                coursepersonnel="1" />
                    295:   </state>
                    296: 
                    297: CHOOSE_STUDENTS
                    298: 
                    299:   return $result;
                    300: }
                    301: 
                    302: # Generate the text needed for a resource chooser given the top level of
                    303: # the sequence/page
                    304: #
                    305: # Parameters:
                    306: #     this_state    - State name of the chooser.
                    307: #     prompt_text   - Text to use to prompt user.
                    308: #     resource_options - Resource tag options e.g.
1.627.2.24  raeburn   309: #                        "multichoice='1', toponly='1', addstatus='1',
                    310: #                         modallink='1'"
1.562     foxr      311: #                     that control the selection and appearance of the
                    312: #                     resource selector.
                    313: #     variable      - Name of the variable to hold the choice
                    314: #     next_state    - Name of the next state the helper should transition
                    315: #                     to
                    316: #     top_url       - Top level URL within which to make the selector.
                    317: #                     If empty the top level sequence is shown.
                    318: #     filter        - How to filter the resources.
                    319: #     value_func    - <valuefunc> function.
                    320: #     choice_func   - If not empty generates a <choicefunc> with this function.
1.627.2.38! raeburn   321: #     start_new_option
1.562     foxr      322: #                   - Fragment appended after valuefunc.
                    323: #
                    324: #
                    325: sub generate_resource_chooser {
                    326:     my ($this_state,
                    327: 	$prompt_text,
                    328: 	$resource_options,
                    329: 	$variable,
                    330: 	$next_state,
                    331: 	$top_url,
                    332: 	$filter,
                    333: 	$choice_func,
                    334: 	$value_func,
                    335: 	$start_new_option)  = @_;
                    336: 
                    337:     my $result = <<CHOOSE_RESOURCES;
                    338: <state name="$this_state" title="$prompt_text">
                    339:     <resource variable="$variable" $resource_options
                    340:               closeallpages="1">
                    341:       <nextstate>$next_state</nextstate>
                    342:       <filterfunc>return $filter;</filterfunc>
                    343: CHOOSE_RESOURCES
                    344:     if ($choice_func ne '') {
                    345: 	$result .= "<choicefunc>return $choice_func;</choicefunc>";
                    346:     }
                    347:     if ($top_url ne '') {
                    348: 	$result .=  "<mapurl>$top_url</mapurl>";
                    349:     }
                    350:     $result .= <<CHOOSE_RESOURCES;
                    351:       <valuefunc>return $value_func;</valuefunc>
                    352:       $start_new_option
                    353:       </resource>
                    354:     </state>
                    355: CHOOSE_RESOURCES
                    356:     return $result;
                    357: }
                    358: #
                    359: #   Generate the helper XML for a code choice helper dialog:
                    360: #
                    361: # Paramters:
                    362: #   $helper       - Reference to the helper.
                    363: #   $state        - Name of the state for the chooser.
                    364: #   $next_state   - Name fo the state to follow the chooser.
                    365: #   $bubble_types - Populates the bubble sheet type dropt down.
                    366: #   $code_selections - Provides set of code choices that have been used
                    367: #   $saved_codes  - Provides the list of saved codes.
                    368: #
                    369: # Returns;
                    370: #   The Xml of the code chooser.
                    371: #
                    372: sub generate_code_selector {
                    373:     my ($helper,
                    374: 	$state,
                    375: 	$next_state,
                    376: 	$bubble_types,
                    377: 	$code_selections,
                    378: 	$saved_codes) = @_;	# Unpack the parameters.
                    379: 
                    380:     my $result = <<CHOOSE_ANON1;
                    381:   <state name="$state" title="Specify CODEd Assignments">
                    382:     <nextstate>$next_state</nextstate>
                    383:     <message><h4>Fill out one of the forms below</h4></message>
                    384:     <message><br /><hr /> <br /></message>
                    385:     <message><h3>Generate new CODEd Assignments</h3></message>
                    386:     <message><table><tr><td><b>Number of CODEd assignments to print:</b></td><td></message>
1.579     foxr      387:     <string variable="NUMBER_TO_PRINT_TOTAL" maxlength="5" size="5"  noproceed="1">
1.562     foxr      388:        <validator>
                    389: 	if (((\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}+0) < 1) &&
                    390: 	    !\$helper->{'VARS'}{'REUSE_OLD_CODES'}                &&
                    391:             !\$helper->{'VARS'}{'SINGLE_CODE'}                    &&
1.578     foxr      392: 	    !\$helper->{'VARS'}{'CODE_SELECTED_FROM_LIST'} ) {
                    393: 
1.562     foxr      394: 	    return "You need to specify the number of assignments to print";
                    395: 	}
1.578     foxr      396:         if (((\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}+0) >= 1)  &&
                    397:              (\$helper->{'VARS'}{'SINGLE_CODE'} ne '') ) {
                    398:             return 'Specifying number of codes to print and a specific code is not compatible';
                    399:         }
1.562     foxr      400: 	return undef;
                    401:        </validator>
                    402:     </string>
                    403:     <message></td></tr><tr><td></message>
                    404:     <message><b>Names to save the CODEs under for later:</b></message>
                    405:     <message></td><td></message>
                    406:     <string variable="ANON_CODE_STORAGE_NAME" maxlength="50" size="20" />
                    407:     <message></td></tr><tr><td></message>
1.599     raeburn   408:     <message><b>Bubblesheet type:</b></message>
1.562     foxr      409:     <message></td><td></message>
                    410:     <dropdown variable="CODE_OPTION" multichoice="0" allowempty="0">
                    411:     $bubble_types
                    412:     </dropdown>
                    413:     <message></td></tr><tr><td colspan="2"></td></tr><tr><td></message>
                    414:     <message></td></tr><tr><td></table></message>
                    415:     <message><br /><hr /><h3>Print a Specific CODE </h3><br /><table></message>
                    416:     <message><tr><td><b>Enter a CODE to print:</b></td><td></message>
                    417:     <string variable="SINGLE_CODE" size="10">
                    418:         <validator>
                    419: 	   if(!\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}           &&
                    420: 	      !\$helper->{'VARS'}{'REUSE_OLD_CODES'}                 &&
                    421: 	      !\$helper->{'VARS'}{'CODE_SELECTED_FROM_LIST'}) {
                    422: 	      return &Apache::lonprintout::is_code_valid(\$helper->{'VARS'}{'SINGLE_CODE'},
                    423: 						      \$helper->{'VARS'}{'CODE_OPTION'});
1.577     foxr      424: 	  } elsif (\$helper->{'VARS'}{'SINGLE_CODE'} ne ''){
1.578     foxr      425: 	      return 'Specifying a code name is incompatible with specifying number of codes.';
1.562     foxr      426: 	   } else {
                    427: 	       return undef;	# Other forces control us.
                    428: 	   }
                    429:         </validator>
                    430:     </string>
                    431:     <message></td></tr><tr><td></message>
                    432:         $code_selections
                    433:     <message></td></tr></table></message>
                    434:     <message><hr /><h3>Reprint a Set of Saved CODEs</h3><table><tr><td></message>
                    435:     <message><b>Select saved CODEs:</b></message>
                    436:     <message></td><td></message>
                    437:     <dropdown variable="REUSE_OLD_CODES">
                    438:         $saved_codes
                    439:     </dropdown>
                    440:     <message></td></tr></table></message>
                    441:   </state>
                    442: CHOOSE_ANON1
                    443: 
                    444:    return $result;
                    445: }
                    446: 
1.627.2.38! raeburn   447: #  Returns the XML for choosing how assignments are to be formatted
1.598     raeburn   448: #  that text must still be parsed by the helper xml parser.
                    449: # Parameters: 3 (required)
                    450: 
                    451: #   helper       - The helper; $helper->{'VARS'}->{'PRINT_TYPE'} used
                    452: #                  to check if splitting PDFs by section can be offered.
1.627.2.38! raeburn   453: #   title        - Title for the current state.
1.598     raeburn   454: #   this_state   - State name of the chooser.
                    455: 
1.586     raeburn   456: sub generate_format_selector {
1.598     raeburn   457:     my ($helper,$title,$this_state) = @_;
1.586     raeburn   458:     my $secpdfoption;
                    459:     unless (($helper->{'VARS'}->{'PRINT_TYPE'} eq 'problems_for_anon')     ||
                    460:             ($helper->{'VARS'}->{'PRINT_TYPE'} eq 'problems_for_anon_page') ||
                    461:             ($helper->{'VARS'}->{'PRINT_TYPE'} eq 'resources_for_anon')  ) {
                    462:         $secpdfoption =  '<choice computer="sections">Each PDF contains exactly one section</choice>';
                    463:     }
                    464:     return <<RESOURCE_SELECTOR;
1.598     raeburn   465:     <state name="$this_state" title="$title">
1.586     raeburn   466:     <message><br /><big><i><b>How should the results be printed?</b></i></big><br /></message>
                    467:     <choices variable="EMPTY_PAGES">
                    468:       <choice computer='0'>Start each student\'s assignment on a new page/column (add a pagefeed after each assignment)</choice>
                    469:       <choice computer='1'>Add one empty page/column after each student\'s assignment</choice>
                    470:       <choice computer='2'>Add two empty pages/column after each student\'s assignment</choice>
                    471:       <choice computer='3'>Add three empty pages/column after each student\'s assignment</choice>
                    472:     </choices>
                    473:     <nextstate>PAGESIZE</nextstate>
                    474:     <message><hr width='33%' /><b>How do you want assignments split into PDF files? </b></message>
                    475:     <choices variable="SPLIT_PDFS">
                    476:        <choice computer="all">All assignments in a single PDF file</choice>
                    477:        $secpdfoption
                    478:        <choice computer="oneper">Each PDF contains exactly one assignment</choice>
                    479:        <choice computer="usenumber" relatedvalue="NUMBER_TO_PRINT">
                    480:             Specify the number of assignments per PDF:</choice>
                    481:     </choices>
                    482:     </state>
                    483: RESOURCE_SELECTOR
                    484: }
                    485: 
1.562     foxr      486: #-----------------------------------------------------------------------
                    487: 
1.616     foxr      488: # Computes an open and close date from a list of open/close dates for a resource's
                    489: # parts.
                    490: #
                    491: # @param \@opens - reference to an array of open dates.
                    492: # @param \@closes - reference to an array of close dates.
                    493: #
1.627.2.38! raeburn   494: # @return ($open, $close)
1.616     foxr      495: #
1.627.2.9  raeburn   496: # @note If open/close dates are not defined they will be returned as undef
1.627.2.38! raeburn   497: # @note It is possible for there to be no overlap in which case -1,-1
1.616     foxr      498: #       will be returned.
                    499: # @note The algorithm used is to take the latest open date and the earliest end date.
                    500: #
                    501: sub compute_open_window {
                    502:     my ($opensref, $closesref) = @_;
                    503: 
                    504:     my @opens   = @$opensref;
                    505:     my @closes  = @$closesref;
                    506: 
                    507:     # latest open date:
                    508:     my $latest_open;
                    509: 
                    510:     foreach my $open (@opens) {
                    511: 	if (!defined($latest_open) || ($open > $latest_open)) {
                    512: 	    $latest_open = $open;
                    513: 	}
                    514:     }
                    515:     # Earliest close:
                    516: 
                    517:     my $earliest_close;
                    518:     foreach my $close (@closes) {
                    519: 	if (!defined($earliest_close) || ($close < $earliest_close)) {
                    520: 	    $earliest_close = $close;
                    521: 	}
                    522:     }
                    523: 
                    524:     # If no overlap...both are -1 as promised.
                    525: 
1.627.2.9  raeburn   526:     if (($earliest_close ne '') && ($latest_open ne '')
1.616     foxr      527: 	 && ($earliest_close < $latest_open)) {
                    528: 	$latest_open  = -1;
                    529: 	$earliest_close = -1;
                    530:     }
1.627.2.38! raeburn   531: 
1.616     foxr      532:     return ($latest_open, $earliest_close);
1.627.2.38! raeburn   533: 
1.616     foxr      534: }
                    535: 
                    536: ##
                    537: #  Determines if 'now' is within the set of printable dates.
                    538: #
                    539: #  @param $open_date - Starting date/timestamp.
                    540: #  @param $close_date - Ending date/timestamp.
                    541: #
                    542: #  @return 0 - Not open.
                    543: #  @return 1 - open.
                    544: #
                    545: sub printable {
                    546:     my ($open_date, $close_date) = @_;
                    547: 
                    548: 
                    549:     my $now = time();
                    550: 
                    551:     # Have to do a bit of fancy footwork around undefined open/close dates:
                    552: 
                    553:     if ($open_date && ($open_date > $now)) {
                    554: 	return 0;
                    555:     }
                    556: 
                    557:     if ($close_date && ($close_date < $now)) {
                    558: 	return 0;
                    559:     }
1.627.2.38! raeburn   560: 
1.616     foxr      561:     return 1;
                    562: 
                    563: }
                    564: 
1.615     foxr      565: ##
                    566: # Returns the innermost print start/print end dates for a resource.
                    567: # This is done by looking at the start/end dates for its parts and choosing
                    568: # the intersection of those dates.
1.627.2.38! raeburn   569: #
1.615     foxr      570: # @param res - lonnvamaps::resource object that represents the resource.
                    571: #
                    572: # @return (opendate, closedate)
                    573: #
1.627.2.9  raeburn   574: # @note If open/close dates are not defined they will be returned as undef
1.627.2.38! raeburn   575: # @note It is possible for there to be no overlap in which case -1,-1
1.615     foxr      576: #       will be returned.
                    577: # @note The algorithm used is to take the latest open date and the earliest end date.
1.627.2.9  raeburn   578: #       For consistency with &printable() in lonnavmaps.pm determination of start
                    579: #       date for printing checks printstartdate param first, then, if not set,
                    580: #       opendate param, then, if not set, contentopen param.
                    581: 
1.615     foxr      582: sub get_print_dates {
                    583:     my $res = shift;
                    584:     my $partsref = $res->parts();
1.620     raeburn   585:     my @parts;
                    586:     if (ref($partsref) eq 'ARRAY') {
                    587:         @parts   = @{$partsref};
                    588:     }
1.615     foxr      589:     my $open_date;
                    590:     my $close_date;
1.616     foxr      591:     my @open_dates;
                    592:     my @close_dates;
                    593: 
1.615     foxr      594: 
1.620     raeburn   595:     if (@parts) {
1.615     foxr      596: 	foreach my $part (@parts) {
                    597: 	    my $partopen  = $res->parmval('printstartdate', $part);
                    598: 	    my $partclose = $res->parmval('printenddate',  $part);
1.627.2.9  raeburn   599:             if (!$partopen) {
                    600:                 $partopen = $res->parmval('opendate',$part);
                    601:             }
                    602:             if (!$partopen) {
                    603:                 $partopen = $res->parmval('contentopen',$part);
                    604:             }
                    605:             if ($partopen) {
                    606:                 push(@open_dates, $partopen);
                    607:             }
                    608:             if ($partclose) {
                    609:                 push(@close_dates, $partclose);
                    610:             }
1.616     foxr      611: 	    push(@open_dates, $partopen);
                    612: 	    push(@close_dates, $partclose);
                    613: 	}
                    614:     }
                    615: 
                    616:     ($open_date, $close_date)  = &compute_open_window(\@open_dates, \@close_dates);
                    617: 
                    618:     return ($open_date, $close_date);
                    619: }
                    620: 
                    621: ##
                    622: # Get the dates for which a course says a resource can be printed.  This is like
                    623: # get_print_dates but namvaps::course_print_dates are gotten...and not converted
                    624: # to times either.
                    625: #
1.627.2.38! raeburn   626: # @param $res - Reference to a resource hash from lonnavmaps::resource.
1.616     foxr      627: #
                    628: # @return (opendate, closedate)
                    629: #
                    630: sub course_print_dates {
                    631:     my $res = shift;
                    632:     my $partsref = $res->parts();
                    633:     my @parts    = @$partsref;
                    634:     my $open_date;
                    635:     my $close_date;
                    636:     my @open_dates;
                    637:     my @close_dates;
                    638:     my $navmap = $res->{NAV_MAP}; # Slightly OO dirty.
                    639: 
1.627.2.38! raeburn   640:     # Don't bother looping over undefined or empty parts array;
1.616     foxr      641: 
1.620     raeburn   642:     if (@parts) {
1.616     foxr      643: 	foreach my $part (@parts) {
                    644: 	    my ($partopen, $partclose) = $navmap->course_printdates($res, $part);
                    645: 	    push(@open_dates, $partopen);
                    646: 	    push(@close_dates, $partclose);
1.615     foxr      647: 	}
1.616     foxr      648: 	($open_date, $close_date) = &compute_open_window(\@open_dates, \@close_dates);
1.615     foxr      649:     }
1.616     foxr      650:     return ($open_date, $close_date);
                    651: }
                    652: ##
                    653: # Same as above but for the enclosing map:
                    654: #
                    655: sub map_print_dates {
                    656:     my $res = shift;
                    657:     my $partsref = $res->parts();
                    658:     my @parts    = @$partsref;
                    659:     my $open_date;
                    660:     my $close_date;
                    661:     my @open_dates;
                    662:     my @close_dates;
                    663:     my $navmap = $res->{NAV_MAP}; # slightly OO dirty.
                    664: 
                    665: 
1.627.2.38! raeburn   666:     # Don't bother looping over undefined or empty parts array;
1.615     foxr      667: 
1.620     raeburn   668:     if (@parts) {
1.616     foxr      669: 	foreach my $part (@parts) {
                    670: 	    my ($partopen, $partclose) = $navmap->map_printdates($res, $part);
                    671: 	    push(@open_dates, $partopen);
                    672: 	    push(@close_dates, $partclose);
                    673: 	}
                    674: 	($open_date, $close_date) = &compute_open_window(\@open_dates, \@close_dates);
                    675:     }
1.615     foxr      676:     return ($open_date, $close_date);
                    677: }
                    678: 
1.591     foxr      679: # Determine if a resource is incomplete given the map:
                    680: # Parameters:
                    681: #   $username - Name of user for whom we are checking.
                    682: #   $domain   - Domain of user we are checking.
                    683: #   $map - map name.
                    684: # Returns:
                    685: #     0 - map is not incomplete.
                    686: #     1 - map is incomplete.
                    687: #
                    688: sub incomplete {
                    689:     my ($username, $domain, $map) = @_;
                    690: 
                    691: 
1.595     foxr      692:     my $navmap = Apache::lonnavmaps::navmap->new($username, $domain);
1.627.2.38! raeburn   693: 
1.591     foxr      694: 
                    695:     if (defined($navmap)) {
                    696: 	my $res = $navmap->getResourceByUrl($map);
                    697: 	my $result = $res->is_incomplete();
                    698: 	return $result;
                    699:     } else {
                    700: 	return 1;
                    701:     }
                    702: }
1.595     foxr      703: #
1.627.2.15  raeburn   704: #  When printing for students, the resources and order of the
1.595     foxr      705: #  resources may need to be altered if there are folders with
                    706: #  random selectiopn or random ordering (or both) enabled.
                    707: #  This sub computes the set of resources to print for a student
                    708: #  modified both by random ordering and selection and filtered
1.627.2.15  raeburn   709: #  to only those that are in the original set selected to be printed.
1.595     foxr      710: #
                    711: # Parameters:
1.626     raeburn   712: #   $map - The URL of the folder being printed.
                    713: #          Used to determine which startResource and finishResource
                    714: #          to use when using the navmap's getIterator method.
                    715: #   $seq   - The original set of resources to print.
1.600     foxr      716: #            (really an array of resource names (array of symb's).
1.595     foxr      717: #   $who   - Student/domain for whome the sequence will be generated.
1.626     raeburn   718: #   $code  - CODE being printed when printing Problems/Resources
                    719: #            from folder for CODEd assignments
1.627.2.15  raeburn   720: #   $nohidemap - If true, parameter in map for hiddenresource will be
                    721: #                ignored.  The user calling the routine should have
                    722: #                both the pav and vgr privileges if this is set to true).
1.595     foxr      723: #
                    724: # Implicit inputs:
                    725: #   $
                    726: # Returns:
                    727: #   reference to an array of resources that can be passed to
                    728: #   print_resources.
1.627.2.38! raeburn   729: #
1.595     foxr      730: sub master_seq_to_person_seq {
1.627.2.5  raeburn   731:     my ($map, $seq, $who, $code, $nohidemap) = @_;
1.595     foxr      732: 
                    733: 
                    734:     my ($username, $userdomain, $usersection) = split(/:/, $who);
                    735: 
                    736:     # Toss the sequence up into a hash so that we have O(1) lookup time.
                    737:     # on the items that come out of the user's list of resources.
                    738:     #
1.626     raeburn   739: 
1.595     foxr      740:     my %seq_hash = map {$_  => 1} @$seq;
                    741:     my @output_seq;
1.627.2.5  raeburn   742: 
                    743:     my $unhidden;
1.627.2.15  raeburn   744:     if ($nohidemap) {
1.627.2.5  raeburn   745:         $unhidden = &Apache::lonnet::clutter($map);
                    746:     }
1.627.2.38! raeburn   747: 
1.625     raeburn   748:     my $navmap           = Apache::lonnavmaps::navmap->new($username, $userdomain,
1.627.2.38! raeburn   749:                                                            $code, $unhidden);
1.626     raeburn   750:     my ($start,$finish);
1.595     foxr      751: 
1.626     raeburn   752:     if ($map) {
                    753:         my $mapres = $navmap->getResourceByUrl($map);
                    754:         if ($mapres->is_map()) {
                    755:             $start = $mapres->map_start();
                    756:             $finish = $mapres->map_finish();
                    757:         }
                    758:     }
                    759:     unless ($start && $finish) {
                    760:         $start = $navmap->firstResource();
                    761:         $finish = $navmap->finishResource();
                    762:     }
                    763: 
                    764:     my $iterator         = $navmap->getIterator($start,$finish,{},1);
1.595     foxr      765: 
                    766:     #  Iterate on the resource..select the items that are randomly selected
1.627.2.38! raeburn   767:     #  and that are in the seq_hash.  Presumably the iterator will take care
        !           768:     #  of the random ordering part of the deal.
        !           769:     #
1.595     foxr      770:     my $curres;
                    771:     while ($curres = $iterator->next()) {
                    772: 	#
1.600     foxr      773: 	#  Only process resources..that are not removed by randomout...
                    774: 	#  and are selected for printint as well.
1.595     foxr      775: 	#
1.626     raeburn   776:         if (ref($curres) && ! $curres->randomout()) {
                    777:             my $currsymb = $curres->symb();
                    778:             if (exists($seq_hash{$currsymb})) {
                    779:                 push(@output_seq, $currsymb);
1.595     foxr      780: 	    }
                    781: 	}
                    782:     }
                    783: 
                    784:     return \@output_seq;		# for now.
1.627.2.38! raeburn   785: 
1.595     foxr      786: }
                    787: 
1.515     foxr      788: 
1.498     foxr      789: # Fetch the contents of a resource, uninterpreted.
                    790: # This is used here to fetch a latex file to be included
                    791: # verbatim into the printout<
                    792: # NOTE: Ask Guy if there is a lonnet function similar to this?
                    793: #
                    794: # Parameters:
                    795: #   URL of the file
                    796: #
                    797: sub fetch_raw_resource {
                    798:     my ($url) = @_;
                    799: 
                    800:     my $filename  = &Apache::lonnet::filelocation("", $url);
1.500     foxr      801:     my $contents  = &Apache::lonnet::getfile($filename);
1.498     foxr      802: 
1.500     foxr      803:     if ($contents == -1) {
                    804: 	return "File open failed for $filename";      # This will bomb the print.
1.498     foxr      805:     }
1.500     foxr      806:     return $contents;
1.498     foxr      807: 
1.627.2.38! raeburn   808: 
1.498     foxr      809: }
                    810: 
1.627.2.38! raeburn   811: #  Fetch the annotations associated with a URL and
1.511     foxr      812: #  put a centered 'annotations:' title.
                    813: #  This is all suppressed if the annotations are empty.
                    814: #
                    815: sub annotate {
                    816:     my ($symb) = @_;
                    817: 
1.559     foxr      818:     my $annotation_text = &Apache::loncommon::get_annotation($symb, 1);
1.511     foxr      819: 
                    820: 
                    821:     my $result = "";
                    822: 
                    823:     if (length($annotation_text) > 0) {
                    824: 	$result .= '\\hspace*{\\fill} \\\\[\\baselineskip] \textbf{Annotations:} \\\\ ';
                    825: 	$result .= "\n";
                    826: 	$result .= &Apache::lonxml::latex_special_symbols($annotation_text,"");	# Escape latex.
                    827: 	$result .= "\n\n";
                    828:     }
                    829:     return $result;
                    830: }
                    831: 
1.556     foxr      832: #
                    833: #   Set a global document font size:
                    834: #   This is done by replacing \begin{document}
                    835: #   with \begin{document}{\some-font-directive
                    836: #   and \end{document} with
                    837: #   }\end{document
                    838: #
                    839: sub set_font_size {
                    840: 
                    841:     my ($text) = @_;
                    842: 
1.575     foxr      843:     # There appear to be cases where the font directive is empty.. in which
1.627.2.38! raeburn   844:     # case the first substitution would insert a spurious \ oh happy day.
1.575     foxr      845:     # as this has been the cause of much mystery and hair pulling _sigh_
                    846: 
                    847:     if ($font_size ne '') {
                    848: 
1.627.2.17  raeburn   849: 	$text =~ s/\\begin\{document}/\\begin{document}{\\$font_size/;
1.627.2.27  raeburn   850: 	$text =~ s/\\end\{document}/}\\end{document}/;
1.575     foxr      851:     }
1.556     foxr      852:     return $text;
                    853: 
                    854: 
                    855: }
                    856: 
1.627.2.38! raeburn   857: # include_pdf - PDF files are included into the
1.550     foxr      858: # output as follows:
                    859: #  - The PDF, if necessary, is replicated.
                    860: #  - The PDF is added to the list of files to convert to postscript (along with the images).
                    861: #  - The LaTeX is added to include the final converted postscript in the file as an included
1.627.2.38! raeburn   862: #    job.  The assumption is that the includepsheader.ps header will be included.
1.550     foxr      863: #
                    864: # Parameters:
                    865: #   pdf_uri   - URI of the PDF file to include.
1.627.2.38! raeburn   866: #
1.550     foxr      867: # Returns:
                    868: #  The LaTeX to include.
                    869: #
                    870: # Assumptions:
                    871: #    The uri is actually a PDF file
                    872: #    The postscript will have the includepsheader.ps included.
                    873: #
                    874: #
                    875: sub include_pdf {
                    876:     my ($pdf_uri) = @_;
                    877: 
                    878:     # Where is the file? If not local we'll need to repcopy it:'
                    879: 
                    880:     my $file = &Apache::lonnet::filelocation('', $pdf_uri);
                    881:     if (! -e $file) {
                    882: 	&Apache::lonnet::repcopy($file);
                    883: 	$file = &Apache::lonnet::filelocation('',$pdf_uri);
                    884:     }
                    885: 
1.627.2.38! raeburn   886:     #  The file is now replicated locally ... or it did not exist in the first place
1.550     foxr      887:     # (unlikely).  If it did exist, add the pdf to the set of files/images that
1.627.2.38! raeburn   888:     # need to be converted for this print job:
1.550     foxr      889: 
1.608     raeburn   890:     my $londocroot = $Apache::lonnet::perlvar{'lonDocRoot'};
                    891:     $file =~ s{(.*)/res/}{$londocroot/res/};
1.550     foxr      892: 
1.627.2.22  raeburn   893:     open(FILE,">>","$Apache::lonnet::perlvar{'lonPrtDir'}/$env{'user.name'}_$env{'user.domain'}_printout.dat");
1.550     foxr      894:     print FILE ("$file\n");
                    895:     close (FILE);
                    896: 
                    897:     # Construct the special to put out.  To do this we need to get the
                    898:     # resulting filename after conversion.  The file will have the same name
                    899:     # but will be in the user's spool directory with converted images.
                    900: 
                    901:     my $dirname = "/home/httpd/prtspool/$env{'user.name'}/";
                    902:     my ( $base, $path,  $ext) = &fileparse($file, '.pdf');
                    903: #    my $destname = $dirname.'/'.$base.'.eps'; # Not really an eps but easier in printout.pl
                    904:     $base =~ s/ /\_/g;
                    905: 
                    906: 
1.551     foxr      907:     my $output = &print_latex_header();
1.550     foxr      908:     $output    .= '\special{ps: _begin_job_ ('
                    909: 	.$base.'.pdf.eps'.
                    910: 	')run _end_job_}';
                    911: 
                    912:     return $output;
                    913: 
                    914: 
                    915: }
1.612     foxr      916: ##
                    917: #  Collect the various \select_language{language_name}
                    918: #  latex tags to build a \usepackage[lang-list]{babel} which will
                    919: #  appear just prior to the \begin{document} at the front of the concatenated
                    920: #  set of resources:
                    921: # @param doc - The string of latex to search/replace.
                    922: # @return string
                    923: # @retval - the modified document stringt.
                    924: #
                    925: sub collect_languages {
                    926:     my $doc = shift;
                    927:     my %languages;
1.627.2.17  raeburn   928:     while ($doc =~ /\\selectlanguage\{(\w+)}/mg) {
1.612     foxr      929: 	$languages{$1} = 1;	# allows us to request each language exactly once.
                    930:     }
                    931:     my @lang_list = (keys(%languages)); # List of unique languages
                    932:     if (scalar @lang_list) {
                    933: 	my $babel_header = '\usepackage[' . join(',', @lang_list) .']{babel}'. "\n";
1.627.2.17  raeburn   934: 	$doc =~ s/\\begin\{document}/$babel_header\\begin{document}/;
1.612     foxr      935:     }
                    936:     return $doc;
                    937: }
                    938: #-------------------------------------------------------------------
1.515     foxr      939: 
                    940: #
1.559     foxr      941: #   ssi_with_retries- Does the server side include of a resource.
1.515     foxr      942: #                      if the ssi call returns an error we'll retry it up to
                    943: #                      the number of times requested by the caller.
                    944: #                      If we still have a proble, no text is appended to the
                    945: #                      output and we set some global variables.
1.627.2.38! raeburn   946: #                      to indicate to the caller an SSI error occurred.
1.515     foxr      947: #                      All of this is supposed to deal with the issues described
                    948: #                      in LonCAPA BZ 5631 see:
                    949: #                      http://bugs.lon-capa.org/show_bug.cgi?id=5631
                    950: #                      by informing the user that this happened.
                    951: #
                    952: # Parameters:
                    953: #   resource   - The resource to include.  This is passed directly, without
                    954: #                interpretation to lonnet::ssi.
                    955: #   form       - The form hash parameters that guide the interpretation of the resource
1.627.2.38! raeburn   956: #
1.515     foxr      957: #   retries    - Number of retries allowed before giving up completely.
                    958: # Returns:
                    959: #   On success, returns the rendered resource identified by the resource parameter.
                    960: # Side Effects:
                    961: #   The following global variables can be set:
1.523     raeburn   962: #    ssi_error                - If an unrecoverable error occurred this becomes true.
1.515     foxr      963: #                               It is up to the caller to initialize this to false
                    964: #                               if desired.
1.523     raeburn   965: #    ssi_last_error_resource  - If an unrecoverable error occurred, this is the value
1.515     foxr      966: #                               of the resource that could not be rendered by the ssi
                    967: #                               call.
                    968: #    ssi_last_error           - The error string fetched from the ssi response
                    969: #                               in the event of an error.
                    970: #
                    971: sub ssi_with_retries {
                    972:     my ($resource, $retries, %form) = @_;
                    973: 
1.559     foxr      974:     my $target = $form{'grade_target'};
                    975:     my $aom    = $form{'answer_output_mode'};
                    976: 
                    977: 
1.515     foxr      978: 
1.516     foxr      979:     my ($content, $response) = &Apache::loncommon::ssi_with_retries($resource, $retries, %form);
                    980:     if (!$response->is_success) {
1.515     foxr      981: 	$ssi_error               = 1;
                    982: 	$ssi_last_error_resource = $resource;
1.516     foxr      983: 	$ssi_last_error          = $response->code . " " . $response->message;
1.528     raeburn   984:         $content='\section*{!!! An error occurred !!!}';	
1.515     foxr      985:     }
1.516     foxr      986: 
                    987:     return $content;
                    988: 
1.515     foxr      989: }
                    990: 
1.524     www       991: sub get_student_view_with_retries {
                    992:     my ($curresline,$retries,$username,$userdomain,$courseid,$target,$moreenv)=@_;
                    993: 
                    994:     my ($content, $response) = &Apache::loncommon::get_student_view_with_retries($curresline,$retries,$username,$userdomain,$courseid,$target,$moreenv);
                    995:     if (!$response->is_success) {
                    996:         $ssi_error               = 1;
1.526     www       997:         $ssi_last_error_resource = $curresline.' for user '.$username.':'.$userdomain;
1.524     www       998:         $ssi_last_error          = $response->code . " " . $response->message;
1.528     raeburn   999:         $content='\section*{!!! An error occurred !!!}';
1.524     www      1000:     }
                   1001:     return $content;
                   1002: 
                   1003: }
                   1004: 
1.486     foxr     1005: #
                   1006: #   printf_style_subst  item format_string repl
1.627.2.38! raeburn  1007: #
1.486     foxr     1008: # Does printf style substitution for a format string that
                   1009: # can have %[n]item in it.. wherever, %[n]item occurs,
                   1010: # rep is substituted in format_string.  Note that
                   1011: # [n] is an optional integer length.  If provided,
1.627.2.38! raeburn  1012: # repl is truncated to at most [n] characters prior to
1.486     foxr     1013: # substitution.
                   1014: #
                   1015: sub printf_style_subst {
                   1016:     my ($item, $format_string, $repl) = @_;
1.490     foxr     1017:     my $result = "";
                   1018:     while ($format_string =~ /(%)(\d*)\Q$item\E/g ) {
1.488     albertel 1019: 	my $fmt = $1;
                   1020: 	my $size = $2;
1.486     foxr     1021: 	my $subst = $repl;
                   1022: 	if ($size ne "") {
                   1023: 	    $subst = substr($subst, 0, $size);
1.627.2.38! raeburn  1024: 
        !          1025: 	    #  Here's a nice edge case ... suppose the end of the
        !          1026: 	    #  substring is a \.  In that case may have just
1.490     foxr     1027: 	    #  chopped off a TeX escape... in that case, we append
1.627.2.38! raeburn  1028: 	    #   " " for the trailing character, and let the field
1.490     foxr     1029: 	    #  spill over a bit (sigh).
                   1030: 	    #  We don't just chop off the last character in order to deal
                   1031: 	    #  with one last pathology, and that would be if substr had
1.627.2.38! raeburn  1032: 	    #  trimmed us to e.g. \\\
1.490     foxr     1033: 
                   1034: 
                   1035: 	    if ($subst =~ /\\$/) {
                   1036: 		$subst .= " ";
                   1037: 	    }
1.486     foxr     1038: 	}
1.490     foxr     1039: 	my $item_pos = pos($format_string);
                   1040: 	$result .= substr($format_string, 0, $item_pos - length($size) -2) . $subst;
                   1041:         $format_string = substr($format_string, pos($format_string));
1.486     foxr     1042:     }
1.490     foxr     1043: 
                   1044:     # Put the residual format string into the result:
                   1045: 
                   1046:     $result .= $format_string;
                   1047: 
                   1048:     return $result;
1.486     foxr     1049: }
                   1050: 
1.454     foxr     1051: 
1.627.2.38! raeburn  1052: # Format a header according to a format.
        !          1053: #
1.454     foxr     1054: 
                   1055: # Substitutions:
                   1056: #     %a    - Assignment name.
                   1057: #     %c    - Course name.
                   1058: #     %n    - Student name.
1.537     foxr     1059: #     %s    - The section if it is supplied.
1.454     foxr     1060: #
                   1061: sub format_page_header {
1.627.2.1  raeburn  1062:     my ($width, $format, $assignment, $course, $student) = @_;
1.537     foxr     1063: 
1.565     foxr     1064: 
                   1065: 
1.486     foxr     1066:     $width = &recalcto_mm($width); # Get width in mm.
1.627.2.1  raeburn  1067:     my $chars_per_line = int($width/2);   # Character/textline.
1.565     foxr     1068: 
1.454     foxr     1069:     #  Default format?
                   1070: 
                   1071:     if ($format eq '') {
1.486     foxr     1072: 	# For the default format, we may need to truncate
                   1073: 	# elements..  To do this we need to get the page width.
                   1074: 	# we assume that each character is about 2mm in width.
                   1075: 	# (correct for the header text size??).  We ignore
                   1076: 	# any formatting (e.g. boldfacing in this).
1.627.2.38! raeburn  1077: 	#
1.486     foxr     1078: 	# - Allow the student/course to be one line.
                   1079: 	#   but only truncate the course.
                   1080: 	# - Allow the assignment to be 2 lines (wrapped).
                   1081: 	#
1.565     foxr     1082: 
1.627.2.1  raeburn  1083:         my $firstline = "$student $course";
                   1084:         if (length($firstline) > $chars_per_line) {
                   1085:             my $lastchar = $chars_per_line - length($student) - 1;
                   1086:             if ($lastchar > 0) {
                   1087:                 $course = substr($course, 0, $lastchar);
                   1088:             } else {            # Nothing left of course:
                   1089:                 $course = '';
                   1090:             }
                   1091:         }
                   1092:         if (length($assignment) > $chars_per_line) {
                   1093:             $assignment = substr($assignment, 0, $chars_per_line);
                   1094:         }
1.537     foxr     1095: 
1.627.2.1  raeburn  1096:         $format =  "\\textbf{$student} $course \\hfill \\thepage \\\\ \\textit{$assignment}";
1.486     foxr     1097: 
1.627.2.1  raeburn  1098:     } else {
                   1099:         # An open question is how to handle long user formatted page headers...
                   1100:         # A possible future is to support e.g. %na so that the user can control
                   1101:         # the truncation of the elements that can appear in the header.
                   1102:         #
                   1103:         $format =  &printf_style_subst("a", $format, $assignment);
                   1104:         $format =  &printf_style_subst("c", $format, $course);
                   1105:         $format =  &printf_style_subst("n", $format, $student);
                   1106: 
                   1107:         # If the user put %'s in the format string, they must be escaped
                   1108:         # to \% else LaTeX will think they are comments and terminate
                   1109:         # the line.. which is bad!!!
1.486     foxr     1110: 
1.627.2.1  raeburn  1111:     }
1.489     foxr     1112: 
1.538     onken    1113:     # If the user has role author, $course and $assignment are empty so
                   1114:     # there is '\\ \\ ' in the page header. That's cause a error in LaTeX
                   1115:     if($format =~ /\\\\\s\\\\\s/) {
                   1116:         #TODO find sensible caption for page header
1.627.2.6  raeburn  1117:         my $testPrintout = '\\\\'.&mt('Authoring Space').' \\\\'.&mt('Test-Printout ');
1.538     onken    1118:         $format =~ s/\\\\\s\\\\\s/$testPrintout/;
                   1119:     }
1.627.2.38! raeburn  1120: 
        !          1121:     return $format;
        !          1122: 
1.565     foxr     1123:     #
                   1124:     #  We're going to trust LaTeX to break lines appropriately, but
                   1125:     #  we'll truncate anything that's more than 3 lines worth of
                   1126:     # text.  This is also assuming (which will probably end badly)
                   1127:     # nobody's going to embed LaTeX control sequences in the title
                   1128:     # header or rather that those control sequences won't get broken
                   1129:     # by the stuff below.
                   1130:     #
                   1131:     my $total_length = 3*$chars_per_line;
                   1132:     if (length($format) > $total_length) {
                   1133: 	$format = substr($format, 0, $total_length);
                   1134:     }
                   1135: 
1.454     foxr     1136: 
                   1137:     return $format;
1.627.2.38! raeburn  1138: 
1.454     foxr     1139: }
1.397     albertel 1140: 
1.385     foxr     1141: #
                   1142: #   Convert a numeric code to letters
                   1143: #
                   1144: sub num_to_letters {
                   1145:     my ($num) = @_;
                   1146:     my @nums= split('',$num);
                   1147:     my @num_to_let=('A'..'Z');
                   1148:     my $word;
                   1149:     foreach my $digit (@nums) { $word.=$num_to_let[$digit]; }
                   1150:     return $word;
                   1151: }
                   1152: #   Convert a letter code to numeric.
                   1153: #
                   1154: sub letters_to_num {
                   1155:     my ($letters) = @_;
                   1156:     my @letters = split('', uc($letters));
1.490     foxr     1157:    my %substitution;
1.385     foxr     1158:     my $digit = 0;
                   1159:     foreach my $letter ('A'..'J') {
                   1160: 	$substitution{$letter} = $digit;
                   1161: 	$digit++;
                   1162:     }
                   1163:     #  The substitution is done as below to preserve leading
                   1164:     #  zeroes which are needed to keep the code size exact
                   1165:     #
                   1166:     my $result ="";
                   1167:     foreach my $letter (@letters) {
                   1168: 	$result.=$substitution{$letter};
                   1169:     }
                   1170:     return $result;
                   1171: }
                   1172: 
1.383     foxr     1173: #  Determine if a code is a valid numeric code.  Valid
                   1174: #  numeric codes must be comprised entirely of digits and
1.384     albertel 1175: #  have a correct number of digits.
1.383     foxr     1176: #
                   1177: #  Parameters:
                   1178: #     value      - proposed code value.
1.384     albertel 1179: #     num_digits - Number of digits required.
1.383     foxr     1180: #
                   1181: sub is_valid_numeric_code {
1.384     albertel 1182:     my ($value, $num_digits) = @_;
1.383     foxr     1183:     #   Remove leading/trailing whitespace;
1.387     foxr     1184:     $value =~ s/^\s*//g;
                   1185:     $value =~ s/\s*$//g;
1.627.2.38! raeburn  1186: 
1.383     foxr     1187:     #  All digits?
1.387     foxr     1188:     if ($value !~ /^[0-9]+$/) {
1.383     foxr     1189: 	return "Numeric code $value has invalid characters - must only be digits";
                   1190:     }
1.384     albertel 1191:     if (length($value) != $num_digits) {
                   1192: 	return "Numeric code $value incorrect number of digits (correct = $num_digits)";
                   1193:     }
1.385     foxr     1194:     return undef;
1.383     foxr     1195: }
                   1196: #   Determines if a code is a valid alhpa code.  Alpha codes
                   1197: #   are ciphers that map  [A-J,a-j] -> 0..9 0..9.
1.384     albertel 1198: #   They also have a correct digit count.
1.383     foxr     1199: # Parameters:
                   1200: #     value          - Proposed code value.
1.384     albertel 1201: #     num_letters    - correct number of letters.
1.383     foxr     1202: # Note:
                   1203: #    leading and trailing whitespace are ignored.
                   1204: #
                   1205: sub is_valid_alpha_code {
1.384     albertel 1206:     my ($value, $num_letters) = @_;
1.627.2.38! raeburn  1207: 
1.383     foxr     1208:      # strip leading and trailing spaces.
                   1209: 
                   1210:     $value =~ s/^\s*//g;
                   1211:     $value =~ s/\s*$//g;
                   1212: 
                   1213:     #  All alphas in the right range?
1.384     albertel 1214:     if ($value !~ /^[A-J,a-j]+$/) {
1.383     foxr     1215: 	return "Invalid letter code $value must only contain A-J";
                   1216:     }
1.384     albertel 1217:     if (length($value) != $num_letters) {
                   1218: 	return "Letter code $value has incorrect number of letters (correct = $num_letters)";
                   1219:     }
1.385     foxr     1220:     return undef;
1.383     foxr     1221: }
                   1222: 
1.382     foxr     1223: #   Determine if a code entered by the user in a helper is valid.
                   1224: #   valid depends on the code type and the type of code selected.
1.627.2.38! raeburn  1225: #   The type of code selected can either be numeric or
1.382     foxr     1226: #   Alphabetic.  If alphabetic, the code, in fact is a simple
                   1227: #   substitution cipher for the actual numeric code: 0->A, 1->B ...
                   1228: #   We'll be nice and be case insensitive for alpha codes.
                   1229: # Parameters:
                   1230: #    code_value    - the value of the code the user typed in.
                   1231: #    code_option   - The code type selected from the set in the scantron format
                   1232: #                    table.
                   1233: # Returns:
                   1234: #    undef         - The code is valid.
                   1235: #    other         - An error message indicating what's wrong.
                   1236: #
                   1237: sub is_code_valid {
                   1238:     my ($code_value, $code_option) = @_;
1.383     foxr     1239:     my ($code_type, $code_length) = ('letter', 6);	# defaults.
1.627.2.29  raeburn  1240:     my @lines = &Apache::lonnet::get_scantronformat_file();
1.542     raeburn  1241:     foreach my $line (@lines) {
1.627.2.8  raeburn  1242:         next if (($line =~ /^\#/) || ($line eq ''));
1.383     foxr     1243: 	my ($name, $type, $length) = (split(/:/, $line))[0,2,4];
                   1244: 	if($name eq $code_option) {
                   1245: 	    $code_length = $length;
                   1246: 	    if($type eq 'number') {
                   1247: 		$code_type = 'number';
                   1248: 	    }
                   1249: 	}
                   1250:     }
                   1251:     my $valid;
                   1252:     if ($code_type eq 'number') {
1.385     foxr     1253: 	return &is_valid_numeric_code($code_value, $code_length);
1.383     foxr     1254:     } else {
1.385     foxr     1255: 	return &is_valid_alpha_code($code_value, $code_length);
1.383     foxr     1256:     }
1.382     foxr     1257: 
                   1258: }
1.618     foxr     1259: #
                   1260: # Compare two students by section (Used to sort by section).
                   1261: #
1.627.2.38! raeburn  1262: #  Implicit inputs,
1.618     foxr     1263: #    $a - The first one
                   1264: #    $b - The second one.
                   1265: #
                   1266: #  Returns:
                   1267: #     a-section cmp b-section
                   1268: #
                   1269: sub compare_sections {
                   1270:     my ($u1, $d1, $s1, $n1, $stat1) = split(/:/, $a);
                   1271:     my ($u2, $d2, $s2, $n2, $stat2) = split(/:/, $b);
                   1272: 
                   1273:     return $s1 cmp $s2;
                   1274: }
1.382     foxr     1275: 
1.341     foxr     1276: #   Compare two students by name.  The students are in the form
                   1277: #   returned by the helper:
                   1278: #      user:domain:section:last,   first:status
                   1279: #   This is a helper function for the perl sort built-in  therefore:
                   1280: # Implicit Inputs:
                   1281: #    $a     - The first element to compare (global)
                   1282: #    $b     - The second element to compare (global)
                   1283: # Returns:
                   1284: #   -1   - $a < $b
                   1285: #    0   - $a == $b
                   1286: #   +1   - $a > $b
                   1287: #   Note that the initial comparison is done on the last names with the
                   1288: #   first names only used to break the tie.
                   1289: #
                   1290: #
                   1291: sub compare_names {
                   1292:     #  First split the names up into the primary fields.
                   1293: 
                   1294:     my ($u1, $d1, $s1, $n1, $stat1) = split(/:/, $a);
                   1295:     my ($u2, $d2, $s2, $n2, $stat2) = split(/:/, $b);
                   1296: 
                   1297:     # Now split the last name and first name of each n:
                   1298:     #
                   1299: 
                   1300:     my ($l1,$f1) = split(/,/, $n1);
                   1301:     my ($l2,$f2) = split(/,/, $n2);
                   1302: 
                   1303:     # We don't bother to remove the leading/trailing whitespace from the
                   1304:     # firstname, unless the last names compare identical.
                   1305: 
                   1306:     if($l1 lt $l2) {
                   1307: 	return -1;
                   1308:     }
                   1309:     if($l1 gt $l2) {
                   1310: 	return  1;
                   1311:     }
                   1312: 
                   1313:     # Break the tie on the first name, but there are leading (possibly trailing
1.627.2.38! raeburn  1314:     # whitespaces to get rid of first)
1.341     foxr     1315:     #
                   1316:     $f1 =~ s/^\s+//;		# Remove leading...
                   1317:     $f1 =~ s/\s+$//;		# Trailing spaces from first 1...
1.627.2.38! raeburn  1318: 
1.341     foxr     1319:     $f2 =~ s/^\s+//;
                   1320:     $f2 =~ s/\s+$//;		# And the same for first 2...
                   1321: 
                   1322:     if($f1 lt $f2) {
                   1323: 	return -1;
                   1324:     }
                   1325:     if($f1 gt $f2) {
                   1326: 	return 1;
                   1327:     }
1.627.2.38! raeburn  1328: 
1.341     foxr     1329:     #  Must be the same name.
                   1330: 
                   1331:     return 0;
                   1332: }
                   1333: 
1.71      sakharuk 1334: sub latex_header_footer_remove {
                   1335:     my $text = shift;
1.627.2.17  raeburn  1336:     $text =~ s/\\end\{document}//;
                   1337:     $text =~ s/\\documentclass([^&]*)\\begin\{document}//;
1.71      sakharuk 1338:     return $text;
                   1339: }
1.423     foxr     1340: #
1.627.2.38! raeburn  1341: #  If necessary, encapsulate text inside
1.423     foxr     1342: #  a minipage env.
                   1343: #  necessity is determined by the problem_split param.
                   1344: #
                   1345: sub encapsulate_minipage {
1.627.2.36  raeburn  1346:     my ($text,$problem_split) = @_;
                   1347:     if (!($problem_split =~ /yes/i)) {
1.423     foxr     1348: 	$text = '\begin{minipage}{\textwidth}'.$text.'\end{minipage}';
                   1349:     }
                   1350:     return $text;
                   1351: }
1.429     foxr     1352: #
                   1353: #  The NUMBER_TO_PRINT and SPLIT_PDFS
                   1354: #  variables interact, this sub looks at these two parameters
                   1355: #  and comes up with a final value for NUMBER_TO_PRINT which can be:
                   1356: #     all     - if SPLIT_PDFS eq 'all'.
                   1357: #     1       - if SPLIT_PDFS eq 'oneper'
                   1358: #     section - if SPLIT_PDFS eq 'sections'
                   1359: #     <unchanged> - if SPLIT_PDFS eq 'usenumber'
                   1360: #
                   1361: sub adjust_number_to_print {
                   1362:     my $helper = shift;
1.71      sakharuk 1363: 
1.429     foxr     1364:     my $split_pdf = $helper->{'VARS'}->{'SPLIT_PDFS'};
1.627.2.38! raeburn  1365: 
1.429     foxr     1366:     if ($split_pdf eq 'all') {
                   1367: 	$helper->{'VARS'}->{'NUMBER_TO_PRINT'} = 'all';
                   1368:     } elsif ($split_pdf eq 'oneper') {
                   1369: 	$helper->{'VARS'}->{'NUMBER_TO_PRINT'} = 1;
                   1370:     } elsif ($split_pdf eq 'sections') {
                   1371: 	$helper->{'VARS'}->{'NUMBER_TO_PRINT'} = 'section';
                   1372:     } elsif ($split_pdf eq 'usenumber') {
                   1373: 	#  Unmodified.
                   1374:     } else {
                   1375: 	# Error!!!!
1.536     foxr     1376: 	
                   1377: 	croak "bad SPLIT_PDFS: $split_pdf in lonprintout::adjust_number_to_print";
1.429     foxr     1378: 
                   1379:     }
                   1380: }
1.71      sakharuk 1381: 
1.531     foxr     1382: 
1.37      sakharuk 1383: sub character_chart {
1.531     foxr     1384:     my $result = shift;
                   1385:     return  &Apache::entities::replace_entities($result);
                   1386: }
                   1387: 
                   1388: sub old_character_chart {
1.37      sakharuk 1389:     my $result = shift;	
1.116     sakharuk 1390:     $result =~ s/&\#0?0?(7|9);//g;
                   1391:     $result =~ s/&\#0?(10|13);//g;
                   1392:     $result =~ s/&\#0?32;/ /g;
                   1393:     $result =~ s/&\#0?33;/!/g;
                   1394:     $result =~ s/&(\#0?34|quot);/\"/g;
                   1395:     $result =~ s/&\#0?35;/\\\#/g;
                   1396:     $result =~ s/&\#0?36;/\\\$/g;
1.627.2.38! raeburn  1397:     $result =~ s/&\#0?37;/\\%/g;
        !          1398:     $result =~ s/&(\#0?38|amp);/\\&/g;
1.116     sakharuk 1399:     $result =~ s/&\#(0?39|146);/\'/g;
                   1400:     $result =~ s/&\#0?40;/(/g;
                   1401:     $result =~ s/&\#0?41;/)/g;
                   1402:     $result =~ s/&\#0?42;/\*/g;
                   1403:     $result =~ s/&\#0?43;/\+/g;
                   1404:     $result =~ s/&\#(0?44|130);/,/g;
                   1405:     $result =~ s/&\#0?45;/-/g;
                   1406:     $result =~ s/&\#0?46;/\./g;
                   1407:     $result =~ s/&\#0?47;/\//g;
                   1408:     $result =~ s/&\#0?48;/0/g;
                   1409:     $result =~ s/&\#0?49;/1/g;
                   1410:     $result =~ s/&\#0?50;/2/g;
                   1411:     $result =~ s/&\#0?51;/3/g;
                   1412:     $result =~ s/&\#0?52;/4/g;
                   1413:     $result =~ s/&\#0?53;/5/g;
                   1414:     $result =~ s/&\#0?54;/6/g;
                   1415:     $result =~ s/&\#0?55;/7/g;
                   1416:     $result =~ s/&\#0?56;/8/g;
                   1417:     $result =~ s/&\#0?57;/9/g;
1.269     albertel 1418:     $result =~ s/&\#0?58;/:/g;
1.116     sakharuk 1419:     $result =~ s/&\#0?59;/;/g;
                   1420:     $result =~ s/&(\#0?60|lt|\#139);/\$<\$/g;
1.281     sakharuk 1421:     $result =~ s/&\#0?61;/\\ensuremath\{=\}/g;
                   1422:     $result =~ s/&(\#0?62|gt|\#155);/\\ensuremath\{>\}/g;
1.116     sakharuk 1423:     $result =~ s/&\#0?63;/\?/g;
                   1424:     $result =~ s/&\#0?65;/A/g;
                   1425:     $result =~ s/&\#0?66;/B/g;
                   1426:     $result =~ s/&\#0?67;/C/g;
                   1427:     $result =~ s/&\#0?68;/D/g;
                   1428:     $result =~ s/&\#0?69;/E/g;
                   1429:     $result =~ s/&\#0?70;/F/g;
                   1430:     $result =~ s/&\#0?71;/G/g;
                   1431:     $result =~ s/&\#0?72;/H/g;
                   1432:     $result =~ s/&\#0?73;/I/g;
                   1433:     $result =~ s/&\#0?74;/J/g;
                   1434:     $result =~ s/&\#0?75;/K/g;
                   1435:     $result =~ s/&\#0?76;/L/g;
                   1436:     $result =~ s/&\#0?77;/M/g;
                   1437:     $result =~ s/&\#0?78;/N/g;
                   1438:     $result =~ s/&\#0?79;/O/g;
                   1439:     $result =~ s/&\#0?80;/P/g;
                   1440:     $result =~ s/&\#0?81;/Q/g;
                   1441:     $result =~ s/&\#0?82;/R/g;
                   1442:     $result =~ s/&\#0?83;/S/g;
                   1443:     $result =~ s/&\#0?84;/T/g;
                   1444:     $result =~ s/&\#0?85;/U/g;
                   1445:     $result =~ s/&\#0?86;/V/g;
                   1446:     $result =~ s/&\#0?87;/W/g;
                   1447:     $result =~ s/&\#0?88;/X/g;
                   1448:     $result =~ s/&\#0?89;/Y/g;
                   1449:     $result =~ s/&\#0?90;/Z/g;
                   1450:     $result =~ s/&\#0?91;/[/g;
1.281     sakharuk 1451:     $result =~ s/&\#0?92;/\\ensuremath\{\\setminus\}/g;
1.116     sakharuk 1452:     $result =~ s/&\#0?93;/]/g;
1.281     sakharuk 1453:     $result =~ s/&\#(0?94|136);/\\ensuremath\{\\wedge\}/g;
1.116     sakharuk 1454:     $result =~ s/&\#(0?95|138|154);/\\underline{\\makebox[2mm]{\\strut}}/g;
                   1455:     $result =~ s/&\#(0?96|145);/\`/g;
                   1456:     $result =~ s/&\#0?97;/a/g;
                   1457:     $result =~ s/&\#0?98;/b/g;
                   1458:     $result =~ s/&\#0?99;/c/g;
                   1459:     $result =~ s/&\#100;/d/g;
                   1460:     $result =~ s/&\#101;/e/g;
                   1461:     $result =~ s/&\#102;/f/g;
                   1462:     $result =~ s/&\#103;/g/g;
                   1463:     $result =~ s/&\#104;/h/g;
                   1464:     $result =~ s/&\#105;/i/g;
                   1465:     $result =~ s/&\#106;/j/g;
                   1466:     $result =~ s/&\#107;/k/g;
                   1467:     $result =~ s/&\#108;/l/g;
                   1468:     $result =~ s/&\#109;/m/g;
                   1469:     $result =~ s/&\#110;/n/g;
                   1470:     $result =~ s/&\#111;/o/g;
                   1471:     $result =~ s/&\#112;/p/g;
                   1472:     $result =~ s/&\#113;/q/g;
                   1473:     $result =~ s/&\#114;/r/g;
                   1474:     $result =~ s/&\#115;/s/g;
                   1475:     $result =~ s/&\#116;/t/g;
                   1476:     $result =~ s/&\#117;/u/g;
                   1477:     $result =~ s/&\#118;/v/g;
                   1478:     $result =~ s/&\#119;/w/g;
                   1479:     $result =~ s/&\#120;/x/g;
                   1480:     $result =~ s/&\#121;/y/g;
                   1481:     $result =~ s/&\#122;/z/g;
                   1482:     $result =~ s/&\#123;/\\{/g;
                   1483:     $result =~ s/&\#124;/\|/g;
                   1484:     $result =~ s/&\#125;/\\}/g;
                   1485:     $result =~ s/&\#126;/\~/g;
                   1486:     $result =~ s/&\#131;/\\textflorin /g;
                   1487:     $result =~ s/&\#132;/\"/g;
1.281     sakharuk 1488:     $result =~ s/&\#133;/\\ensuremath\{\\ldots\}/g;
                   1489:     $result =~ s/&\#134;/\\ensuremath\{\\dagger\}/g;
                   1490:     $result =~ s/&\#135;/\\ensuremath\{\\ddagger\}/g;
1.116     sakharuk 1491:     $result =~ s/&\#137;/\\textperthousand /g;
                   1492:     $result =~ s/&\#140;/{\\OE}/g;
                   1493:     $result =~ s/&\#147;/\`\`/g;
                   1494:     $result =~ s/&\#148;/\'\'/g;
1.281     sakharuk 1495:     $result =~ s/&\#149;/\\ensuremath\{\\bullet\}/g;
1.494     albertel 1496:     $result =~ s/&(\#150|\#8211);/--/g;
1.116     sakharuk 1497:     $result =~ s/&\#151;/---/g;
1.281     sakharuk 1498:     $result =~ s/&\#152;/\\ensuremath\{\\sim\}/g;
1.116     sakharuk 1499:     $result =~ s/&\#153;/\\texttrademark /g;
                   1500:     $result =~ s/&\#156;/\\oe/g;
                   1501:     $result =~ s/&\#159;/\\\"Y/g;
1.283     albertel 1502:     $result =~ s/&(\#160|nbsp);/~/g;
1.116     sakharuk 1503:     $result =~ s/&(\#161|iexcl);/!\`/g;
                   1504:     $result =~ s/&(\#162|cent);/\\textcent /g;
1.627.2.38! raeburn  1505:     $result =~ s/&(\#163|pound);/\\pounds /g;
1.116     sakharuk 1506:     $result =~ s/&(\#164|curren);/\\textcurrency /g;
                   1507:     $result =~ s/&(\#165|yen);/\\textyen /g;
                   1508:     $result =~ s/&(\#166|brvbar);/\\textbrokenbar /g;
                   1509:     $result =~ s/&(\#167|sect);/\\textsection /g;
1.530     foxr     1510:     $result =~ s/&(\#168|uml);/\\"\{\} /g;
1.116     sakharuk 1511:     $result =~ s/&(\#169|copy);/\\copyright /g;
                   1512:     $result =~ s/&(\#170|ordf);/\\textordfeminine /g;
1.281     sakharuk 1513:     $result =~ s/&(\#172|not);/\\ensuremath\{\\neg\}/g;
1.116     sakharuk 1514:     $result =~ s/&(\#173|shy);/ - /g;
                   1515:     $result =~ s/&(\#174|reg);/\\textregistered /g;
1.281     sakharuk 1516:     $result =~ s/&(\#175|macr);/\\ensuremath\{^{-}\}/g;
                   1517:     $result =~ s/&(\#176|deg);/\\ensuremath\{^{\\circ}\}/g;
                   1518:     $result =~ s/&(\#177|plusmn);/\\ensuremath\{\\pm\}/g;
                   1519:     $result =~ s/&(\#178|sup2);/\\ensuremath\{^2\}/g;
                   1520:     $result =~ s/&(\#179|sup3);/\\ensuremath\{^3\}/g;
1.530     foxr     1521:     $result =~ s/&(\#180|acute);/\\'\{\} /g;
1.281     sakharuk 1522:     $result =~ s/&(\#181|micro);/\\ensuremath\{\\mu\}/g;
1.116     sakharuk 1523:     $result =~ s/&(\#182|para);/\\P/g;
1.281     sakharuk 1524:     $result =~ s/&(\#183|middot);/\\ensuremath\{\\cdot\}/g;
1.116     sakharuk 1525:     $result =~ s/&(\#184|cedil);/\\c{\\strut}/g;
1.281     sakharuk 1526:     $result =~ s/&(\#185|sup1);/\\ensuremath\{^1\}/g;
1.116     sakharuk 1527:     $result =~ s/&(\#186|ordm);/\\textordmasculine /g;
                   1528:     $result =~ s/&(\#188|frac14);/\\textonequarter /g;
                   1529:     $result =~ s/&(\#189|frac12);/\\textonehalf /g;
                   1530:     $result =~ s/&(\#190|frac34);/\\textthreequarters /g;
1.627.2.38! raeburn  1531:     $result =~ s/&(\#191|iquest);/?\`/g;
        !          1532:     $result =~ s/&(\#192|Agrave);/\\\`{A}/g;
        !          1533:     $result =~ s/&(\#193|Aacute);/\\\'{A}/g;
1.116     sakharuk 1534:     $result =~ s/&(\#194|Acirc);/\\^{A}/g;
                   1535:     $result =~ s/&(\#195|Atilde);/\\~{A}/g;
1.627.2.38! raeburn  1536:     $result =~ s/&(\#196|Auml);/\\\"{A}/g;
1.116     sakharuk 1537:     $result =~ s/&(\#197|Aring);/{\\AA}/g;
                   1538:     $result =~ s/&(\#198|AElig);/{\\AE}/g;
                   1539:     $result =~ s/&(\#199|Ccedil);/\\c{c}/g;
1.627.2.38! raeburn  1540:     $result =~ s/&(\#200|Egrave);/\\\`{E}/g;
        !          1541:     $result =~ s/&(\#201|Eacute);/\\\'{E}/g;
1.116     sakharuk 1542:     $result =~ s/&(\#202|Ecirc);/\\^{E}/g;
                   1543:     $result =~ s/&(\#203|Euml);/\\\"{E}/g;
                   1544:     $result =~ s/&(\#204|Igrave);/\\\`{I}/g;
1.627.2.38! raeburn  1545:     $result =~ s/&(\#205|Iacute);/\\\'{I}/g;
1.116     sakharuk 1546:     $result =~ s/&(\#206|Icirc);/\\^{I}/g;
1.627.2.38! raeburn  1547:     $result =~ s/&(\#207|Iuml);/\\\"{I}/g;
1.116     sakharuk 1548:     $result =~ s/&(\#209|Ntilde);/\\~{N}/g;
                   1549:     $result =~ s/&(\#210|Ograve);/\\\`{O}/g;
                   1550:     $result =~ s/&(\#211|Oacute);/\\\'{O}/g;
                   1551:     $result =~ s/&(\#212|Ocirc);/\\^{O}/g;
                   1552:     $result =~ s/&(\#213|Otilde);/\\~{O}/g;
1.627.2.38! raeburn  1553:     $result =~ s/&(\#214|Ouml);/\\\"{O}/g;
1.281     sakharuk 1554:     $result =~ s/&(\#215|times);/\\ensuremath\{\\times\}/g;
1.116     sakharuk 1555:     $result =~ s/&(\#216|Oslash);/{\\O}/g;
1.627.2.38! raeburn  1556:     $result =~ s/&(\#217|Ugrave);/\\\`{U}/g;
1.116     sakharuk 1557:     $result =~ s/&(\#218|Uacute);/\\\'{U}/g;
                   1558:     $result =~ s/&(\#219|Ucirc);/\\^{U}/g;
                   1559:     $result =~ s/&(\#220|Uuml);/\\\"{U}/g;
                   1560:     $result =~ s/&(\#221|Yacute);/\\\'{Y}/g;
1.329     sakharuk 1561:     $result =~ s/&(\#223|szlig);/{\\ss}/g;
1.116     sakharuk 1562:     $result =~ s/&(\#224|agrave);/\\\`{a}/g;
                   1563:     $result =~ s/&(\#225|aacute);/\\\'{a}/g;
                   1564:     $result =~ s/&(\#226|acirc);/\\^{a}/g;
                   1565:     $result =~ s/&(\#227|atilde);/\\~{a}/g;
                   1566:     $result =~ s/&(\#228|auml);/\\\"{a}/g;
                   1567:     $result =~ s/&(\#229|aring);/{\\aa}/g;
                   1568:     $result =~ s/&(\#230|aelig);/{\\ae}/g;
                   1569:     $result =~ s/&(\#231|ccedil);/\\c{c}/g;
                   1570:     $result =~ s/&(\#232|egrave);/\\\`{e}/g;
                   1571:     $result =~ s/&(\#233|eacute);/\\\'{e}/g;
                   1572:     $result =~ s/&(\#234|ecirc);/\\^{e}/g;
                   1573:     $result =~ s/&(\#235|euml);/\\\"{e}/g;
                   1574:     $result =~ s/&(\#236|igrave);/\\\`{i}/g;
                   1575:     $result =~ s/&(\#237|iacute);/\\\'{i}/g;
                   1576:     $result =~ s/&(\#238|icirc);/\\^{i}/g;
                   1577:     $result =~ s/&(\#239|iuml);/\\\"{i}/g;
1.281     sakharuk 1578:     $result =~ s/&(\#240|eth);/\\ensuremath\{\\partial\}/g;
1.116     sakharuk 1579:     $result =~ s/&(\#241|ntilde);/\\~{n}/g;
                   1580:     $result =~ s/&(\#242|ograve);/\\\`{o}/g;
                   1581:     $result =~ s/&(\#243|oacute);/\\\'{o}/g;
                   1582:     $result =~ s/&(\#244|ocirc);/\\^{o}/g;
                   1583:     $result =~ s/&(\#245|otilde);/\\~{o}/g;
                   1584:     $result =~ s/&(\#246|ouml);/\\\"{o}/g;
1.281     sakharuk 1585:     $result =~ s/&(\#247|divide);/\\ensuremath\{\\div\}/g;
1.116     sakharuk 1586:     $result =~ s/&(\#248|oslash);/{\\o}/g;
1.627.2.38! raeburn  1587:     $result =~ s/&(\#249|ugrave);/\\\`{u}/g;
1.116     sakharuk 1588:     $result =~ s/&(\#250|uacute);/\\\'{u}/g;
                   1589:     $result =~ s/&(\#251|ucirc);/\\^{u}/g;
                   1590:     $result =~ s/&(\#252|uuml);/\\\"{u}/g;
                   1591:     $result =~ s/&(\#253|yacute);/\\\'{y}/g;
                   1592:     $result =~ s/&(\#255|yuml);/\\\"{y}/g;
1.399     albertel 1593:     $result =~ s/&\#295;/\\ensuremath\{\\hbar\}/g;
1.281     sakharuk 1594:     $result =~ s/&\#952;/\\ensuremath\{\\theta\}/g;
1.117     sakharuk 1595: #Greek Alphabet
1.281     sakharuk 1596:     $result =~ s/&(alpha|\#945);/\\ensuremath\{\\alpha\}/g;
                   1597:     $result =~ s/&(beta|\#946);/\\ensuremath\{\\beta\}/g;
                   1598:     $result =~ s/&(gamma|\#947);/\\ensuremath\{\\gamma\}/g;
                   1599:     $result =~ s/&(delta|\#948);/\\ensuremath\{\\delta\}/g;
                   1600:     $result =~ s/&(epsilon|\#949);/\\ensuremath\{\\epsilon\}/g;
                   1601:     $result =~ s/&(zeta|\#950);/\\ensuremath\{\\zeta\}/g;
                   1602:     $result =~ s/&(eta|\#951);/\\ensuremath\{\\eta\}/g;
                   1603:     $result =~ s/&(theta|\#952);/\\ensuremath\{\\theta\}/g;
                   1604:     $result =~ s/&(iota|\#953);/\\ensuremath\{\\iota\}/g;
                   1605:     $result =~ s/&(kappa|\#954);/\\ensuremath\{\\kappa\}/g;
                   1606:     $result =~ s/&(lambda|\#955);/\\ensuremath\{\\lambda\}/g;
                   1607:     $result =~ s/&(mu|\#956);/\\ensuremath\{\\mu\}/g;
                   1608:     $result =~ s/&(nu|\#957);/\\ensuremath\{\\nu\}/g;
                   1609:     $result =~ s/&(xi|\#958);/\\ensuremath\{\\xi\}/g;
1.199     sakharuk 1610:     $result =~ s/&(omicron|\#959);/o/g;
1.281     sakharuk 1611:     $result =~ s/&(pi|\#960);/\\ensuremath\{\\pi\}/g;
                   1612:     $result =~ s/&(rho|\#961);/\\ensuremath\{\\rho\}/g;
                   1613:     $result =~ s/&(sigma|\#963);/\\ensuremath\{\\sigma\}/g;
                   1614:     $result =~ s/&(tau|\#964);/\\ensuremath\{\\tau\}/g;
                   1615:     $result =~ s/&(upsilon|\#965);/\\ensuremath\{\\upsilon\}/g;
                   1616:     $result =~ s/&(phi|\#966);/\\ensuremath\{\\phi\}/g;
                   1617:     $result =~ s/&(chi|\#967);/\\ensuremath\{\\chi\}/g;
                   1618:     $result =~ s/&(psi|\#968);/\\ensuremath\{\\psi\}/g;
                   1619:     $result =~ s/&(omega|\#969);/\\ensuremath\{\\omega\}/g;
                   1620:     $result =~ s/&(thetasym|\#977);/\\ensuremath\{\\vartheta\}/g;
                   1621:     $result =~ s/&(piv|\#982);/\\ensuremath\{\\varpi\}/g;
1.199     sakharuk 1622:     $result =~ s/&(Alpha|\#913);/A/g;
                   1623:     $result =~ s/&(Beta|\#914);/B/g;
1.281     sakharuk 1624:     $result =~ s/&(Gamma|\#915);/\\ensuremath\{\\Gamma\}/g;
                   1625:     $result =~ s/&(Delta|\#916);/\\ensuremath\{\\Delta\}/g;
1.199     sakharuk 1626:     $result =~ s/&(Epsilon|\#917);/E/g;
                   1627:     $result =~ s/&(Zeta|\#918);/Z/g;
                   1628:     $result =~ s/&(Eta|\#919);/H/g;
1.281     sakharuk 1629:     $result =~ s/&(Theta|\#920);/\\ensuremath\{\\Theta\}/g;
1.199     sakharuk 1630:     $result =~ s/&(Iota|\#921);/I/g;
                   1631:     $result =~ s/&(Kappa|\#922);/K/g;
1.281     sakharuk 1632:     $result =~ s/&(Lambda|\#923);/\\ensuremath\{\\Lambda\}/g;
1.199     sakharuk 1633:     $result =~ s/&(Mu|\#924);/M/g;
                   1634:     $result =~ s/&(Nu|\#925);/N/g;
1.281     sakharuk 1635:     $result =~ s/&(Xi|\#926);/\\ensuremath\{\\Xi\}/g;
1.199     sakharuk 1636:     $result =~ s/&(Omicron|\#927);/O/g;
1.281     sakharuk 1637:     $result =~ s/&(Pi|\#928);/\\ensuremath\{\\Pi\}/g;
1.199     sakharuk 1638:     $result =~ s/&(Rho|\#929);/P/g;
1.281     sakharuk 1639:     $result =~ s/&(Sigma|\#931);/\\ensuremath\{\\Sigma\}/g;
1.199     sakharuk 1640:     $result =~ s/&(Tau|\#932);/T/g;
1.281     sakharuk 1641:     $result =~ s/&(Upsilon|\#933);/\\ensuremath\{\\Upsilon\}/g;
                   1642:     $result =~ s/&(Phi|\#934);/\\ensuremath\{\\Phi\}/g;
1.199     sakharuk 1643:     $result =~ s/&(Chi|\#935);/X/g;
1.281     sakharuk 1644:     $result =~ s/&(Psi|\#936);/\\ensuremath\{\\Psi\}/g;
                   1645:     $result =~ s/&(Omega|\#937);/\\ensuremath\{\\Omega\}/g;
1.199     sakharuk 1646: #Arrows (extended HTML 4.01)
1.281     sakharuk 1647:     $result =~ s/&(larr|\#8592);/\\ensuremath\{\\leftarrow\}/g;
                   1648:     $result =~ s/&(uarr|\#8593);/\\ensuremath\{\\uparrow\}/g;
                   1649:     $result =~ s/&(rarr|\#8594);/\\ensuremath\{\\rightarrow\}/g;
                   1650:     $result =~ s/&(darr|\#8595);/\\ensuremath\{\\downarrow\}/g;
                   1651:     $result =~ s/&(harr|\#8596);/\\ensuremath\{\\leftrightarrow\}/g;
                   1652:     $result =~ s/&(lArr|\#8656);/\\ensuremath\{\\Leftarrow\}/g;
                   1653:     $result =~ s/&(uArr|\#8657);/\\ensuremath\{\\Uparrow\}/g;
                   1654:     $result =~ s/&(rArr|\#8658);/\\ensuremath\{\\Rightarrow\}/g;
                   1655:     $result =~ s/&(dArr|\#8659);/\\ensuremath\{\\Downarrow\}/g;
                   1656:     $result =~ s/&(hArr|\#8660);/\\ensuremath\{\\Leftrightarrow\}/g;
1.199     sakharuk 1657: #Mathematical Operators (extended HTML 4.01)
1.281     sakharuk 1658:     $result =~ s/&(forall|\#8704);/\\ensuremath\{\\forall\}/g;
                   1659:     $result =~ s/&(part|\#8706);/\\ensuremath\{\\partial\}/g;
                   1660:     $result =~ s/&(exist|\#8707);/\\ensuremath\{\\exists\}/g;
                   1661:     $result =~ s/&(empty|\#8709);/\\ensuremath\{\\emptyset\}/g;
                   1662:     $result =~ s/&(nabla|\#8711);/\\ensuremath\{\\nabla\}/g;
                   1663:     $result =~ s/&(isin|\#8712);/\\ensuremath\{\\in\}/g;
                   1664:     $result =~ s/&(notin|\#8713);/\\ensuremath\{\\notin\}/g;
                   1665:     $result =~ s/&(ni|\#8715);/\\ensuremath\{\\ni\}/g;
                   1666:     $result =~ s/&(prod|\#8719);/\\ensuremath\{\\prod\}/g;
                   1667:     $result =~ s/&(sum|\#8721);/\\ensuremath\{\\sum\}/g;
                   1668:     $result =~ s/&(minus|\#8722);/\\ensuremath\{-\}/g;
1.390     albertel 1669:     $result =~ s/–/\\ensuremath\{-\}/g;
1.281     sakharuk 1670:     $result =~ s/&(lowast|\#8727);/\\ensuremath\{*\}/g;
                   1671:     $result =~ s/&(radic|\#8730);/\\ensuremath\{\\surd\}/g;
                   1672:     $result =~ s/&(prop|\#8733);/\\ensuremath\{\\propto\}/g;
                   1673:     $result =~ s/&(infin|\#8734);/\\ensuremath\{\\infty\}/g;
                   1674:     $result =~ s/&(ang|\#8736);/\\ensuremath\{\\angle\}/g;
                   1675:     $result =~ s/&(and|\#8743);/\\ensuremath\{\\wedge\}/g;
                   1676:     $result =~ s/&(or|\#8744);/\\ensuremath\{\\vee\}/g;
                   1677:     $result =~ s/&(cap|\#8745);/\\ensuremath\{\\cap\}/g;
                   1678:     $result =~ s/&(cup|\#8746);/\\ensuremath\{\\cup\}/g;
                   1679:     $result =~ s/&(int|\#8747);/\\ensuremath\{\\int\}/g;
                   1680:     $result =~ s/&(sim|\#8764);/\\ensuremath\{\\sim\}/g;
                   1681:     $result =~ s/&(cong|\#8773);/\\ensuremath\{\\cong\}/g;
                   1682:     $result =~ s/&(asymp|\#8776);/\\ensuremath\{\\approx\}/g;
                   1683:     $result =~ s/&(ne|\#8800);/\\ensuremath\{\\not=\}/g;
                   1684:     $result =~ s/&(equiv|\#8801);/\\ensuremath\{\\equiv\}/g;
                   1685:     $result =~ s/&(le|\#8804);/\\ensuremath\{\\leq\}/g;
                   1686:     $result =~ s/&(ge|\#8805);/\\ensuremath\{\\geq\}/g;
                   1687:     $result =~ s/&(sub|\#8834);/\\ensuremath\{\\subset\}/g;
                   1688:     $result =~ s/&(sup|\#8835);/\\ensuremath\{\\supset\}/g;
                   1689:     $result =~ s/&(nsub|\#8836);/\\ensuremath\{\\not\\subset\}/g;
                   1690:     $result =~ s/&(sube|\#8838);/\\ensuremath\{\\subseteq\}/g;
                   1691:     $result =~ s/&(supe|\#8839);/\\ensuremath\{\\supseteq\}/g;
                   1692:     $result =~ s/&(oplus|\#8853);/\\ensuremath\{\\oplus\}/g;
                   1693:     $result =~ s/&(otimes|\#8855);/\\ensuremath\{\\otimes\}/g;
                   1694:     $result =~ s/&(perp|\#8869);/\\ensuremath\{\\perp\}/g;
                   1695:     $result =~ s/&(sdot|\#8901);/\\ensuremath\{\\cdot\}/g;
1.199     sakharuk 1696: #Geometric Shapes (extended HTML 4.01)
1.281     sakharuk 1697:     $result =~ s/&(loz|\#9674);/\\ensuremath\{\\Diamond\}/g;
1.199     sakharuk 1698: #Miscellaneous Symbols (extended HTML 4.01)
1.281     sakharuk 1699:     $result =~ s/&(spades|\#9824);/\\ensuremath\{\\spadesuit\}/g;
                   1700:     $result =~ s/&(clubs|\#9827);/\\ensuremath\{\\clubsuit\}/g;
                   1701:     $result =~ s/&(hearts|\#9829);/\\ensuremath\{\\heartsuit\}/g;
                   1702:     $result =~ s/&(diams|\#9830);/\\ensuremath\{\\diamondsuit\}/g;
1.495     foxr     1703: #   Chemically useful 'things' contributed by Hon Kie (bug 4652).
1.515     foxr     1704: 
1.495     foxr     1705:     $result =~ s/&\#8636;/\\ensuremath\{\\leftharpoonup\}/g;
                   1706:     $result =~ s/&\#8637;/\\ensuremath\{\\leftharpoondown\}/g;
                   1707:     $result =~ s/&\#8640;/\\ensuremath\{\\rightharpoonup\}/g;
                   1708:     $result =~ s/&\#8641;/\\ensuremath\{\\rightharpoondown\}/g;
                   1709:     $result =~ s/&\#8652;/\\ensuremath\{\\rightleftharpoons\}/g;
                   1710:     $result =~ s/&\#8605;/\\ensuremath\{\\leadsto\}/g;
                   1711:     $result =~ s/&\#8617;/\\ensuremath\{\\hookleftarrow\}/g;
                   1712:     $result =~ s/&\#8618;/\\ensuremath\{\\hookrightarrow\}/g;
                   1713:     $result =~ s/&\#8614;/\\ensuremath\{\\mapsto\}/g;
                   1714:     $result =~ s/&\#8599;/\\ensuremath\{\\nearrow\}/g;
                   1715:     $result =~ s/&\#8600;/\\ensuremath\{\\searrow\}/g;
                   1716:     $result =~ s/&\#8601;/\\ensuremath\{\\swarrow\}/g;
                   1717:     $result =~ s/&\#8598;/\\ensuremath\{\\nwarrow\}/g;
1.513     foxr     1718: 
                   1719:     # Left/right quotations:
                   1720: 
                   1721:     $result =~ s/&(ldquo|#8220);/\`\`/g;
                   1722:     $result =~ s/&(rdquo|#8221);/\'\'/g;
                   1723: 
                   1724: 
1.559     foxr     1725: 
1.37      sakharuk 1726:     return $result;
                   1727: }
1.41      sakharuk 1728: 
                   1729: 
1.327     albertel 1730:                   #width, height, oddsidemargin, evensidemargin, topmargin
                   1731: my %page_formats=
                   1732:     ('letter' => {
                   1733: 	 'book' => {
1.627.2.38! raeburn  1734: 	     '1' => [ '7.1 in','9.7 in', '-0.57 in','-0.57 in','-0.5 in'],
        !          1735: 	     '2' => ['3.66 in','9.8 in', '-0.57 in','-0.57 in','-0.5 in']
1.327     albertel 1736: 	 },
                   1737: 	 'album' => {
1.627.2.38! raeburn  1738: 	     '1' => [ '8.8 in', '6.8 in','-0.55 in',  '-0.55 in','-0.5 in'],
        !          1739: 	     '2' => [ '4.8 in', '6.7 in','-0.5 in', '-1.0 in','3.0 in']
1.327     albertel 1740: 	 },
                   1741:      },
                   1742:      'legal' => {
                   1743: 	 'book' => {
                   1744: 	     '1' => ['7.1 in','13 in',,'-0.57 in','-0.57 in','-0.5 in'],
1.514     foxr     1745: 	     '2' => ['3.66 in','13 in','-0.57 in','-0.57 in','-0.5 in']
1.327     albertel 1746: 	 },
                   1747: 	 'album' => {
1.376     albertel 1748: 	     '1' => ['12 in','7.1 in',,'-0.57 in','-0.57 in','-0.5 in'],
1.627.2.38! raeburn  1749:              '2' => ['5.7 in','7.1 in','-1 in','-1 in','5 in']
1.327     albertel 1750:           },
                   1751:      },
                   1752:      'tabloid' => {
                   1753: 	 'book' => {
                   1754: 	     '1' => ['9.8 in','16 in','-0.57 in','-0.57 in','-0.5 in'],
                   1755: 	     '2' => ['4.9 in','16 in','-0.57 in','-0.57 in','-0.5 in']
                   1756: 	 },
                   1757: 	 'album' => {
1.376     albertel 1758: 	     '1' => ['16 in','9.8 in','-0.57 in','-0.57 in','-0.5 in'],
                   1759: 	     '2' => ['16 in','4.9 in','-0.57 in','-0.57 in','-0.5 in']
1.327     albertel 1760:           },
                   1761:      },
                   1762:      'executive' => {
                   1763: 	 'book' => {
                   1764: 	     '1' => ['6.8 in','9 in','-0.57 in','-0.57 in','1.2 in'],
                   1765: 	     '2' => ['3.1 in','9 in','-0.57 in','-0.57 in','1.2 in']
                   1766: 	 },
                   1767: 	 'album' => {
                   1768: 	     '1' => [],
                   1769: 	     '2' => []
                   1770:           },
                   1771:      },
                   1772:      'a2' => {
                   1773: 	 'book' => {
                   1774: 	     '1' => [],
                   1775: 	     '2' => []
                   1776: 	 },
                   1777: 	 'album' => {
                   1778: 	     '1' => [],
                   1779: 	     '2' => []
                   1780:           },
                   1781:      },
                   1782:      'a3' => {
                   1783: 	 'book' => {
                   1784: 	     '1' => [],
                   1785: 	     '2' => []
                   1786: 	 },
                   1787: 	 'album' => {
                   1788: 	     '1' => [],
                   1789: 	     '2' => []
                   1790:           },
                   1791:      },
                   1792:      'a4' => {
                   1793: 	 'book' => {
1.493     foxr     1794: 	     '1' => ['17.6 cm','27.2 cm','-1.397 cm','-2.11 cm','-1.27 cm'],
1.496     foxr     1795: 	     '2' => [ '9.1 cm','27.2 cm','-1.397 cm','-2.11 cm','-1.27 cm']
1.327     albertel 1796: 	 },
                   1797: 	 'album' => {
1.627.2.38! raeburn  1798: 	     '1' => ['24.0 cm','18.0 cm','-1.0 cm','-1.0 cm','-1.25 cm'],
        !          1799: 	     '2' => ['11.5 cm','18.0 cm','-0.7 cm','-1.7 cm','-1.25 cm']
1.327     albertel 1800: 	 },
                   1801:      },
                   1802:      'a5' => {
                   1803: 	 'book' => {
                   1804: 	     '1' => [],
                   1805: 	     '2' => []
                   1806: 	 },
                   1807: 	 'album' => {
                   1808: 	     '1' => [],
                   1809: 	     '2' => []
                   1810:           },
                   1811:      },
                   1812:      'a6' => {
                   1813: 	 'book' => {
                   1814: 	     '1' => [],
                   1815: 	     '2' => []
                   1816: 	 },
                   1817: 	 'album' => {
                   1818: 	     '1' => [],
                   1819: 	     '2' => []
                   1820:           },
                   1821:      },
                   1822:      );
                   1823: 
1.177     sakharuk 1824: sub page_format {
1.140     sakharuk 1825: #
1.326     sakharuk 1826: #Supported paper format: "Letter [8 1/2x11 in]",      "Legal [8 1/2x14 in]",
                   1827: #                        "Ledger/Tabloid [11x17 in]", "Executive [7 1/2x10 in]",
                   1828: #                        "A2 [420x594 mm]",           "A3 [297x420 mm]",
                   1829: #                        "A4 [210x297 mm]",           "A5 [148x210 mm]",
                   1830: #                        "A6 [105x148 mm]"
1.627.2.38! raeburn  1831: #
        !          1832:     my ($papersize,$layout,$numberofcolumns) = @_;
1.327     albertel 1833:     return @{$page_formats{$papersize}->{$layout}->{$numberofcolumns}};
1.140     sakharuk 1834: }
1.76      sakharuk 1835: 
                   1836: 
1.126     albertel 1837: sub get_name {
                   1838:     my ($uname,$udom)=@_;
1.373     albertel 1839:     if (!defined($uname)) { $uname=$env{'user.name'}; }
                   1840:     if (!defined($udom)) { $udom=$env{'user.domain'}; }
1.126     albertel 1841:     my $plainname=&Apache::loncommon::plainname($uname,$udom);
1.213     albertel 1842:     if ($plainname=~/^\s*$/) { $plainname=$uname.'@'.$udom; }
1.453     foxr     1843:    $plainname=&Apache::lonxml::latex_special_symbols($plainname,'header');
1.213     albertel 1844:     return $plainname;
1.126     albertel 1845: }
                   1846: 
1.213     albertel 1847: sub get_course {
                   1848:     my $courseidinfo;
1.373     albertel 1849:     if (defined($env{'request.course.id'})) {
1.439     www      1850: 	$courseidinfo = &Apache::lonxml::latex_special_symbols(&unescape($env{'course.'.$env{'request.course.id'}.'.description'}),'header');
1.627.2.38! raeburn  1851: 
1.213     albertel 1852:     }
                   1853:     return $courseidinfo;
                   1854: }
1.177     sakharuk 1855: 
1.76      sakharuk 1856: sub page_format_transformation {
1.627.2.38! raeburn  1857:     my ($papersize,$layout,$numberofcolumns,$choice,$text,$assignment,$tableofcontents,
        !          1858:         $indexlist,$selectionmade,$mostrecent) = @_;
1.202     sakharuk 1859:     my ($textwidth,$textheight,$oddoffset,$evenoffset,$topmargin);
1.454     foxr     1860: 
1.312     sakharuk 1861:     if ($selectionmade eq '4') {
1.502     foxr     1862: 	if ($choice eq 'all_problems') {
1.561     bisitz   1863:             $assignment=&mt('Problems from the Whole Course');
1.502     foxr     1864: 	} else {
1.561     bisitz   1865:             $assignment=&mt('Resources from the Whole Course');
1.502     foxr     1866: 	}
1.312     sakharuk 1867:     } else {
                   1868: 	$assignment=&Apache::lonxml::latex_special_symbols($assignment,'header');
                   1869:     }
1.261     sakharuk 1870:     ($textwidth,$textheight,$oddoffset,$evenoffset,$topmargin) = &page_format($papersize,$layout,$numberofcolumns,$topmargin);
1.454     foxr     1871: 
1.627.2.38! raeburn  1872:     my $name;
        !          1873:     if ($mostrecent ne '') {
        !          1874:         $name = $mostrecent;
        !          1875:     } else {
        !          1876:         $name = &get_name();
        !          1877:     }
1.213     albertel 1878:     my $courseidinfo = &get_course();
1.627.2.1  raeburn  1879:     if (defined($courseidinfo)) { $courseidinfo=' - '.$courseidinfo }
1.455     albertel 1880:     my $header_text  = $parmhash{'print_header_format'};
1.486     foxr     1881:     $header_text     = &format_page_header($textwidth, $header_text, $assignment,
1.455     albertel 1882: 					   $courseidinfo, $name);
1.319     sakharuk 1883:     my $topmargintoinsert = '';
                   1884:     if ($topmargin ne '0') {$topmargintoinsert='\setlength{\topmargin}{'.$topmargin.'}';}
1.325     sakharuk 1885:     my $fancypagestatement='';
                   1886:     if ($numberofcolumns eq '2') {
1.455     albertel 1887: 	$fancypagestatement="\\fancyhead{}\\fancyhead[LO]{$header_text}";
1.627.2.38! raeburn  1888: 	if ($parmhash{'print_header_format'} eq '') {
        !          1889: 	    $fancypagestatement .= "\\fancyhead[RE]{\\thepage \\\\[\\baselineskip]}";
        !          1890: 	}
1.325     sakharuk 1891:     } else {
1.455     albertel 1892: 	$fancypagestatement="\\rhead{}\\chead{}\\lhead{$header_text}";
1.325     sakharuk 1893:     }
1.627.2.38! raeburn  1894:     $fancypagestatement .= "\\fancyfoot{}";
        !          1895:     my ($paperwidth,$paperheight);
1.140     sakharuk 1896:     if ($layout eq 'album') {
1.627.2.17  raeburn  1897: 	    $text =~ s/\\begin\{document}/\\setlength{\\oddsidemargin}{$oddoffset}\\setlength{\\evensidemargin}{$evenoffset}$topmargintoinsert\n\\setlength{\\textwidth}{$textwidth}\\setlength{\\textheight}{$textheight}\\setlength{\\textfloatsep}{8pt plus 2\.0pt minus 4\.0pt}\n\\newlength{\\minipagewidth}\\setlength{\\minipagewidth}{\\textwidth\/\$number_of_columns-0\.2cm}\\usepackage{fancyhdr}\\addtolength{\\headheight}{\\baselineskip}\n\\pagestyle{fancy}$fancypagestatement\\usepackage{booktabs}\\begin{document}\\voffset=-0\.8 cm\\setcounter{page}{1}\n /;
1.627.2.38! raeburn  1898:         if ($papersize eq 'a4') {
        !          1899:             $paperwidth = '29.7cm';
        !          1900:             $paperheight = '21.0cm';
        !          1901:         } elsif ($numberofcolumns eq '1') {
        !          1902:             if ($papersize eq 'letter') {
        !          1903:                 $paperwidth = '11.0in';
        !          1904:                 $paperheight = '8.5in';
        !          1905:             } elsif ($papersize eq 'legal') {
        !          1906:                 $paperwidth = '14.0in';
        !          1907:                 $paperheight = '8.5in';
        !          1908:             }
        !          1909:         }
1.140     sakharuk 1910:     } elsif ($layout eq 'book') {
1.627.2.38! raeburn  1911: 	if ($choice ne 'All class print') {
1.627.2.17  raeburn  1912: 	    $text =~ s/\\begin\{document}/\\textheight $textheight\\oddsidemargin = $evenoffset\\evensidemargin = $evenoffset $topmargintoinsert\n\\textwidth= $textwidth\\newlength{\\minipagewidth}\\setlength{\\minipagewidth}{\\textwidth\/\$number_of_columns-0\.2cm}\n\\renewcommand{\\ref}{\\keephidden\}\\usepackage{fancyhdr}\\addtolength{\\headheight}{\\baselineskip}\\pagestyle{fancy}$fancypagestatement\\usepackage{booktabs}\\begin{document}\n\\voffset=-0\.8 cm\\setcounter{page}{1}\n/;
1.140     sakharuk 1913: 	} else {
1.627.2.17  raeburn  1914: 	    $text =~ s/\\pagestyle\{fancy}\\rhead\{}\\chead\{}\s*\\begin\{document}/\\textheight = $textheight\\oddsidemargin = $evenoffset\n\\evensidemargin = $evenoffset $topmargintoinsert\\textwidth= $textwidth\\newlength{\\minipagewidth}\n\\setlength{\\minipagewidth}{\\textwidth\/\$number_of_columns-0\.2cm}\\renewcommand{\\ref}{\\keephidden\}\\pagestyle{fancy}\\rhead{}\\chead{}\\usepackage{booktabs}\\begin{document}\\voffset=-0\.8cm\n\\setcounter{page}{1}  \\vskip 5 mm\n /;
1.319     sakharuk 1915: 	}
1.627.2.38! raeburn  1916:         if ($papersize eq 'a4') {
        !          1917:             $paperwidth = '21.0cm';
        !          1918:             $paperheight = '29.7cm';
        !          1919:         } elsif ($papersize eq 'letter') {
        !          1920:             $paperwidth = '8.5in';
        !          1921:             $paperheight = '11.0in';
        !          1922:          } elsif ($papersize eq 'legal') {
        !          1923:             $paperwidth = '8.5in';
        !          1924:             $paperheight = '14.0in';
        !          1925:         }
        !          1926:     }
        !          1927:     if ($paperwidth ne '' && $paperheight ne '') {
        !          1928:         my $papersize_text;
        !          1929:         if ($perm{'pav'}) {
        !          1930:             $papersize_text = '\\special{papersize='.$paperwidth.','.$paperheight.'}';
        !          1931:         } else {
        !          1932:             $papersize_text = '\special{papersize='.$paperwidth.','.$paperheight.'}';
        !          1933:         }
        !          1934:         $text =~ s/(\\begin\{document})/$1$papersize_text/;
1.140     sakharuk 1935:     }
1.214     sakharuk 1936:     if ($tableofcontents eq 'yes') {$text=~s/(\\setcounter\{page\}\{1\})/$1 \\tableofcontents\\newpage /;}
                   1937:     if ($indexlist eq 'yes') {
1.627.2.17  raeburn  1938: 	$text=~s/(\\begin\{document})/\\makeindex $1/;
                   1939: 	$text=~s/(\\end\{document})/\\strut\\\\\\strut\\printindex $1/;
1.214     sakharuk 1940:     }
1.140     sakharuk 1941:     return $text;
                   1942: }
                   1943: 
                   1944: 
1.33      sakharuk 1945: sub page_cleanup {
                   1946:     my $result = shift;	
1.627.2.38! raeburn  1947: 
1.627.2.17  raeburn  1948:     $result =~ m/\\end\{document}(\d*)$/;
1.34      sakharuk 1949:     my $number_of_columns = $1;
1.33      sakharuk 1950:     my $insert = '{';
1.34      sakharuk 1951:     for (my $id=1;$id<=$number_of_columns;$id++) { $insert .='l'; }
1.33      sakharuk 1952:     $insert .= '}';
1.627.2.17  raeburn  1953:     $result =~ s/(\\begin\{longtable})INSERTTHEHEADOFLONGTABLE\\endfirsthead\\endhead/$1$insert/g;
1.34      sakharuk 1954:     $result =~ s/&\s*REMOVETHEHEADOFLONGTABLE\\\\/\\\\/g;
                   1955:     return $result,$number_of_columns;
1.7       sakharuk 1956: }
1.5       sakharuk 1957: 
1.3       sakharuk 1958: 
1.60      sakharuk 1959: sub details_for_menu {
1.335     albertel 1960:     my ($helper)=@_;
1.373     albertel 1961:     my $postdata=$env{'form.postdata'};
1.335     albertel 1962:     if (!$postdata) { $postdata=$helper->{VARS}{'postdata'}; }
                   1963:     my $name_of_resource = &Apache::lonnet::gettitle($postdata);
                   1964:     my $symbolic = &Apache::lonnet::symbread($postdata);
1.482     albertel 1965:     return if ( $symbolic eq '');
                   1966: 
1.233     www      1967:     my ($map,$id,$resource)=&Apache::lonnet::decode_symb($symbolic);
1.123     albertel 1968:     $map=&Apache::lonnet::clutter($map);
1.269     albertel 1969:     my $name_of_sequence = &Apache::lonnet::gettitle($map);
1.63      albertel 1970:     if ($name_of_sequence =~ /^\s*$/) {
1.123     albertel 1971: 	$map =~ m|([^/]+)$|;
                   1972: 	$name_of_sequence = $1;
1.63      albertel 1973:     }
1.373     albertel 1974:     my $name_of_map = &Apache::lonnet::gettitle($env{'request.course.uri'});
1.63      albertel 1975:     if ($name_of_map =~ /^\s*$/) {
1.373     albertel 1976: 	$env{'request.course.uri'} =~ m|([^/]+)$|;
1.123     albertel 1977: 	$name_of_map = $1;
                   1978:     }
1.335     albertel 1979:     return ($name_of_resource,$name_of_sequence,$name_of_map);
1.76      sakharuk 1980: }
                   1981: 
1.476     albertel 1982: sub copyright_line {
                   1983:     return '\noindent\makebox[\textwidth/$number_of_columns][b]{\hrulefill}\vspace*{-2 mm}\newline\noindent{\tiny Printed from LON-CAPA\copyright MSU{\hfill} Licensed under GNU General Public License } ';
                   1984: }
                   1985: my $end_of_student = "\n".'\special{ps:ENDOFSTUDENTSTAMP}'."\n";
1.76      sakharuk 1986: 
                   1987: sub latex_corrections {
1.408     albertel 1988:     my ($number_of_columns,$result,$selectionmade,$answer_mode) = @_;
1.627.2.17  raeburn  1989: #    $result =~ s/\\includegraphics\{/\\includegraphics\[width=\\minipagewidth\]{/g;
1.476     albertel 1990:     my $copyright = &copyright_line();
1.408     albertel 1991:     if ($selectionmade eq '1' || $answer_mode eq 'only') {
1.627.2.17  raeburn  1992: 	$result =~ s/(\\end\{document})/\\strut\\vskip 0 mm $copyright $end_of_student $1/;
1.408     albertel 1993:     } else {
1.627.2.17  raeburn  1994: 	$result =~ s/(\\end\{document})/\\strut\\vspace\*{-4 mm}\\newline $copyright $end_of_student $1/;
1.316     sakharuk 1995:     }
1.476     albertel 1996:     $result =~ s/\$number_of_columns/$number_of_columns/g;
1.627.2.23  raeburn  1997:     $result =~ s/(\\end\{longtable}\s*)(\\strut\\newline\\noindent\\makebox\[\\textwidth\/$number_of_columns\]\[b\]\{\\hrulefill})/$2$1/g;
1.627.2.17  raeburn  1998:     $result =~ s/(\\end\{longtable}\s*)\\strut\\newline/$1/g;
1.627.2.38! raeburn  1999: #-- LaTeX corrections
1.76      sakharuk 2000:     my $first_comment = index($result,'<!--',0);
                   2001:     while ($first_comment != -1) {
                   2002: 	my $end_comment = index($result,'-->',$first_comment);
                   2003: 	substr($result,$first_comment,$end_comment-$first_comment+3) = '';
                   2004: 	$first_comment = index($result,'<!--',$first_comment);
                   2005:     }
                   2006:     $result =~ s/^\s+$//gm; #remove empty lines
1.377     albertel 2007:     #removes more than one empty space
                   2008:     $result =~ s|(\s\s+)|($1=~/[\n\r]/)?"\n":" "|ge;
1.76      sakharuk 2009:     $result =~ s/\\\\\s*\\vskip/\\vskip/gm;
                   2010:     $result =~ s/\\\\\s*\\noindent\s*(\\\\)+/\\\\\\noindent /g;
                   2011:     $result =~ s/{\\par }\s*\\\\/\\\\/gm;
1.313     sakharuk 2012:     $result =~ s/\\\\\s+\[/ \[/g;
1.76      sakharuk 2013:     #conversion of html characters to LaTeX equivalents
                   2014:     if ($result =~ m/&(\w+|#\d+);/) {
                   2015: 	$result = &character_chart($result);
                   2016:     }
1.627.2.18  raeburn  2017:     $result =~ s/(\\end\{tabular})\s*\\vskip 0 mm/$1/g;
                   2018:     $result =~ s/(\\begin\{enumerate})\s*\\noindent/$1/g;
1.76      sakharuk 2019:     return $result;
1.60      sakharuk 2020: }
                   2021: 
1.3       sakharuk 2022: 
1.214     sakharuk 2023: sub index_table {
                   2024:     my $currentURL = shift;
                   2025:     my $insex_string='';
                   2026:     $currentURL=~s/\.([^\/+])$/\.$1\.meta/;
                   2027:     $insex_string=&Apache::lonnet::metadata($currentURL,'keywords');
                   2028:     return $insex_string;
                   2029: }
                   2030: 
                   2031: 
1.215     sakharuk 2032: sub IndexCreation {
                   2033:     my ($texversion,$currentURL)=@_;
                   2034:     my @key_words=split(/,/,&index_table($currentURL));
                   2035:     my $chunk='';
                   2036:     my $st=index $texversion,'\addcontentsline{toc}{subsection}{';
                   2037:     if ($st>0) {
                   2038: 	for (my $i=0;$i<3;$i++) {$st=(index $texversion,'}',$st+1);}
                   2039: 	$chunk=substr($texversion,0,$st+1);
                   2040: 	substr($texversion,0,$st+1)=' ';
                   2041:     }
                   2042:     foreach my $key_word (@key_words) {
                   2043: 	if ($key_word=~/\S+/) {
                   2044: 	    $texversion=~s/\b($key_word)\b/$1 \\index{$key_word} /i;
                   2045: 	}
                   2046:     }			
                   2047:     if ($st>0) {substr($texversion,0,1)=$chunk;}
                   2048:     return $texversion;
                   2049: }
                   2050: 
1.242     sakharuk 2051: sub print_latex_header {
                   2052:     my $mode=shift;
1.550     foxr     2053: 
                   2054:     return &Apache::londefdef::latex_header($mode);
1.242     sakharuk 2055: }
                   2056: 
                   2057: sub path_to_problem {
1.328     albertel 2058:     my ($urlp,$colwidth)=@_;
1.404     albertel 2059:     $urlp=&Apache::lonnet::clutter($urlp);
                   2060: 
1.242     sakharuk 2061:     my $newurlp = '';
1.328     albertel 2062:     $colwidth=~s/\s*mm\s*$//;
                   2063: #characters average about 2 mm in width
1.360     albertel 2064:     if (length($urlp)*2 > $colwidth) {
1.404     albertel 2065: 	my @elements = split('/',$urlp);
1.328     albertel 2066: 	my $curlength=0;
                   2067: 	foreach my $element (@elements) {
1.404     albertel 2068: 	    if ($element eq '') { next; }
1.328     albertel 2069: 	    if ($curlength+(length($element)*2) > $colwidth) {
1.404     albertel 2070: 		$newurlp .=  '|\vskip -1 mm \verb|';
                   2071: 		$curlength=length($element)*2;
1.328     albertel 2072: 	    } else {
                   2073: 		$curlength+=length($element)*2;
1.242     sakharuk 2074: 	    }
1.328     albertel 2075: 	    $newurlp.='/'.$element;
1.242     sakharuk 2076: 	}
1.253     sakharuk 2077:     } else {
                   2078: 	$newurlp=$urlp;
1.242     sakharuk 2079:     }
                   2080:     return '{\small\noindent\verb|'.$newurlp.'|\vskip 0 mm}';
                   2081: }
1.215     sakharuk 2082: 
1.275     sakharuk 2083: sub recalcto_mm {
                   2084:     my $textwidth=shift;
                   2085:     my $LaTeXwidth;
1.339     albertel 2086:     if ($textwidth=~/(-?\d+\.?\d*)\s*cm/) {
1.275     sakharuk 2087: 	$LaTeXwidth = $1*10;
1.339     albertel 2088:     } elsif ($textwidth=~/(-?\d+\.?\d*)\s*mm/) {
1.275     sakharuk 2089: 	$LaTeXwidth = $1;
1.339     albertel 2090:     } elsif ($textwidth=~/(-?\d+\.?\d*)\s*in/) {
1.275     sakharuk 2091: 	$LaTeXwidth = $1*25.4;
                   2092:     }
                   2093:     $LaTeXwidth.=' mm';
                   2094:     return $LaTeXwidth;
                   2095: }
                   2096: 
1.285     albertel 2097: sub get_textwidth {
                   2098:     my ($helper,$LaTeXwidth)=@_;
1.286     albertel 2099:     my $textwidth=$LaTeXwidth;
1.285     albertel 2100:     if ($helper->{'VARS'}->{'pagesize.width'}=~/\d+/ &&
                   2101: 	$helper->{'VARS'}->{'pagesize.widthunit'}=~/\w+/) {
1.286     albertel 2102: 	$textwidth=&recalcto_mm($helper->{'VARS'}->{'pagesize.width'}.' '.
                   2103: 				$helper->{'VARS'}->{'pagesize.widthunit'});
1.285     albertel 2104:     }
1.286     albertel 2105:     return $textwidth;
1.285     albertel 2106: }
                   2107: 
1.296     sakharuk 2108: 
                   2109: sub unsupported {
1.414     albertel 2110:     my ($currentURL,$mode,$symb)=@_;
1.627.2.30  raeburn  2111:     my $cleanURL=&Apache::lonenc::check_decrypt($currentURL);
                   2112:     my $shown = $currentURL;
                   2113:     if (($cleanURL ne $currentURL) || ($symb =~ m{/^enc/})) {
                   2114:         $shown = &mt('URL not shown (encrypted)');
                   2115:     }
1.307     sakharuk 2116:     if ($mode ne '') {$mode='\\'.$mode}
1.627.2.30  raeburn  2117:     my $result = &print_latex_header($mode);
1.627.2.38! raeburn  2118:     if ($cleanURL=~m|^(/adm/wrapper)?/ext/|) {
        !          2119:         $cleanURL=~s|^(/adm/wrapper)?/ext/|http://|;
1.627.2.30  raeburn  2120:         $cleanURL=~s|^http://https://|https://|;
                   2121:         if ($shown eq $currentURL) {
                   2122:             $shown = &Apache::lonxml::latex_special_symbols($cleanURL);
                   2123:         }
                   2124:         my $title=&Apache::lonnet::gettitle($symb);
                   2125:         $title = &Apache::lonxml::latex_special_symbols($title);
                   2126:         $result.=' \strut \\\\ \textit{'.$title.'} \strut \\\\ '.$shown.' ';
1.296     sakharuk 2127:     } else {
1.627.2.30  raeburn  2128:         if ($shown eq $currentURL) {
                   2129: 	    $result.=&Apache::lonxml::latex_special_symbols($currentURL);
                   2130:         } else {
1.627.2.38! raeburn  2131:             $result.=$shown;
1.627.2.30  raeburn  2132:         }
1.296     sakharuk 2133:     }
1.419     albertel 2134:     $result.= '\vskip 0.5mm\noindent\makebox[\textwidth/$number_of_columns][b]{\hrulefill} \end{document}';
1.296     sakharuk 2135:     return $result;
                   2136: }
                   2137: 
1.559     foxr     2138: #
                   2139: #  Map from helper layout style to the book/album:
                   2140: #
                   2141: sub map_laystyle {
                   2142:     my ($laystyle) = @_;
                   2143:     if ($laystyle eq 'L') {
                   2144: 	$laystyle='album';
                   2145:     } else {
                   2146: 	$laystyle='book';
                   2147:     }
                   2148:     return $laystyle;
                   2149: }
                   2150: 
                   2151: sub print_page_in_course {
                   2152:     my ($helper, $rparmhash, $currentURL, $resources) = @_;
1.602     www      2153: 
1.559     foxr     2154:     my %parmhash       = %$rparmhash;
                   2155:     my @page_resources = @$resources;
                   2156:     my $mode = $helper->{'VARS'}->{'LATEX_TYPE'};
                   2157:     my $symb = $helper->{'VARS'}->{'symb'};
                   2158: 
                   2159: 
                   2160:     my $format_from_helper = $helper->{'VARS'}->{'FORMAT'};
                   2161: 
                   2162: 
                   2163:     my @temporary_array=split /\|/,$format_from_helper;
                   2164:     my ($laystyle,$numberofcolumns,$papersize,$pdfFormFields)=@temporary_array;
                   2165:     $laystyle = &map_laystyle($laystyle);
                   2166:     my ($textwidth,$textheight,$oddoffset,$evenoffset) = &page_format($papersize,$laystyle,
                   2167: 								      $numberofcolumns);
1.627.2.38! raeburn  2168:     my $LaTeXwidth=&recalcto_mm($textwidth);
1.559     foxr     2169: 
                   2170:     if ($mode ne '') {$mode='\\'.$mode}
1.562     foxr     2171:     my $result   =    &print_latex_header($mode);
1.627.2.30  raeburn  2172: 
                   2173:     my $title=&Apache::lonnet::gettitle($currentURL);
                   2174:     $title = &Apache::lonxml::latex_special_symbols($title);
                   2175:     $result .= '\noindent\textit{'.$title.'}\\\\';
1.559     foxr     2176: 
                   2177:     if ($helper->{'VARS'}->{'style_file'}=~/\w/) {
                   2178: 	&Apache::lonnet::appenv({'construct.style' =>
                   2179: 				$helper->{'VARS'}->{'style_file'}});
                   2180:     } elsif ($env{'construct.style'}) {
                   2181: 	&Apache::lonnet::delenv('construct.style');
                   2182:     }
                   2183: 
1.627.2.38! raeburn  2184:     # First is the overall page description.  This is then followed by the
1.559     foxr     2185:     # components of the page. Each of which must be printed independently.
1.627.2.38! raeburn  2186:     my $the_page = shift(@page_resources);
1.559     foxr     2187: 
                   2188: 
                   2189:     foreach my $resource (@page_resources) {
                   2190: 	my $resource_src   = $resource->src(); # Essentially the URL of the resource.
1.627.2.30  raeburn  2191:         my $current_url = $resource->link();
1.559     foxr     2192: 
                   2193: 	# Recurse if a .page:
                   2194: 
                   2195: 	if ($resource_src =~ /.page$/i) {
                   2196: 	    my $navmap         = Apache::lonnavmaps::navmap->new();
                   2197: 	    my @page_resources = $navmap->retrieveResources($resource_src);
1.627.2.38! raeburn  2198: 	    $result           .= &print_page_in_course($helper, $rparmhash,
1.559     foxr     2199: 						       $resource_src, \@page_resources);
1.624     raeburn  2200:         } elsif ($resource->ext()) {
1.627.2.30  raeburn  2201:             $result.=&latex_header_footer_remove(&unsupported($current_url,$mode,$resource->symb));
                   2202: 	} elsif ($resource_src =~ /\.(problem|exam|quiz|assess|survey|form|library|xml|html|htm|xhtml|xhtm)$/)  {
1.627.2.38! raeburn  2203:             # these resources go through the XML transformer:
1.627.2.30  raeburn  2204:             $result .= &Apache::lonxml::latex_special_symbols($resource->title()) . '\\\\';
1.602     www      2205: 
1.559     foxr     2206: 	    my $urlp = &Apache::lonnet::clutter($resource_src);
1.602     www      2207: 
1.559     foxr     2208: 	    my %form;
                   2209: 	    my %moreenv;
                   2210: 
                   2211: 	    &Apache::lonxml::remember_problem_counter();
                   2212: 	    $moreenv{'request.filename'}=$urlp;
                   2213: 	    if ($helper->{'VARS'}->{'probstatus'} eq 'exam') {$form{'problemtype'}='exam';}
                   2214: 
                   2215: 	    $form{'grade_target'}  = 'tex';
                   2216: 	    $form{'textwidth'}    = &get_textwidth($helper, $LaTeXwidth);
1.627.2.38! raeburn  2217: 	    $form{'pdfFormFields'} = 'no'; #
        !          2218: 	    $form{'showallfoils'} = $helper->{'VARS'}->{'showallfoils'};
        !          2219: 
1.559     foxr     2220: 	    $form{'problem_split'}=$parmhash{'problem_stream_switch'};
                   2221: 	    $form{'suppress_tries'}=$parmhash{'suppress_tries'};
                   2222: 	    $form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
                   2223: 	    $form{'print_discussions'}=$helper->{'VARS'}->{'PRINT_DISCUSSIONS'};
                   2224: 	    $form{'print_annotations'}=$helper->{'VARS'}->{'PRINT_ANNOTATIONS'};
                   2225: 	    if (($helper->{'VARS'}->{'PRINT_DISCUSSIONS'} eq 'yes') ||
                   2226: 		($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes')) {
                   2227: 		$form{'problem_split'}='yes';
                   2228: 	    }
                   2229: 	    my $rndseed = time;
                   2230: 	    if ($helper->{'VARS'}->{'curseed'}) {
                   2231: 		$rndseed=$helper->{'VARS'}->{'curseed'};
                   2232: 	    }
                   2233: 	    $form{'rndseed'}=$rndseed;
                   2234: 	    &Apache::lonnet::appenv(\%moreenv);
1.627.2.38! raeburn  2235: 
1.559     foxr     2236: 	    &Apache::lonxml::clear_problem_counter();
                   2237: 
                   2238: 	    my $texversion = &ssi_with_retries($urlp, $ssi_retry_count, %form);
                   2239: 
                   2240: 
                   2241: 	    # current document with answers.. no need to encap in minipage
                   2242: 	    #  since there's only one answer.
                   2243: 
                   2244: 	    if(($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
                   2245: 	       ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) {
                   2246: 		my %answerform = %form;
                   2247: 
                   2248: 
                   2249: 		$answerform{'problem_split'}=$parmhash{'problem_stream_switch'};
                   2250: 		$answerform{'grade_target'}='answer';
                   2251: 		$answerform{'answer_output_mode'}='tex';
                   2252: 		$answerform{'rndseed'}=$rndseed;
                   2253:                 if ($helper->{'VARS'}->{'probstatus'} eq 'exam') {
                   2254: 		    $answerform{'problemtype'}='exam';
                   2255: 		}
                   2256: 		$resources_printed .= $urlp.':';
                   2257: 		my $answer=&ssi_with_retries($urlp,$ssi_retry_count, %answerform);
                   2258: 
                   2259: 		if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.627.2.17  raeburn  2260: 		    $texversion=~s/(\\keephidden\{ENDOFPROBLEM})/$answer$1/;
1.559     foxr     2261: 		} else {
1.562     foxr     2262: 		    $texversion= &print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
1.559     foxr     2263: 		    if ($helper->{'VARS'}->{'construction'} ne '1') {
                   2264: 			my $title = &Apache::lonnet::gettitle($helper->{'VARS'}->{'symb'});
                   2265: 			$title = &Apache::lonxml::latex_special_symbols($title);
                   2266: 			$texversion.='\vskip 0 mm \noindent\textbf{'.$title.'}\vskip 0 mm ';
                   2267: 			$texversion.=&path_to_problem($urlp,$LaTeXwidth);
                   2268: 		    } else {
1.602     www      2269: 			$texversion.='\vskip 0 mm \noindent\textbf{'.
1.627.2.6  raeburn  2270:                         &mt("Printing from Authoring Space: No Title").'}\vskip 0 mm ';
1.602     www      2271: 			$texversion.=&path_to_problem($urlp,$LaTeXwidth);
1.559     foxr     2272: 		    }
                   2273: 		    $texversion.='\vskip 1 mm '.$answer.'\end{document}';
                   2274: 		}
                   2275: 	    }
                   2276: 	    # Print annotations.
                   2277: 
                   2278: 
                   2279: 	    if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   2280: 		my $annotation .= &annotate($currentURL);
1.627.2.17  raeburn  2281: 		$texversion =~ s/(\\keephidden\{ENDOFPROBLEM})/$annotation$1/;
1.559     foxr     2282: 	    }
1.627.2.38! raeburn  2283: 
1.559     foxr     2284: 	    if ($helper->{'VARS'}->{'TABLE_INDEX'} eq 'yes') {
                   2285: 		$texversion=&IndexCreation($texversion,$currentURL);
                   2286: 	    }
                   2287: 	    if ($helper->{'VARS'}->{'CONSTR_RESOURSE_URL'} eq 'yes') {
                   2288: 		$texversion=~s/(\\addcontentsline\{toc\}\{subsection\}\{[^\}]*\})/$1 URL: \\verb|$currentURL| \\strut\\\\\\strut /;
                   2289: 
                   2290: 	    }
1.562     foxr     2291: 	    $texversion = &latex_header_footer_remove($texversion);
                   2292: 
                   2293: 	    # the first remaining line is a comment from londefdef the second
                   2294: 	    # line  seems to be an extraneous \vskip 1mm \\\\ :
                   2295:             # (imperfect removal from header_footer_remove?
                   2296: 
                   2297: 	    $texversion =~ s/\\vskip 1mm \\\\\\\\//;
                   2298: 
1.559     foxr     2299: 	    $result .= $texversion;
                   2300: 	    if ($currentURL=~m/\.page\s*$/) {
                   2301: 		($result,$numberofcolumns) = &page_cleanup($result);
                   2302: 	    }
                   2303: 	}
                   2304:     }
                   2305: 
                   2306:     $result.= '\vskip 0.5mm\noindent\makebox[\textwidth/$number_of_columns][b]{\hrulefill} \end{document}';
                   2307:     return $result;
                   2308: }
                   2309: 
1.296     sakharuk 2310: 
1.363     foxr     2311: #
1.395     www      2312: # List of recently generated print files
                   2313: #
                   2314: sub recently_generated {
1.584     raeburn  2315:     my ($prtspool) = @_;
                   2316:     my $output;
1.400     albertel 2317:     my $zip_result;
                   2318:     my $pdf_result;
1.395     www      2319:     opendir(DIR,$prtspool);
1.400     albertel 2320: 
1.627.2.38! raeburn  2321:     my @files =
1.400     albertel 2322: 	grep(/^$env{'user.name'}_$env{'user.domain'}_printout_(\d+)_.*\.(pdf|zip)$/,readdir(DIR));
1.395     www      2323:     closedir(DIR);
1.400     albertel 2324: 
                   2325:     @files = sort {
                   2326: 	my ($actime) = (stat($prtspool.'/'.$a))[10];
                   2327: 	my ($bctime) = (stat($prtspool.'/'.$b))[10];
                   2328: 	return $bctime <=> $actime;
                   2329:     } (@files);
                   2330: 
                   2331:     foreach my $filename (@files) {
                   2332: 	my ($ext) = ($filename =~ m/(pdf|zip)$/);
                   2333: 	my ($cdev,$cino,$cmode,$cnlink,
                   2334: 	    $cuid,$cgid,$crdev,$csize,
                   2335: 	    $catime,$cmtime,$cctime,
                   2336: 	    $cblksize,$cblocks)=stat($prtspool.'/'.$filename);
1.544     bisitz   2337:         my $ext_text = 'pdf' ? &mt('PDF File'):&mt('Zip File');
                   2338: 	my $result=&Apache::loncommon::start_data_table_row()
                   2339:                   .'<td>'
                   2340:                   .'<a href="/prtspool/'.$filename.'">'.$ext_text.'</a>'
                   2341:                   .'</td>'
                   2342:                   .'<td>'.&Apache::lonlocal::locallocaltime($cctime).'</td>'
                   2343:                   .'<td align="right">'.$csize.'</td>'
                   2344:                   .&Apache::loncommon::end_data_table_row();
1.400     albertel 2345: 	if ($ext eq 'pdf') { $pdf_result .= $result; }
                   2346: 	if ($ext eq 'zip') { $zip_result .= $result; }
                   2347:     }
1.544     bisitz   2348:     if ($zip_result || $pdf_result) {
1.584     raeburn  2349:         $output ='<hr />';
1.544     bisitz   2350:     }
1.400     albertel 2351:     if ($zip_result) {
1.584     raeburn  2352: 	$output .='<h3>'.&mt('Recently generated printout zip files')."</h3>\n"
1.544     bisitz   2353:                   .&Apache::loncommon::start_data_table()
                   2354:                   .&Apache::loncommon::start_data_table_header_row()
                   2355:                   .'<th>'.&mt('Download').'</th>'
                   2356:                   .'<th>'.&mt('Creation Date').'</th>'
                   2357:                   .'<th>'.&mt('File Size (Bytes)').'</th>'
                   2358:                   .&Apache::loncommon::end_data_table_header_row()
                   2359:                   .$zip_result
1.584     raeburn  2360:                   .&Apache::loncommon::end_data_table();
1.400     albertel 2361:     }
                   2362:     if ($pdf_result) {
1.584     raeburn  2363: 	$output .='<h3>'.&mt('Recently generated printouts')."</h3>\n"
1.544     bisitz   2364:                   .&Apache::loncommon::start_data_table()
                   2365:                   .&Apache::loncommon::start_data_table_header_row()
                   2366:                   .'<th>'.&mt('Download').'</th>'
                   2367:                   .'<th>'.&mt('Creation Date').'</th>'
                   2368:                   .'<th>'.&mt('File Size (Bytes)').'</th>'
                   2369:                   .&Apache::loncommon::end_data_table_header_row()
                   2370:                   .$pdf_result
1.584     raeburn  2371:                   .&Apache::loncommon::end_data_table();
1.396     albertel 2372:     }
1.584     raeburn  2373:     return $output;
1.395     www      2374: }
                   2375: 
                   2376: #
1.363     foxr     2377: #   Retrieve the hash of page breaks.
                   2378: #
                   2379: #  Inputs:
                   2380: #    helper   - reference to helper object.
                   2381: #  Outputs
                   2382: #    A reference to a page break hash.
                   2383: #
                   2384: #
1.610     foxr     2385: # use Data::Dumper;
1.569     foxr     2386: # sub dump_helper_vars {
1.418     foxr     2387: #    my ($helper) = @_;
                   2388: #    my $helpervars = Dumper($helper->{'VARS'});
                   2389: #    &Apache::lonnet::logthis("Dump of helper vars:\n $helpervars");
                   2390: #}
1.363     foxr     2391: 
1.481     albertel 2392: sub get_page_breaks  {
                   2393:     my ($helper) = @_;
                   2394:     my %page_breaks;
                   2395: 
                   2396:     foreach my $break (split /\|\|\|/, $helper->{'VARS'}->{'FINISHPAGE'}) {
                   2397: 	$page_breaks{$break} = 1;
                   2398:     }
                   2399:     return %page_breaks;
                   2400: }
1.627.2.38! raeburn  2401: #
1.569     foxr     2402: #   Returns text to insert for any extra vskip prior to the resource.
                   2403: #   Parameters:
                   2404: #     helper   - Reference to the helper object driving the printout.
                   2405: #     resource - Identifies the resource about to be printed.
                   2406: #
                   2407: #   This is done as follows:
                   2408: #    POSSIBLE_RESOURCES has the list of possible resources.
                   2409: #    EXTRASPACE         has the list of extra space values.
1.570     foxr     2410: #    EXTRASPACE_UNITS   is the set of resources for which the units are
                   2411: #                       mm. All others are 'in'.
1.627.2.38! raeburn  2412: #
1.569     foxr     2413: #    The resource is found in the POSSIBLE_RESOURCES to get the index
                   2414: #    of the EXTRASPACE value.
                   2415: #
                   2416: #   In order to speed this up for lengthy printouts, the first time,
                   2417: #   POSSIBLE_RESOURCES is turned into a look up hash and
                   2418: #   EXTRASPACE is turned into an array.
                   2419: #
                   2420: 
                   2421: 
                   2422: my %possible_resources;
1.570     foxr     2423: my %extraspace_mm;
1.569     foxr     2424: my @extraspace;
                   2425: my $skips_loaded       = 0;
                   2426: 
                   2427: #  Function to load the skips hash and array
                   2428: 
                   2429: sub load_skips {
                   2430: 
                   2431:     my ($helper)  = @_;
                   2432: 
1.627.2.38! raeburn  2433:     # If this is the first time, unwrap the resources and extra spaces:
1.569     foxr     2434: 
                   2435:     if (!$skips_loaded) {
                   2436: 	@extraspace = (split(/\|\|\|/, $helper->{'VARS'}->{'EXTRASPACE'}));
                   2437: 	my @resource_list = (split(/\|\|\|/, $helper->{'VARS'}->{'POSSIBLE_RESOURCES'}));
                   2438: 	my $i = 0;
                   2439: 	foreach my $resource (@resource_list) {
                   2440: 	    $possible_resources{$resource} = $i;
                   2441: 	    $i++;
                   2442: 	}
1.570     foxr     2443: 	foreach my $mm_resource (split(/\|\|\|/, $helper->{'VARS'}->{'EXTRASPACE_UNITS'})) {
                   2444: 	    $extraspace_mm{$mm_resource} = 1;
                   2445: 	}
1.569     foxr     2446: 	$skips_loaded = 1;
                   2447:     }
                   2448: }
                   2449: 
                   2450: sub get_extra_vspaces {
                   2451:     my ($helper, $resource) = @_;
                   2452: 
                   2453:     &load_skips($helper);
                   2454: 
                   2455:     #  Lookup the resource in the possible resources hash.. that is the index
                   2456:     # into the extraspace array that gives us either an empty string or
                   2457:     # the number of mm to skip:
                   2458: 
                   2459:     my $index = $possible_resources{$resource};
                   2460:     my $skip  = $extraspace[$index];
                   2461: 
                   2462:     my $result = '';
                   2463:     if ($skip ne '') {
1.570     foxr     2464: 	my $units = 'in';
                   2465: 	if (defined($extraspace_mm{$resource})) {
                   2466: 	    $units = 'mm';
                   2467: 	}
                   2468: 	$result = '\vskip '.$skip.' '.$units;
1.569     foxr     2469:     }
1.570     foxr     2470: 
                   2471: 	
1.569     foxr     2472:     return $result;
                   2473: 
                   2474: 
                   2475: }
                   2476: 
                   2477: #
                   2478: #  The resource chooser part of the helper needs more than just
                   2479: #  the value of the extraspaces var to recover the value into a text
                   2480: #  field option.  This sub produces the required format for the saved var:
1.627.2.38! raeburn  2481: #  specifically
1.569     foxr     2482: #    ||| separated fields of the form resourcename=value
                   2483: #
                   2484: #  Parameters:
                   2485: #    $helper     - Refers to the helper we are configuring
                   2486: #  Implicit input:
                   2487: #     $helper->{'VARS'}->{'EXTRASPACE'}  - the spaces helper var has the text field
                   2488: #                                          value.
1.570     foxr     2489: #     $helper->{'VARS'}->{'EXTRASPACE_UNITS'} - units for the skips (checkboxes).
1.569     foxr     2490: #     $helper->{'VARS'}->{'POSSIBLE_RESOURCES'}  - has the list of resources. |||
                   2491: #                                          separated of course.
                   2492: #  Implicit outputs:
1.570     foxr     2493: #     $env{'form.extraspace'}
                   2494: #     $env{'form.extraspace_units'}
1.569     foxr     2495: #
                   2496: sub set_form_extraspace {
                   2497:     my ($helper) = @_;
                   2498: 
                   2499:     # the most convenient way to do this is to drive from the skips arrays/hash.
                   2500:     # may not be the fastest, but this is once per print request so it's not so
                   2501:     # speed critical:
                   2502: 
                   2503:     &load_skips($helper);
                   2504: 
                   2505:     my $result = '';
                   2506: 
                   2507:     foreach my $resource (keys(%possible_resources)) {
                   2508: 	my $vskip = $extraspace[$possible_resources{$resource}];
                   2509: 	$result  .= $resource .'=' . $vskip . '|||';
                   2510:     }
                   2511: 
                   2512:     $env{'form.extraspace'}  = $result;
1.570     foxr     2513:     $env{'form.extraspace_units'} = $helper->{'VARS'}->{'EXTRASPACE_UNITS'};
1.569     foxr     2514:     return $result;
1.627.2.38! raeburn  2515: 
1.569     foxr     2516: }
1.363     foxr     2517: 
1.459     foxr     2518: #  Output a sequence (recursively if neeed)
                   2519: #  from construction space.
                   2520: # Parameters:
                   2521: #    url     = URL of the sequence to print.
                   2522: #    helper  - Reference to the helper hash.
                   2523: #    form    - Copy of the format hash.
                   2524: #    LaTeXWidth
                   2525: # Returns:
                   2526: #   Text to add to the printout.
                   2527: #   NOTE if the first element of the outermost sequence
                   2528: #   is itself a sequence, the outermost caller may need to
                   2529: #   prefix the latex with the page headers stuff.
                   2530: #
                   2531: sub print_construction_sequence {
                   2532:     my ($currentURL, $helper, %form, $LaTeXwidth) = @_;
1.590     foxr     2533: 
1.459     foxr     2534:     my $result;
                   2535:     my $rndseed=time;
                   2536:     if ($helper->{'VARS'}->{'curseed'}) {
                   2537: 	$rndseed=$helper->{'VARS'}->{'curseed'};
                   2538:     }
1.606     www      2539:     my $errtext=&LONCAPA::map::mapread(&Apache::lonnet::filelocation('',$currentURL));
                   2540: 
1.627.2.38! raeburn  2541:     #
1.459     foxr     2542:     #  These make this all support recursing for subsequences.
                   2543:     #
1.491     albertel 2544:     my @order    = @LONCAPA::map::order;
1.627.2.38! raeburn  2545:     my @resources = @LONCAPA::map::resources;
1.606     www      2546: 
1.459     foxr     2547:     for (my $member=0;$member<=$#order;$member++) {
                   2548: 	$resources[$order[$member]]=~/^([^:]*):([^:]*):/;
                   2549: 	my $urlp=$2;
                   2550: 	if ($urlp=~/\.(problem|exam|quiz|assess|survey|form|library|xml|html|htm|xhtml|xhtm)$/) {
                   2551: 	    my $texversion='';
                   2552: 	    if ($helper->{'VARS'}->{'ANSWER_TYPE'} ne 'only') {
                   2553: 		$form{'problem_split'}=$parmhash{'problem_stream_switch'};
                   2554: 		$form{'suppress_tries'}=$parmhash{'suppress_tries'};
                   2555: 		$form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
                   2556: 		$form{'rndseed'}=$rndseed;
                   2557: 		$resources_printed .=$urlp.':';
1.515     foxr     2558: 		$texversion=&ssi_with_retries($urlp, $ssi_retry_count, %form);
1.459     foxr     2559: 	    }
                   2560: 	    if((($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
1.627.2.38! raeburn  2561: 		($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) &&
1.609     www      2562: 	       ($urlp=~/$LONCAPA::assess_page_re/)) {
1.459     foxr     2563: 		#  Don't permanently modify %$form...
                   2564: 		my %answerform = %form;
                   2565: 		$answerform{'grade_target'}='answer';
                   2566: 		$answerform{'answer_output_mode'}='tex';
                   2567: 		$answerform{'rndseed'}=$rndseed;
                   2568: 		$answerform{'problem_split'}=$parmhash{'problem_stream_switch'};
1.481     albertel 2569: 		if ($urlp=~/\/res\//) {$env{'request.state'}='published';}
1.459     foxr     2570: 		$resources_printed .= $urlp.':';
1.515     foxr     2571: 		my $answer=&ssi_with_retries($urlp, $ssi_retry_count, %answerform);
1.459     foxr     2572: 		if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.627.2.17  raeburn  2573: 		    $texversion=~s/(\\keephidden\{ENDOFPROBLEM})/$answer$1/;
1.459     foxr     2574: 		} else {
                   2575: 		    # If necessary, encapsulate answer in minipage:
1.627.2.38! raeburn  2576: 
1.459     foxr     2577: 		    $texversion=&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
1.477     albertel 2578: 		    my $title = &Apache::lonnet::gettitle($helper->{'VARS'}->{'symb'});
                   2579: 		    $title = &Apache::lonxml::latex_special_symbols($title);
                   2580: 		    my $body ='\vskip 0 mm \noindent\textbf{'.$title.'}\vskip 0 mm ';
1.459     foxr     2581: 		    $body.=&path_to_problem($urlp,$LaTeXwidth);
                   2582: 		    $body.='\vskip 1 mm '.$answer.'\end{document}';
1.627.2.36  raeburn  2583: 		    $body = &encapsulate_minipage($body,$answerform{'problem_split'});
1.459     foxr     2584: 		    $texversion.=$body;
                   2585: 		}
                   2586: 	    }
                   2587: 	    $texversion = &latex_header_footer_remove($texversion);
                   2588: 
                   2589: 	    if ($helper->{'VARS'}->{'TABLE_INDEX'} eq 'yes') {
                   2590: 		$texversion=&IndexCreation($texversion,$urlp);
                   2591: 	    }
                   2592: 	    if ($helper->{'VARS'}->{'CONSTR_RESOURSE_URL'} eq 'yes') {
                   2593: 		$texversion=~s/(\\addcontentsline\{toc\}\{subsection\}\{[^\}]*\})/$1 URL: \\verb|$urlp| \\strut\\\\\\strut /;
                   2594: 	    }
                   2595: 	    $result.=$texversion;
                   2596: 
                   2597: 	} elsif ($urlp=~/\.(sequence|page)$/) {
1.627.2.38! raeburn  2598: 
1.459     foxr     2599: 	    # header:
                   2600: 
                   2601: 	    $result.='\strut\newline\noindent Sequence/page '.$urlp.'\strut\newline\noindent\makebox[\textwidth/$number_of_columns][b]{\hrulefill}\newline\noindent ';
                   2602: 
                   2603: 	    # IF sequence, recurse:
1.627.2.38! raeburn  2604: 
1.459     foxr     2605: 	    if ($urlp =~ /\.sequence$/) {
1.627.2.38! raeburn  2606: 		$result .= &print_construction_sequence($urlp,
        !          2607: 							$helper, %form,
1.459     foxr     2608: 							$LaTeXwidth);
                   2609: 	    }
1.550     foxr     2610: 	}
                   2611: 	elsif ($urlp =~ /\.pdf$/i) {
1.552     foxr     2612: 	    my $texversion;
                   2613: 	    if ($member != 0) {
                   2614: 		$texversion .= '\cleardoublepage';
                   2615: 	    }
                   2616: 
                   2617: 	    $texversion .= &include_pdf($urlp);
                   2618: 	    $texversion = &latex_header_footer_remove($texversion);
                   2619: 	    if ($member != $#order) {
                   2620: 		$texversion .= '\\ \cleardoublepage';
                   2621: 	    }
1.627.2.38! raeburn  2622: 
1.551     foxr     2623: 	    $result .= $texversion;
1.550     foxr     2624: 	}
1.459     foxr     2625:     }
1.627.2.18  raeburn  2626:     if ($helper->{VARS}->{'construction'} eq '1') {$result=~s/(\\begin\{document})/$1 \\fbox\{RANDOM SEED IS $rndseed\} /;}
1.459     foxr     2627:     return $result;
                   2628: }
                   2629: 
1.590     foxr     2630: #
                   2631: #  Top level for generating print output.
                   2632: #
                   2633: #  May call print_resources if multiple resources will be printed.
                   2634: #
                   2635: #  The main driver is $selectionmade which reflects the type of print out
                   2636: #  requested:
                   2637: #   Value    Print type:
                   2638: #   1        Print resource that's being looked at.
                   2639: #   2        Print problems in a map or in a page.
                   2640: #   3        Print pages in a map or resources in a page.
                   2641: #   4        Print all problems  or all resources.
                   2642: #   5        Print problems for seleted students.
                   2643: #   6        Print selected problems from a folder.
                   2644: #   7        Print print selected resources from some scope.
                   2645: #   8        Print resources for selected students.
1.627.2.38! raeburn  2646: #   9        Print for anonymous CODEs
1.590     foxr     2647: #
                   2648: #BZ 5209
                   2649: #   2        map_incomplete_problems_seq Print incomplete problems from the current
                   2650: #            folder in student context.
1.591     foxr     2651: #   5      map_incomplete_problems_people_seq Print incomplete problems from the
1.590     foxr     2652: #            current folder in privileged context.
1.591     foxr     2653: #    5      incomplete_problems_selpeople_course Print incomplete problems for
1.590     foxr     2654: #            selected people from the entire course.
                   2655: #
                   2656: #   Item 101 has much the same processing as 8,
                   2657: #
                   2658: #  Differences:  Item 101, 102 require per-student filtering of the resource
                   2659: #  set so that only the incomplete resources are printed.
                   2660: #  For item 100, filtering was done at the helper level.
                   2661: 
1.177     sakharuk 2662: sub output_data {
1.621     foxr     2663: 
1.184     sakharuk 2664:     my ($r,$helper,$rparmhash) = @_;
                   2665:     my %parmhash = %$rparmhash;
1.515     foxr     2666:     $ssi_error = 0;		# This will be set nonzero by failing ssi's.
1.459     foxr     2667:     $resources_printed = '';
1.556     foxr     2668:     $font_size = $helper->{'VARS'}->{'fontsize'};
1.590     foxr     2669:     my $print_type = $helper->{'VARS'}->{'PRINT_TYPE'}; # Allows textual simplification.
1.499     foxr     2670:     my $do_postprocessing = 1;
1.433     albertel 2671:     my $js = <<ENDPART;
                   2672: <script type="text/javascript">
1.264     sakharuk 2673:     var editbrowser;
                   2674:     function openbrowser(formname,elementname,only,omit) {
                   2675:         var url = '/res/?';
                   2676:         if (editbrowser == null) {
                   2677:             url += 'launch=1&';
                   2678:         }
                   2679:         url += 'catalogmode=interactive&';
                   2680:         url += 'mode=parmset&';
                   2681:         url += 'form=' + formname + '&';
                   2682:         if (only != null) {
                   2683:             url += 'only=' + only + '&';
1.627.2.38! raeburn  2684:         }
1.264     sakharuk 2685:         if (omit != null) {
                   2686:             url += 'omit=' + omit + '&';
                   2687:         }
                   2688:         url += 'element=' + elementname + '';
                   2689:         var title = 'Browser';
                   2690:         var options = 'scrollbars=1,resizable=1,menubar=0';
                   2691:         options += ',width=700,height=600';
                   2692:         editbrowser = open(url,title,options,'1');
                   2693:         editbrowser.focus();
                   2694:     }
                   2695: </script>
1.140     sakharuk 2696: ENDPART
                   2697: 
1.512     foxr     2698: 
1.558     bisitz   2699:     # Breadcrumbs
                   2700:     #FIXME: Choose better/different breadcrumbs?!? Links?
                   2701:     my $brcrum = [{'href' => '',
                   2702:                    'text' => 'Helper'}, #FIXME: Different origin possible than print out helper?
                   2703:                   {'href' => '',
                   2704:                    'text' => 'Preparing Printout'}];
                   2705: 
                   2706:     my $start_page  = &Apache::loncommon::start_page('Preparing Printout',
                   2707:                                                      $js,
                   2708:                                                      {'bread_crumbs' => $brcrum,});
1.433     albertel 2709:     my $msg = &mt('Please stand by while processing your print request, this may take some time ...');
1.363     foxr     2710: 
1.433     albertel 2711:     $r->print($start_page."\n<p>\n$msg\n</p>\n");
1.372     foxr     2712: 
1.363     foxr     2713:     # fetch the pagebreaks and store them in the course environment
                   2714:     # The page breaks will be pulled into the hash %page_breaks which is
                   2715:     # indexed by symb and contains 1's for each break.
                   2716: 
1.373     albertel 2717:     $env{'form.pagebreaks'}  = $helper->{'VARS'}->{'FINISHPAGE'};
1.569     foxr     2718:     &set_form_extraspace($helper);
1.627.2.38! raeburn  2719:     $env{'form.lastprinttype'} = $print_type;
1.363     foxr     2720:     &Apache::loncommon::store_course_settings('print',
1.366     foxr     2721: 					      {'pagebreaks'    => 'scalar',
1.569     foxr     2722: 					       'extraspace'    => 'scalar',
1.570     foxr     2723: 					       'extraspace_units' => 'scalar',
1.366     foxr     2724: 					       'lastprinttype' => 'scalar'});
1.364     albertel 2725:     my %page_breaks  = &get_page_breaks($helper);
1.363     foxr     2726: 
1.140     sakharuk 2727:     my $format_from_helper = $helper->{'VARS'}->{'FORMAT'};
                   2728:     my ($result,$selectionmade) = ('','');
                   2729:     my $number_of_columns = 1; #used only for pages to determine the width of the cell
                   2730:     my @temporary_array=split /\|/,$format_from_helper;
1.539     onken    2731:     my ($laystyle,$numberofcolumns,$papersize,$pdfFormFields)=@temporary_array;
1.559     foxr     2732: 
                   2733:     $laystyle = &map_laystyle($laystyle);
1.177     sakharuk 2734:     my ($textwidth,$textheight,$oddoffset,$evenoffset) = &page_format($papersize,$laystyle,$numberofcolumns);
1.373     albertel 2735:     my $assignment =  $env{'form.assignment'};
1.627.2.38! raeburn  2736:     my $LaTeXwidth=&recalcto_mm($textwidth);
1.272     sakharuk 2737:     my @print_array=();
1.274     sakharuk 2738:     my @student_names=();
1.627.2.38! raeburn  2739:     my $lastprinted;
1.360     albertel 2740: 
1.627.2.38! raeburn  2741:     #  Common settings for the %form hash:
        !          2742:     # In some cases these settings get overridden by specific cases, but the
1.360     albertel 2743:     # settings are common enough to make it worthwhile factoring them out
                   2744:     # here.
                   2745:     #
                   2746:     my %form;
                   2747:     $form{'grade_target'} = 'tex';
                   2748:     $form{'textwidth'}    = &get_textwidth($helper, $LaTeXwidth);
1.627.2.2  raeburn  2749:     $form{'pdfFormFields'} = 'no';
1.372     foxr     2750: 
                   2751:     # If form.showallfoils is set, then request all foils be shown:
1.627.2.38! raeburn  2752:     # privilege will be enforced both by not allowing the
1.372     foxr     2753:     # check box selecting this option to be presnt unless it's ok,
                   2754:     # and by lonresponse's priv. check.
                   2755:     # The if is here because lonresponse.pm only cares that
                   2756:     # showallfoils is defined, not what the value is.
                   2757: 
1.627.2.38! raeburn  2758:     if ($helper->{'VARS'}->{'showallfoils'} eq "1") {
1.372     foxr     2759: 	$form{'showallfoils'} = $helper->{'VARS'}->{'showallfoils'};
                   2760:     }
1.627.2.38! raeburn  2761: 
1.504     albertel 2762:     if ($helper->{'VARS'}->{'style_file'}=~/\w/) {
1.520     raeburn  2763: 	&Apache::lonnet::appenv({'construct.style' =>
                   2764: 				$helper->{'VARS'}->{'style_file'}});
1.504     albertel 2765:     } elsif ($env{'construct.style'}) {
1.549     raeburn  2766: 	&Apache::lonnet::delenv('construct.style');
1.504     albertel 2767:     }
                   2768: 
1.590     foxr     2769:     if ($print_type eq 'current_document') {
1.143     sakharuk 2770:       #-- single document - problem, page, html, xml, ...
1.343     albertel 2771: 	my ($currentURL,$cleanURL);
1.375     foxr     2772: 
1.162     sakharuk 2773: 	if ($helper->{'VARS'}->{'construction'} ne '1') {
1.185     sakharuk 2774:             #prints published resource
1.153     sakharuk 2775: 	    $currentURL=$helper->{'VARS'}->{'postdata'};
1.343     albertel 2776: 	    $cleanURL=&Apache::lonenc::check_decrypt($currentURL);
1.143     sakharuk 2777: 	} else {
1.512     foxr     2778: 
1.185     sakharuk 2779:             #prints resource from the construction space
1.602     www      2780: 	    $currentURL=$helper->{'VARS'}->{'filename'};
1.343     albertel 2781: 	    $cleanURL=$currentURL;
1.143     sakharuk 2782: 	}
1.140     sakharuk 2783: 	$selectionmade = 1;
1.627.2.19  raeburn  2784: 
1.413     albertel 2785: 	if ($cleanURL!~m|^/adm/|
1.557     foxr     2786: 	    && $cleanURL=~/\.(problem|exam|quiz|assess|survey|form|library|xml|html|htm|xhtml|xhtm)$/) {
1.169     albertel 2787: 	    my $rndseed=time;
1.242     sakharuk 2788: 	    my $texversion='';
                   2789: 	    if ($helper->{'VARS'}->{'ANSWER_TYPE'} ne 'only') {
                   2790: 		my %moreenv;
1.343     albertel 2791: 		$moreenv{'request.filename'}=$cleanURL;
1.290     sakharuk 2792:                 if ($helper->{'VARS'}->{'probstatus'} eq 'exam') {$form{'problemtype'}='exam';}
1.242     sakharuk 2793: 		$form{'problem_split'}=$parmhash{'problem_stream_switch'};
1.310     sakharuk 2794: 		$form{'suppress_tries'}=$parmhash{'suppress_tries'};
1.242     sakharuk 2795: 		$form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
1.309     sakharuk 2796: 		$form{'print_discussions'}=$helper->{'VARS'}->{'PRINT_DISCUSSIONS'};
1.511     foxr     2797: 		$form{'print_annotations'}=$helper->{'VARS'}->{'PRINT_ANNOTATIONS'};
                   2798: 		if (($helper->{'VARS'}->{'PRINT_DISCUSSIONS'} eq 'yes') ||
                   2799: 		    ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes')) {
                   2800: 		    $form{'problem_split'}='yes';
                   2801: 		}
1.242     sakharuk 2802: 		if ($helper->{'VARS'}->{'curseed'}) {
                   2803: 		    $rndseed=$helper->{'VARS'}->{'curseed'};
                   2804: 		}
                   2805: 		$form{'rndseed'}=$rndseed;
1.520     raeburn  2806: 		&Apache::lonnet::appenv(\%moreenv);
1.428     albertel 2807: 
                   2808: 		&Apache::lonxml::clear_problem_counter();
                   2809: 
1.375     foxr     2810: 		$resources_printed .= $currentURL.':';
1.515     foxr     2811: 		$texversion.=&ssi_with_retries($currentURL,$ssi_retry_count, %form);
1.428     albertel 2812: 
1.511     foxr     2813: 		#  Add annotations if required:
1.627.2.38! raeburn  2814: 
1.428     albertel 2815: 		&Apache::lonxml::clear_problem_counter();
                   2816: 
1.242     sakharuk 2817: 		&Apache::lonnet::delenv('request.filename');
1.230     albertel 2818: 	    }
1.423     foxr     2819: 	    # current document with answers.. no need to encap in minipage
                   2820: 	    #  since there's only one answer.
                   2821: 
1.242     sakharuk 2822: 	    if(($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
                   2823: 	       ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) {
1.602     www      2824: 
1.353     foxr     2825: 		$form{'problem_split'}=$parmhash{'problem_stream_switch'};
1.166     albertel 2826: 		$form{'grade_target'}='answer';
1.167     albertel 2827: 		$form{'answer_output_mode'}='tex';
1.169     albertel 2828: 		$form{'rndseed'}=$rndseed;
1.401     albertel 2829:                 if ($helper->{'VARS'}->{'probstatus'} eq 'exam') {
                   2830: 		    $form{'problemtype'}='exam';
                   2831: 		}
1.375     foxr     2832: 		$resources_printed .= $currentURL.':';
1.515     foxr     2833: 		my $answer=&ssi_with_retries($currentURL,$ssi_retry_count, %form);
1.511     foxr     2834: 		
                   2835: 
1.242     sakharuk 2836: 		if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.627.2.17  raeburn  2837: 		    $texversion=~s/(\\keephidden\{ENDOFPROBLEM})/$answer$1/;
1.242     sakharuk 2838: 		} else {
                   2839: 		    $texversion=&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
1.245     sakharuk 2840: 		    if ($helper->{'VARS'}->{'construction'} ne '1') {
1.477     albertel 2841: 			my $title = &Apache::lonnet::gettitle($helper->{'VARS'}->{'symb'});
                   2842: 			$title = &Apache::lonxml::latex_special_symbols($title);
                   2843: 			$texversion.='\vskip 0 mm \noindent\textbf{'.$title.'}\vskip 0 mm ';
1.343     albertel 2844: 			$texversion.=&path_to_problem($cleanURL,$LaTeXwidth);
1.245     sakharuk 2845: 		    } else {
1.602     www      2846: 			$texversion.='\vskip 0 mm \noindent\textbf{'.
1.627.2.6  raeburn  2847:                         &mt("Printing from Authoring Space: No Title").'}\vskip 0 mm ';
1.602     www      2848: 
                   2849: 			$texversion.=&path_to_problem($cleanURL,$LaTeXwidth);
1.245     sakharuk 2850: 		    }
1.242     sakharuk 2851: 		    $texversion.='\vskip 1 mm '.$answer.'\end{document}';
                   2852: 		}
1.511     foxr     2853: 
                   2854: 
                   2855: 		
                   2856: 
1.627.2.38! raeburn  2857: 
1.511     foxr     2858: 	    }
                   2859: 	    # Print annotations.
                   2860: 
                   2861: 
                   2862: 	    if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   2863: 		my $annotation .= &annotate($currentURL);
1.627.2.17  raeburn  2864: 		$texversion =~ s/(\\keephidden\{ENDOFPROBLEM})/$annotation$1/;
1.163     sakharuk 2865: 	    }
1.511     foxr     2866: 
                   2867: 
1.214     sakharuk 2868: 	    if ($helper->{'VARS'}->{'TABLE_INDEX'} eq 'yes') {
1.215     sakharuk 2869: 		$texversion=&IndexCreation($texversion,$currentURL);
1.214     sakharuk 2870: 	    }
1.219     sakharuk 2871: 	    if ($helper->{'VARS'}->{'CONSTR_RESOURSE_URL'} eq 'yes') {
                   2872: 		$texversion=~s/(\\addcontentsline\{toc\}\{subsection\}\{[^\}]*\})/$1 URL: \\verb|$currentURL| \\strut\\\\\\strut /;
                   2873: 
                   2874: 	    }
1.162     sakharuk 2875: 	    $result .= $texversion;
                   2876: 	    if ($currentURL=~m/\.page\s*$/) {
                   2877: 		($result,$number_of_columns) = &page_cleanup($result);
                   2878: 	    }
1.413     albertel 2879:         } elsif ($cleanURL!~m|^/adm/|
1.557     foxr     2880: 		 && $currentURL=~/\.(sequence|page)$/ && $helper->{'VARS'}->{'construction'} eq '1') {
1.459     foxr     2881: 	    $result .= &print_construction_sequence($currentURL, $helper, %form,
                   2882: 						    $LaTeXwidth);
1.627.2.38! raeburn  2883: 	    $result .= '\end{document}';
1.459     foxr     2884: 	    if (!($result =~ /\\begin\{document\}/)) {
                   2885: 		$result = &print_latex_header() . $result;
1.227     sakharuk 2886: 	    }
1.459     foxr     2887: 	    # End construction space sequence.
1.627.2.38! raeburn  2888: 	} elsif ($cleanURL=~/\/(smppg|syllabus|aboutme|bulletinboard)$/) {
1.258     sakharuk 2889: 		$form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
1.298     sakharuk 2890: 		if ($currentURL=~/\/syllabus$/) {$currentURL=~s/\/res//;}
1.375     foxr     2891: 		$resources_printed .= $currentURL.':';
1.567     foxr     2892: 		my $texversion = &ssi_with_retries($currentURL, $ssi_retry_count, %form);
1.511     foxr     2893: 		if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   2894: 		    my $annotation = &annotate($currentURL);
1.627.2.17  raeburn  2895: 		    $texversion    =~ s/(\\end\{document})/$annotation$1/;
1.511     foxr     2896: 		}
1.258     sakharuk 2897: 		$result .= $texversion;
1.550     foxr     2898: 	} elsif ($cleanURL =~/\.tex$/) {
1.498     foxr     2899: 	    # For this sort of print of a single LaTeX file,
                   2900: 	    # We can just print the LaTeX file as it is uninterpreted in any way:
                   2901: 	    #
                   2902: 
                   2903: 	    $result = &fetch_raw_resource($currentURL);
1.511     foxr     2904: 	    if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   2905: 		my $annotation = &annotate($currentURL);
1.627.2.17  raeburn  2906: 		$result =~ s/(\\end\{document})/$annotation$1/;
1.511     foxr     2907: 	    }
                   2908: 
1.499     foxr     2909: 	    $do_postprocessing = 0; # Don't massage the result.
1.498     foxr     2910: 
1.550     foxr     2911: 	} elsif ($cleanURL =~ /\.pdf$/i) {
                   2912: 	    $result .= &include_pdf($cleanURL);
1.551     foxr     2913: 	    $result .= '\end{document}';
1.559     foxr     2914: 	} elsif ($cleanURL =~ /\.page$/i) { #  Print page in non construction space contexts.
                   2915: 
                   2916: 	    # Determine the set of resources in the map of the page:
                   2917: 
                   2918: 	    my $navmap         =  Apache::lonnavmaps::navmap->new();
                   2919: 	    my @page_resources =  $navmap->retrieveResources($cleanURL);
                   2920: 	    $result           .=  &print_page_in_course($helper, $rparmhash,
                   2921: 							$cleanURL, \@page_resources);
                   2922: 
1.627.2.38! raeburn  2923: 
1.162     sakharuk 2924: 	} else {
1.414     albertel 2925: 	    $result.=&unsupported($currentURL,$helper->{'VARS'}->{'LATEX_TYPE'},
                   2926: 				  $helper->{'VARS'}->{'symb'});
1.162     sakharuk 2927: 	}
1.590     foxr     2928:     } elsif (($print_type eq 'map_problems')          or
                   2929: 	     ($print_type eq 'map_problems_in_page')  or
                   2930: 	     ($print_type eq 'map_resources_in_page') or
                   2931:              ($print_type eq 'map_problems_pages')    or
                   2932:              ($print_type eq 'all_problems')          or
                   2933: 	     ($print_type eq 'all_resources')         or # BUGBUG
                   2934: 	     ($print_type eq 'select_sequences')      or
                   2935: 	     ($print_type eq 'map_incomplete_problems_seq')
1.582     raeburn  2936: 	     ) {
1.627.2.38! raeburn  2937: 
1.141     sakharuk 2938:         #-- produce an output string
1.590     foxr     2939: 	if (($print_type eq 'map_problems')                or
                   2940: 	    ($print_type eq 'map_incomplete_problems_seq') or
                   2941: 	    ($print_type eq 'map_problems_in_page') ) {
1.296     sakharuk 2942: 	    $selectionmade = 2;
1.590     foxr     2943: 	} elsif (($print_type eq 'map_problems_pages') or
                   2944: 		 ($print_type eq 'map_resources_in_page'))
1.562     foxr     2945: 	{
1.296     sakharuk 2946: 	    $selectionmade = 3;
1.627.2.38! raeburn  2947: 	} elsif (($print_type eq 'all_problems')
1.536     foxr     2948: 		 ) {
1.296     sakharuk 2949: 	    $selectionmade = 4;
1.590     foxr     2950: 	} elsif ($print_type eq 'all_resources') {  #BUGBUG
1.354     foxr     2951: 	    $selectionmade = 4;
1.590     foxr     2952: 	} elsif ($print_type eq 'select_sequences') {
1.296     sakharuk 2953: 	    $selectionmade = 7;
                   2954: 	}
1.590     foxr     2955: 
1.193     sakharuk 2956: 	$form{'problem_split'}=$parmhash{'problem_stream_switch'};
1.310     sakharuk 2957: 	$form{'suppress_tries'}=$parmhash{'suppress_tries'};
1.203     sakharuk 2958: 	$form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
1.309     sakharuk 2959: 	$form{'print_discussions'}=$helper->{'VARS'}->{'PRINT_DISCUSSIONS'};
1.511     foxr     2960: 	$form{'print_annotations'} = $helper->{'VARS'}->{'PRINT_ANNOTATIONS'};
                   2961: 	if (($helper->{'VARS'}->{'PRINT_DISCUSSIONS'} eq 'yes')   ||
                   2962: 	    ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') ) {
                   2963: 	    $form{'problem_split'}='yes';
                   2964: 	}
1.141     sakharuk 2965: 	my $flag_latex_header_remove = 'NO';
                   2966: 	my $flag_page_in_sequence = 'NO';
                   2967: 	my @master_seq=split /\|\|\|/, $helper->{'VARS'}->{'RESOURCES'};
1.193     sakharuk 2968: 	my $prevassignment='';
1.428     albertel 2969: 
                   2970: 	&Apache::lonxml::clear_problem_counter();
                   2971: 
1.141     sakharuk 2972: 	for (my $i=0;$i<=$#master_seq;$i++) {
1.350     foxr     2973: 
1.521     foxr     2974: 	    &Apache::lonenc::reset_enc();
                   2975: 
1.350     foxr     2976: 	    # Note due to document structure, not allowed to put \newpage
                   2977: 	    # prior to the first resource
                   2978: 
1.351     foxr     2979: 	    if (defined $page_breaks{$master_seq[$i]}) {
1.350     foxr     2980: 		if($i != 0) {
                   2981: 		    $result.="\\newpage\n";
                   2982: 		}
                   2983: 	    }
1.569     foxr     2984: 	    $result .= &get_extra_vspaces($helper, $master_seq[$i]);
1.564     foxr     2985: 	    my ($sequence,$middle_thingy,$urlp)=&Apache::lonnet::decode_symb($master_seq[$i]);
1.237     albertel 2986: 	    $urlp=&Apache::lonnet::clutter($urlp);
1.166     albertel 2987: 	    $form{'symb'}=$master_seq[$i];
1.407     albertel 2988: 
                   2989: 	    my $assignment=&Apache::lonxml::latex_special_symbols(&Apache::lonnet::gettitle($sequence),'header'); #title of the assignment which contains this problem
1.521     foxr     2990: 
1.267     sakharuk 2991: 	    if ($selectionmade==7) {$helper->{VARS}->{'assignment'}=$assignment;}
1.247     sakharuk 2992: 	    if ($i==0) {$prevassignment=$assignment;}
1.297     sakharuk 2993: 	    my $texversion='';
1.413     albertel 2994: 	    if ($urlp!~m|^/adm/|
                   2995: 		&& $urlp=~/\.(problem|exam|quiz|assess|survey|form|library|page|xml|html|htm|xhtml|xhtm)$/) {
1.627.2.19  raeburn  2996:                 my $extension = $1;
1.375     foxr     2997: 		$resources_printed .= $urlp.':';
1.428     albertel 2998: 		&Apache::lonxml::remember_problem_counter();
1.566     foxr     2999: 		if ($flag_latex_header_remove eq 'NO') {
                   3000: 		    $texversion.=&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});  # RF
1.582     raeburn  3001:                     unless (($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only') ||
                   3002:                             (($i==0) &&
                   3003:                              (($urlp=~/\.page$/) ||
1.590     foxr     3004:                               ($print_type eq 'map_problems_in_page') ||
1.627.2.19  raeburn  3005:                               (($print_type eq 'map_resources_in_page') && ($extension !~ /^x?html?$/))))) {
1.582     raeburn  3006:                         $flag_latex_header_remove = 'YES';
                   3007:                     }
1.566     foxr     3008: 		}
1.515     foxr     3009: 		$texversion.=&ssi_with_retries($urlp, $ssi_retry_count, %form);
1.296     sakharuk 3010: 		if ($urlp=~/\.page$/) {
                   3011: 		    ($texversion,my $number_of_columns_page) = &page_cleanup($texversion);
1.627.2.38! raeburn  3012: 		    if ($number_of_columns_page > $number_of_columns) {$number_of_columns=$number_of_columns_page;}
1.627.2.17  raeburn  3013: 		    $texversion =~ s/\\end\{document}\d*/\\end{document}/;
1.296     sakharuk 3014: 		    $flag_page_in_sequence = 'YES';
1.582     raeburn  3015: 		}
1.428     albertel 3016: 
1.296     sakharuk 3017: 		if(($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
                   3018: 		   ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) {
1.380     foxr     3019: 		    #  Don't permanently pervert the %form hash
                   3020: 		    my %answerform = %form;
                   3021: 		    $answerform{'grade_target'}='answer';
                   3022: 		    $answerform{'answer_output_mode'}='tex';
1.375     foxr     3023: 		    $resources_printed .= $urlp.':';
1.428     albertel 3024: 
                   3025: 		    &Apache::lonxml::restore_problem_counter();
1.515     foxr     3026: 		    my $answer=&ssi_with_retries($urlp, $ssi_retry_count, %answerform);
1.582     raeburn  3027:                     if ($urlp =~ /\.page$/) {
1.627.2.17  raeburn  3028:                         $answer =~ s/\\end\{document}(\d*)$//;
1.582     raeburn  3029:                     }
1.296     sakharuk 3030: 		    if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.582     raeburn  3031:                         if ($urlp =~ /\.page$/) {
1.627.2.17  raeburn  3032:                             my @probs = split(/\\keephidden\{ENDOFPROBLEM}/,$texversion);
1.582     raeburn  3033:                             my $lastprob = pop(@probs);
                   3034:                             $texversion = join('\keephidden{ENDOFPROBLEM}',@probs).
                   3035:                             $answer.'\keephidden{ENDOFPROBLEM}'.$lastprob;
                   3036:                         } else {
1.627.2.17  raeburn  3037:                             $texversion=~s/(\\keephidden\{ENDOFPROBLEM})/$answer$1/;
1.582     raeburn  3038:                         }
1.249     sakharuk 3039: 		    } else {
1.609     www      3040: 			if ($urlp=~/$LONCAPA::assess_page_re/) {
1.296     sakharuk 3041: 			    $texversion=&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
1.627.2.17  raeburn  3042: #			    $texversion =~ s/\\begin\{document}//; # FIXME
1.477     albertel 3043: 			    my $title = &Apache::lonnet::gettitle($master_seq[$i]);
                   3044: 			    $title = &Apache::lonxml::latex_special_symbols($title);
                   3045: 			    my $body ='\vskip 0 mm \noindent\textbf{'.$title.'}\vskip 0 mm ';
1.423     foxr     3046: 			    $body   .= &path_to_problem ($urlp,$LaTeXwidth);
                   3047: 			    $body   .='\vskip 1 mm '.$answer;
1.627.2.36  raeburn  3048: 			    $body    = &encapsulate_minipage($body,$answerform{'problem_split'});
1.423     foxr     3049: 			    $texversion .= $body;
1.296     sakharuk 3050: 			} else {
                   3051: 			    $texversion='';
                   3052: 			}
1.249     sakharuk 3053: 		    }
1.511     foxr     3054: 
1.246     sakharuk 3055: 		}
1.511     foxr     3056: 		if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   3057: 		    my $annotation .= &annotate($urlp);
1.627.2.17  raeburn  3058: 		    $texversion =~ s/(\\keephidden\{ENDOFPROBLEM})/$annotation$1/;
1.511     foxr     3059: 		}
                   3060: 
1.296     sakharuk 3061: 		if ($flag_latex_header_remove ne 'NO') {
                   3062: 		    $texversion = &latex_header_footer_remove($texversion);
                   3063: 		} else {
1.627.2.17  raeburn  3064: 		    $texversion =~ s/\\end\{document}//;
1.296     sakharuk 3065: 		}
                   3066: 		if ($helper->{'VARS'}->{'TABLE_INDEX'} eq 'yes') {
                   3067: 		    $texversion=&IndexCreation($texversion,$urlp);
                   3068: 		}
                   3069: 		if (($selectionmade == 4) and ($assignment ne $prevassignment)) {
                   3070: 		    my $name = &get_name();
                   3071: 		    my $courseidinfo = &get_course();
1.627.2.1  raeburn  3072:                     if (defined($courseidinfo)) { $courseidinfo=' - '.$courseidinfo }
1.296     sakharuk 3073: 		    $prevassignment=$assignment;
1.455     albertel 3074: 		    my $header_text = $parmhash{'print_header_format'};
1.486     foxr     3075: 		    $header_text    = &format_page_header($textwidth, $header_text,
1.627.2.38! raeburn  3076: 							  $assignment,
        !          3077: 							  $courseidinfo,
1.455     albertel 3078: 							  $name);
1.417     foxr     3079: 		    if ($numberofcolumns eq '1') {
1.455     albertel 3080: 			$result .='\newpage \noindent\parbox{\minipagewidth}{\noindent\\lhead{'.$header_text.'}} \vskip 5 mm ';
1.416     foxr     3081: 		    } else {
1.455     albertel 3082: 			$result .='\newpage \noindent\parbox{\minipagewidth}{\noindent\\fancyhead[LO]{'.$header_text.'}} \vskip 5 mm ';
1.416     foxr     3083: 		    }			
1.296     sakharuk 3084: 		}
                   3085: 		$result .= $texversion;
1.627.2.38! raeburn  3086: 		$flag_latex_header_remove = 'YES';
        !          3087: 	    } elsif ($urlp=~/\/(smppg|syllabus|aboutme|bulletinboard)$/) {
1.301     sakharuk 3088: 		$form{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
                   3089: 		if ($urlp=~/\/syllabus$/) {$urlp=~s/\/res//;}
1.375     foxr     3090: 		$resources_printed .= $urlp.':';
1.567     foxr     3091: 		my $texversion = &ssi_with_retries($urlp, $ssi_retry_count, %form);
1.511     foxr     3092: 		if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   3093: 		    my $annotation = &annotate($urlp);
1.627.2.17  raeburn  3094: 		    $texversion =~ s/(\\end\{document)/$annotation$1/;
1.511     foxr     3095: 		}
                   3096: 
1.301     sakharuk 3097: 		if ($flag_latex_header_remove ne 'NO') {
                   3098: 		    $texversion = &latex_header_footer_remove($texversion);
1.550     foxr     3099: 		} else {	
1.627.2.17  raeburn  3100: 		    $texversion =~ s/\\end\{document}/\\vskip 0\.5mm\\noindent\\makebox\[\\textwidth\/\$number_of_columns\]\[b\]\{\\hrulefill\}/;
1.301     sakharuk 3101: 		}
                   3102: 		$result .= $texversion;
1.627.2.38! raeburn  3103: 		$flag_latex_header_remove = 'YES';
1.550     foxr     3104: 	    } elsif ($urlp=~ /\.pdf$/i) {
                   3105: 		if ($i > 0) {
                   3106: 		    $result .= '\cleardoublepage';
                   3107: 		}
1.580     raeburn  3108:                 my $texfrompdf = &include_pdf($urlp);
                   3109:                 if ($flag_latex_header_remove ne 'NO') {
                   3110:                     $texfrompdf = &latex_header_footer_remove($texfrompdf);
                   3111:                 }
                   3112:                 $result .= $texfrompdf;
1.550     foxr     3113: 		if ($i != $#master_seq) {
                   3114: 		    if ($numberofcolumns eq '1') {
                   3115: 			$result .= '\newpage';
                   3116: 		    } else {
                   3117: 			# the \\'s seem to be needed to let LaTeX know there's something
                   3118: 			# on the page since LaTeX seems to not like to clear an empty page.
                   3119: 			#
                   3120: 			$result .= '\\ \cleardoublepage';
                   3121: 		    }
                   3122: 		}
                   3123: 		$flag_latex_header_remove = 'YES';
                   3124: 
1.141     sakharuk 3125: 	    } else {
1.414     albertel 3126: 		$texversion=&unsupported($urlp,$helper->{'VARS'}->{'LATEX_TYPE'},
                   3127: 					 $master_seq[$i]);
1.297     sakharuk 3128: 		if ($flag_latex_header_remove ne 'NO') {
                   3129: 		    $texversion = &latex_header_footer_remove($texversion);
                   3130: 		} else {
1.627.2.17  raeburn  3131: 		    $texversion =~ s/\\end\{document}//;
1.297     sakharuk 3132: 		}
                   3133: 		$result .= $texversion;
1.627.2.38! raeburn  3134: 		$flag_latex_header_remove = 'YES';
1.582     raeburn  3135: 	    }
1.627.2.38! raeburn  3136: 	    if (&Apache::loncommon::connection_aborted($r)) {
        !          3137: 		last;
1.550     foxr     3138: 	    }
1.141     sakharuk 3139: 	}
1.428     albertel 3140: 	&Apache::lonxml::clear_problem_counter();
1.344     foxr     3141: 	if ($flag_page_in_sequence eq 'YES') {
1.627.2.17  raeburn  3142: 	    $result =~ s/\\usepackage\{calc}/\\usepackage{calc}\\usepackage{longtable}/;
1.344     foxr     3143: 	}	
1.141     sakharuk 3144: 	$result .= '\end{document}';
1.590     foxr     3145:      } elsif (($print_type eq 'problems_for_students')           ||
                   3146: 	      ($print_type eq 'problems_for_students_from_page') ||
                   3147: 	      ($print_type eq 'all_problems_students')           ||
1.591     foxr     3148: 	      ($print_type eq 'resources_for_students')          ||
                   3149: 	      ($print_type eq 'incomplete_problems_selpeople_course') ||
                   3150: 	      ($print_type eq 'map_incomplete_problems_people_seq')){
1.353     foxr     3151: 
                   3152: 
1.627.2.38! raeburn  3153:      #-- prints assignments for whole class or for selected students
1.284     albertel 3154: 	 my $type;
1.590     foxr     3155: 	 if (($print_type eq 'problems_for_students')           ||
                   3156: 	     ($print_type eq 'problems_for_students_from_page') ||
1.591     foxr     3157: 	     ($print_type eq 'all_problems_students')           ||
                   3158: 	     ($print_type eq 'incomplete_problems_selpeople_course') ||
                   3159: 	     ($print_type eq 'map_incomplete_problems_people_seq')) {
1.254     sakharuk 3160: 	     $selectionmade=5;
1.284     albertel 3161: 	     $type='problems';
1.590     foxr     3162: 	 } elsif ($print_type eq 'resources_for_students') {
1.254     sakharuk 3163: 	     $selectionmade=8;
1.284     albertel 3164: 	     $type='resources';
1.254     sakharuk 3165: 	 }
1.150     sakharuk 3166: 	 my @students=split /\|\|\|/, $helper->{'VARS'}->{'STUDENTS'};
1.341     foxr     3167: 	 #   The normal sort order is by section then by students within the
                   3168: 	 #   section. If the helper var student_sort is 1, then the user has elected
                   3169: 	 #   to override this and output the students by name.
                   3170: 	 #    Each element of the students array is of the form:
                   3171: 	 #       username:domain:section:last, first:status
1.627.2.38! raeburn  3172: 	 #
        !          3173: 	 #  Note that student sort is not compatible with printing
1.429     foxr     3174: 	 #  1 section per pdf...so that setting overrides.
1.627.2.38! raeburn  3175: 	 #
        !          3176: 	 if (($helper->{'VARS'}->{'student_sort'}    eq 1)  &&
1.429     foxr     3177: 	     ($helper->{'VARS'}->{'SPLIT_PDFS'} ne "sections")) {
1.341     foxr     3178: 	     @students = sort compare_names  @students;
1.618     foxr     3179: 	 } else {
1.627.2.38! raeburn  3180: 	     @students = sort compare_sections @students;
1.341     foxr     3181: 	 }
1.429     foxr     3182: 	 &adjust_number_to_print($helper);
                   3183: 
1.278     albertel 3184:          if ($helper->{'VARS'}->{'NUMBER_TO_PRINT'} eq '0' ||
                   3185: 	     $helper->{'VARS'}->{'NUMBER_TO_PRINT'} eq 'all' ) {
                   3186: 	     $helper->{'VARS'}->{'NUMBER_TO_PRINT'}=$#students+1;
                   3187: 	 }
1.627.2.38! raeburn  3188: 	 # If we are splitting on section boundaries, we need
        !          3189: 	 # to remember that in split_on_sections and
1.429     foxr     3190: 	 # print all of the students in the list.
                   3191: 	 #
                   3192: 	 my $split_on_sections = 0;
                   3193: 	 if ($helper->{'VARS'}->{'NUMBER_TO_PRINT'} eq 'section') {
                   3194: 	     $split_on_sections = 1;
                   3195: 	     $helper->{'VARS'}->{'NUMBER_TO_PRINT'} = $#students+1;
                   3196: 	 }
1.150     sakharuk 3197: 	 my @master_seq=split /\|\|\|/, $helper->{'VARS'}->{'RESOURCES'};
1.350     foxr     3198: 
1.626     raeburn  3199:          my $map;
                   3200:          if ($helper->{VARS}->{'symb'}) {
1.627.2.33  raeburn  3201:              unless ((($print_type eq 'all_problems_students') ||
                   3202:                       ($print_type eq 'incomplete_problems_selpeople_course')) &&
                   3203:                       $perm{'pfo'}) {
                   3204:                  ($map, my $id, my $resource) =
                   3205:                      &Apache::lonnet::decode_symb($helper->{VARS}->{'symb'});
                   3206:              }
1.626     raeburn  3207:          }
                   3208: 
1.150     sakharuk 3209: 	 #loop over students
1.562     foxr     3210: 
                   3211:  	 my $flag_latex_header_remove = 'NO';
1.150     sakharuk 3212: 	 my %moreenv;
1.330     sakharuk 3213:          $moreenv{'instructor_comments'}='hide';
1.285     albertel 3214: 	 $moreenv{'textwidth'}=&get_textwidth($helper,$LaTeXwidth);
1.309     sakharuk 3215: 	 $moreenv{'print_discussions'}=$helper->{'VARS'}->{'PRINT_DISCUSSIONS'};
1.511     foxr     3216: 	 $moreenv{'print_annotations'} = $helper->{'VARS'}->{'PRINT_ANNOTATIONS'};
1.353     foxr     3217: 	 $moreenv{'problem_split'}    = $parmhash{'problem_stream_switch'};
1.369     foxr     3218: 	 $moreenv{'suppress_tries'}   = $parmhash{'suppress_tries'};
1.511     foxr     3219: 	 if (($helper->{'VARS'}->{'PRINT_DISCUSSIONS'} eq 'yes')  ||
                   3220: 	     ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes')) {
                   3221: 	     $moreenv{'problem_split'}='yes';
                   3222: 	 }
1.611     www      3223: 	 my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin($r,$#students+1);
1.272     sakharuk 3224: 	 my $student_counter=-1;
1.429     foxr     3225: 	 my $i = 0;
1.430     albertel 3226: 	 my $last_section = (split(/:/,$students[0]))[2];
1.627.2.15  raeburn  3227:          my $nohidemap;
                   3228:          if ($perm{'pav'} && $perm{'vgr'}) {
                   3229:              $nohidemap = 1;
                   3230:          }
1.150     sakharuk 3231: 	 foreach my $person (@students) {
1.373     albertel 3232:              my $duefile="/home/httpd/prtspool/$env{'user.name'}_$env{'user.domain'}_printout.due";
1.311     sakharuk 3233: 	     if (-e $duefile) {
                   3234: 		 my $temp_file = Apache::File->new('>>'.$duefile);
                   3235: 		 print $temp_file "1969\n";
                   3236: 	     }
1.272     sakharuk 3237: 	     $student_counter++;
1.429     foxr     3238: 	     if ($split_on_sections) {
1.430     albertel 3239: 		 my $this_section = (split(/:/,$person))[2];
1.429     foxr     3240: 		 if ($this_section ne $last_section) {
                   3241: 		     $i++;
                   3242: 		     $last_section = $this_section;
                   3243: 		 }
                   3244: 	     } else {
                   3245: 		 $i=int($student_counter/$helper->{'VARS'}{'NUMBER_TO_PRINT'});
                   3246: 	     }
1.626     raeburn  3247: 	     my $actual_seq = master_seq_to_person_seq($map, \@master_seq,
1.627.2.15  raeburn  3248:                                                        $person, undef, $nohidemap);
1.375     foxr     3249: 	     my ($output,$fullname, $printed)=&print_resources($r,$helper,
1.353     foxr     3250: 						     $person,$type,
1.595     foxr     3251: 						     \%moreenv,  $actual_seq,
1.360     albertel 3252: 						     $flag_latex_header_remove,
1.422     albertel 3253: 						     $LaTeXwidth);
1.375     foxr     3254: 	     $resources_printed .= ":";
1.284     albertel 3255: 	     $print_array[$i].=$output;
                   3256: 	     $student_names[$i].=$person.':'.$fullname.'_END_';
1.581     bisitz   3257: #	     &Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state,&mt('last student').' '.$fullname);
                   3258: 	     &Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state,'last student');
1.284     albertel 3259: 	     $flag_latex_header_remove = 'YES';
1.627.2.38! raeburn  3260:              if ($printed) {
        !          3261:                  $lastprinted = $fullname;
        !          3262:              }
1.331     albertel 3263: 	     if (&Apache::loncommon::connection_aborted($r)) { last; }
1.284     albertel 3264: 	 }
                   3265: 	 &Apache::lonhtmlcommon::Close_PrgWin($r,\%prog_state);
                   3266: 	 $result .= $print_array[0].'  \end{document}';
1.590     foxr     3267:      } elsif (($print_type eq 'problems_for_anon')      ||
                   3268: 	      ($print_type eq 'problems_for_anon_page') ||
1.627.2.26  raeburn  3269: 	      ($print_type eq 'resources_for_anon')  ) {
1.627.2.38! raeburn  3270:          $selectionmade = 9;
1.373     albertel 3271: 	 my $cdom =$env{'course.'.$env{'request.course.id'}.'.domain'};
                   3272: 	 my $cnum =$env{'course.'.$env{'request.course.id'}.'.num'};
1.288     albertel 3273: 	 my $num_todo=$helper->{'VARS'}->{'NUMBER_TO_PRINT_TOTAL'};
                   3274: 	 my $code_name=$helper->{'VARS'}->{'ANON_CODE_STORAGE_NAME'};
1.292     albertel 3275: 	 my $old_name=$helper->{'VARS'}->{'REUSE_OLD_CODES'};
1.385     foxr     3276: 	 my $single_code = $helper->{'VARS'}->{'SINGLE_CODE'};
1.388     foxr     3277: 	 my $selected_code = $helper->{'VARS'}->{'CODE_SELECTED_FROM_LIST'};
1.381     albertel 3278: 	 my $code_option=$helper->{'VARS'}->{'CODE_OPTION'};
1.627.2.29  raeburn  3279:          my @lines = &Apache::lonnet::get_scantronformat_file();
1.596     raeburn  3280: 	 my ($code_type,$code_length,$bubbles_per_row)=('letter',6,10);
1.542     raeburn  3281: 	 foreach my $line (@lines) {
1.627.2.36  raeburn  3282:              next if (($line =~ /^\#/) || ($line eq ''));
1.627.2.38! raeburn  3283: 	     my ($name,$type,$length,$bubbles_per_item) =
1.596     raeburn  3284:                  (split(/:/,$line))[0,2,4,17];
1.381     albertel 3285: 	     if ($name eq $code_option) {
                   3286: 		 $code_length=$length;
                   3287: 		 if ($type eq 'number') { $code_type = 'number'; }
1.627.2.38! raeburn  3288:                  chomp($bubbles_per_item);
1.596     raeburn  3289:                  if (($bubbles_per_item ne '') && ($bubbles_per_item > 0)) {
1.627.2.38! raeburn  3290:                      $bubbles_per_row = $bubbles_per_item;
1.596     raeburn  3291:                  }
1.381     albertel 3292: 	     }
                   3293: 	 }
1.627.2.32  raeburn  3294:          my $map;
1.625     raeburn  3295:          if ($helper->{VARS}{'symb'}) {
1.626     raeburn  3296:              ($map, my $id, my $resource) =
                   3297:                  &Apache::lonnet::decode_symb($helper->{VARS}{'symb'});
1.625     raeburn  3298:          }
1.288     albertel 3299: 	 my %moreenv = ('textwidth' => &get_textwidth($helper,$LaTeXwidth));
1.353     foxr     3300: 	 $moreenv{'problem_split'}    = $parmhash{'problem_stream_switch'};
1.627.2.38! raeburn  3301:          $moreenv{'suppress_tries'} = $parmhash{'suppress_tries'};
1.420     albertel 3302:          $moreenv{'instructor_comments'}='hide';
1.596     raeburn  3303:          $moreenv{'bubbles_per_row'} = $bubbles_per_row;
1.288     albertel 3304: 	 my $seed=time+($$<<16)+($$);
1.292     albertel 3305: 	 my @allcodes;
                   3306: 	 if ($old_name) {
1.381     albertel 3307: 	     my %result=&Apache::lonnet::get('CODEs',
                   3308: 					     [$old_name,"type\0$old_name"],
                   3309: 					     $cdom,$cnum);
                   3310: 	     $code_type=$result{"type\0$old_name"};
1.292     albertel 3311: 	     @allcodes=split(',',$result{$old_name});
1.336     albertel 3312: 	     $num_todo=scalar(@allcodes);
1.389     foxr     3313: 	 } elsif ($selected_code) { # Selection value is always numeric.
1.388     foxr     3314: 	     $num_todo = 1;
                   3315: 	     @allcodes = ($selected_code);
1.385     foxr     3316: 	 } elsif ($single_code) {
                   3317: 
1.387     foxr     3318: 	     $num_todo    = 1;	# Unconditionally one code to do.
1.385     foxr     3319: 	     # If an alpha code have to convert to numbers so it can be
                   3320: 	     # converted back to letters again :-)
                   3321: 	     #
                   3322: 	     if ($code_type ne 'number') {
                   3323: 		 $single_code = &letters_to_num($single_code);
                   3324: 	     }
                   3325: 	     @allcodes = ($single_code);
1.292     albertel 3326: 	 } else {
                   3327: 	     my %allcodes;
1.299     albertel 3328: 	     srand($seed);
1.292     albertel 3329: 	     for (my $i=0;$i<$num_todo;$i++) {
1.381     albertel 3330: 		 $moreenv{'CODE'}=&get_CODE(\%allcodes,$i,$seed,$code_length,
                   3331: 					    $code_type);
1.292     albertel 3332: 	     }
1.627.2.14  raeburn  3333:              $code_name =~ s/^\s+//;
                   3334:              $code_name =~ s/\s+$//;
1.292     albertel 3335: 	     if ($code_name) {
                   3336: 		 &Apache::lonnet::put('CODEs',
1.381     albertel 3337: 				      {
                   3338: 					$code_name =>join(',',keys(%allcodes)),
                   3339: 					"type\0$code_name" => $code_type
                   3340: 				      },
1.292     albertel 3341: 				      $cdom,$cnum);
                   3342: 	     }
                   3343: 	     @allcodes=keys(%allcodes);
                   3344: 	 }
1.336     albertel 3345: 	 my @master_seq=split /\|\|\|/, $helper->{'VARS'}->{'RESOURCES'};
1.590     foxr     3346: 	 my ($type) = split(/_/,$print_type);
1.452     albertel 3347: 	 &adjust_number_to_print($helper);
1.336     albertel 3348: 	 my $number_per_page=$helper->{'VARS'}->{'NUMBER_TO_PRINT'};
1.587     foxr     3349: 	 if ($number_per_page eq '0' || $number_per_page eq 'all'
                   3350: 	     || $number_per_page eq 'section') {
                   3351: 	     $number_per_page=$num_todo > 0 ? $num_todo : 1;
1.336     albertel 3352: 	 }
1.627.2.38! raeburn  3353: 	 my $flag_latex_header_remove = 'NO';
1.611     www      3354: 	 my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin($r,$num_todo);
1.295     albertel 3355: 	 my $count=0;
1.627.2.15  raeburn  3356:          my $nohidemap;
                   3357:          if ($perm{'pav'} && $perm{'vgr'}) {
1.627.2.38! raeburn  3358:              $nohidemap = 1;
1.627.2.15  raeburn  3359:          }
1.292     albertel 3360: 	 foreach my $code (sort(@allcodes)) {
1.295     albertel 3361: 	     my $file_num=int($count/$number_per_page);
1.627.2.38! raeburn  3362: 	     if ($code_type eq 'number') {
1.381     albertel 3363: 		 $moreenv{'CODE'}=$code;
                   3364: 	     } else {
                   3365: 		 $moreenv{'CODE'}=&num_to_letters($code);
                   3366: 	     }
1.627.2.32  raeburn  3367:              $env{'form.CODE'} = $moreenv{'CODE'};
                   3368:              my $actual_seq = master_seq_to_person_seq($map, \@master_seq,
                   3369:                                                        undef,
                   3370:                                                        $moreenv{'CODE'}, $nohidemap);
                   3371:              delete($env{'form.CODE'});
1.375     foxr     3372: 	     my ($output,$fullname, $printed)=
1.288     albertel 3373: 		 &print_resources($r,$helper,'anonymous',$type,\%moreenv,
1.625     raeburn  3374: 				  $actual_seq,$flag_latex_header_remove,
1.360     albertel 3375: 				  $LaTeXwidth);
1.375     foxr     3376: 	     $resources_printed .= ":";
1.295     albertel 3377: 	     $print_array[$file_num].=$output;
1.288     albertel 3378: 	     &Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state,
                   3379: 				       &mt('last assignment').' '.$fullname);
                   3380: 	     $flag_latex_header_remove = 'YES';
1.295     albertel 3381: 	     $count++;
1.627.2.38! raeburn  3382:              if ($printed) {
        !          3383:                  $lastprinted = $fullname;
        !          3384:              }
1.331     albertel 3385: 	     if (&Apache::loncommon::connection_aborted($r)) { last; }
1.288     albertel 3386: 	 }
                   3387: 	 &Apache::lonhtmlcommon::Close_PrgWin($r,\%prog_state);
                   3388: 	 $result .= $print_array[0].'  \end{document}';
1.627.2.26  raeburn  3389:      } elsif ($print_type eq 'problems_from_directory') {
1.627.2.38! raeburn  3390:     #prints selected problems from the subdirectory
1.151     sakharuk 3391: 	$selectionmade = 6;
                   3392:         my @list_of_files=split /\|\|\|/, $helper->{'VARS'}->{'FILES'};
1.154     sakharuk 3393: 	@list_of_files=sort @list_of_files;
1.627.2.38! raeburn  3394: 	my $flag_latex_header_remove = 'NO';
1.175     sakharuk 3395: 	my $rndseed=time;
1.230     albertel 3396: 	if ($helper->{'VARS'}->{'curseed'}) {
                   3397: 	    $rndseed=$helper->{'VARS'}->{'curseed'};
                   3398: 	}
1.151     sakharuk 3399: 	for (my $i=0;$i<=$#list_of_files;$i++) {
1.521     foxr     3400: 
                   3401: 	    &Apache::lonenc::reset_enc();
                   3402: 
1.152     sakharuk 3403: 	    my $urlp = $list_of_files[$i];
1.253     sakharuk 3404: 	    $urlp=~s|//|/|;
1.152     sakharuk 3405: 	    if ($urlp=~/\//) {
1.353     foxr     3406: 		$form{'problem_split'}=$parmhash{'problem_stream_switch'};
1.175     sakharuk 3407: 		$form{'rndseed'}=$rndseed;
1.603     www      3408: 		$urlp =~ s|^$Apache::lonnet::perlvar{'lonDocRoot'}||;
1.375     foxr     3409: 		$resources_printed .= $urlp.':';
1.515     foxr     3410: 		my $texversion=&ssi_with_retries($urlp, $ssi_retry_count, %form);
1.251     sakharuk 3411: 		if(($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
1.253     sakharuk 3412: 		   ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) {
1.380     foxr     3413: 		    #  Don't permanently pervert %form:
                   3414: 		    my %answerform = %form;
                   3415: 		    $answerform{'grade_target'}='answer';
                   3416: 		    $answerform{'answer_output_mode'}='tex';
                   3417: 		    $answerform{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
                   3418: 		    $answerform{'rndseed'}=$rndseed;
1.375     foxr     3419: 		    $resources_printed .= $urlp.':';
1.515     foxr     3420: 		    my $answer=&ssi_with_retries($urlp, $ssi_retry_count, %answerform);
1.251     sakharuk 3421: 		    if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.627.2.17  raeburn  3422: 			$texversion=~s/(\\keephidden\{ENDOFPROBLEM})/$answer$1/;
1.251     sakharuk 3423: 		    } else {
1.253     sakharuk 3424: 			$texversion=&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
                   3425: 			if ($helper->{'VARS'}->{'construction'} ne '1') {
                   3426: 			    $texversion.='\vskip 0 mm \noindent ';
                   3427: 			    $texversion.=&path_to_problem ($urlp,$LaTeXwidth);
                   3428: 			} else {
1.604     www      3429: 			    $texversion.='\vskip 0 mm \noindent\textbf{'.
1.627.2.6  raeburn  3430:                                          &mt("Printing from Authoring Space: No Title").'}\vskip 0 mm ';
1.604     www      3431: 			    $texversion.=&path_to_problem ($urlp,$LaTeXwidth);
1.253     sakharuk 3432: 			}
                   3433: 			$texversion.='\vskip 1 mm '.$answer.'\end{document}';
1.251     sakharuk 3434: 		    }
1.174     sakharuk 3435: 		}
1.515     foxr     3436:                 #this chunk is responsible for printing the path to problem
                   3437: 
1.603     www      3438: 		my $newurlp=&path_to_problem($urlp,$LaTeXwidth);
1.627.2.18  raeburn  3439: 		$texversion =~ s/(\\begin\{minipage}\{\\textwidth})/$1 $newurlp/;
1.152     sakharuk 3440: 		if ($flag_latex_header_remove ne 'NO') {
                   3441: 		    $texversion = &latex_header_footer_remove($texversion);
                   3442: 		} else {
1.627.2.17  raeburn  3443: 		    $texversion =~ s/\\end\{document}//;
1.216     sakharuk 3444: 		}
                   3445: 		if ($helper->{'VARS'}->{'TABLE_INDEX'} eq 'yes') {
                   3446: 		    $texversion=&IndexCreation($texversion,$urlp);
1.152     sakharuk 3447: 		}
1.219     sakharuk 3448: 		if ($helper->{'VARS'}->{'CONSTR_RESOURSE_URL'} eq 'yes') {
                   3449: 		    $texversion=~s/(\\addcontentsline\{toc\}\{subsection\}\{[^\}]*\})/$1 URL: \\verb|$urlp| \\strut\\\\\\strut /;
1.627.2.38! raeburn  3450: 
1.219     sakharuk 3451: 		}
1.152     sakharuk 3452: 		$result .= $texversion;
                   3453: 	    }
1.627.2.38! raeburn  3454: 	    $flag_latex_header_remove = 'YES';
1.151     sakharuk 3455: 	}
1.175     sakharuk 3456: 	if ($helper->{VARS}->{'construction'} eq '1') {$result=~s/(\\typeout)/ RANDOM SEED IS $rndseed $1/;}
1.152     sakharuk 3457: 	$result .= '\end{document}';      	
1.140     sakharuk 3458:     }
                   3459: #-------------------------------------------------------- corrections for the different page formats
1.499     foxr     3460: 
                   3461:     # Only post process if that has not been turned off e.g. by a raw latex resource.
                   3462: 
                   3463:     if ($do_postprocessing) {
1.627.2.38! raeburn  3464:         my $mostrecent;
        !          3465:         if ((($selectionmade == 5) || ($selectionmade == 8) || ($selectionmade == 9)) &&
        !          3466:             (($numberofcolumns == 1) || ($laystyle eq 'album' && $papersize eq 'a4'))) {
        !          3467:             $mostrecent = $lastprinted;
        !          3468:         }
1.590     foxr     3469: 	$result = &page_format_transformation($papersize,
                   3470: 					      $laystyle,$numberofcolumns,
                   3471: 					      $print_type,$result,
                   3472: 					      $helper->{VARS}->{'assignment'},
                   3473: 					      $helper->{'VARS'}->{'TABLE_CONTENTS'},
                   3474: 					      $helper->{'VARS'}->{'TABLE_INDEX'},
1.627.2.38! raeburn  3475: 					      $selectionmade,$mostrecent);
1.499     foxr     3476: 	$result = &latex_corrections($number_of_columns,$result,$selectionmade,
                   3477: 				     $helper->{'VARS'}->{'ANSWER_TYPE'});
                   3478: 	#if ($numberofcolumns == 1) {
1.451     albertel 3479: 	$result =~ s/\\textwidth\s*=\s*-?\d*\.?\d*\s*(cm|mm|in)/\\textwidth= $helper->{'VARS'}->{'pagesize.width'} $helper->{'VARS'}->{'pagesize.widthunit'} /;
                   3480: 	$result =~ s/\\textheight\s*=?\s*-?\d*\.?\d*\s*(cm|mm|in)/\\textheight $helper->{'VARS'}->{'pagesize.height'} $helper->{'VARS'}->{'pagesize.heightunit'} /;
                   3481: 	$result =~ s/\\evensidemargin\s*=\s*-?\d*\.?\d*\s*(cm|mm|in)/\\evensidemargin= $helper->{'VARS'}->{'pagesize.lmargin'} $helper->{'VARS'}->{'pagesize.lmarginunit'} /;
                   3482: 	$result =~ s/\\oddsidemargin\s*=\s*-?\d*\.?\d*\s*(cm|mm|in)/\\oddsidemargin= $helper->{'VARS'}->{'pagesize.lmargin'} $helper->{'VARS'}->{'pagesize.lmarginunit'} /;
1.499     foxr     3483: 	#}
                   3484:     }
1.367     foxr     3485: 
1.627.2.16  raeburn  3486:     # Set URLback so we can provide a link back to the resource and to change options.
                   3487:     # (Since the browser back button does not currently work with https,
                   3488:     # the back link is useful even when there is an easy-to-miss LON-CAPA back button.)
1.274     sakharuk 3489: 
1.276     sakharuk 3490:     my $URLback=''; #link to original document
1.510     albertel 3491:     if ($helper->{'VARS'}->{'construction'} eq '1') {
1.607     www      3492: 	$URLback=$helper->{'VARS'}->{'filename'};
1.627.2.16  raeburn  3493:     } elsif ($helper->{VARS}{'symb'}) {
                   3494:         my ($map, $id, $url) = &Apache::lonnet::decode_symb($helper->{VARS}{'symb'});
1.627.2.28  raeburn  3495:         my $cdom =$env{'course.'.$env{'request.course.id'}.'.domain'};
                   3496:         my $cnum =$env{'course.'.$env{'request.course.id'}.'.num'};
1.627.2.30  raeburn  3497:         my ($anchor,$usehttp,$plainurl);
1.627.2.20  raeburn  3498:         $url = &Apache::lonnet::clutter($url);
1.627.2.30  raeburn  3499:         $plainurl = $url;
1.627.2.28  raeburn  3500:         if (($ENV{'SERVER_PORT'} == 443) && ($env{'request.course.id'}) &&
                   3501:             (($url =~ m{^\Q/public/$cdom/$cnum/syllabus\E($|\?)}) ||
                   3502:              ($url =~ m{^\Q/adm/wrapper/ext/\E(?!https:)}))) {
1.627.2.31  raeburn  3503:             unless ((&Apache::lonnet::uses_sts()) || (&Apache::lonnet::waf_allssl())) {
1.627.2.28  raeburn  3504:                 $usehttp = 1;
                   3505:             }
                   3506:         }
1.627.2.20  raeburn  3507:         if ($env{'request.enc'}) {
                   3508:             $url = &Apache::lonenc::encrypted($url);
                   3509:         }
1.627.2.21  raeburn  3510:         if ($url ne '') {
1.627.2.28  raeburn  3511:             my $symb = $helper->{VARS}{'symb'};
                   3512:             if ($url =~ m{^\Q/adm/wrapper/ext/\E}) {
                   3513:                 my $link = $url;
                   3514:                 ($link,$anchor) = ($url =~ /^([^\#]+)(?:|(\#[^\#]+))$/);
                   3515:                 if ($anchor) {
                   3516:                     ($symb) = ($helper->{VARS}{'symb'} =~ /^([^\#]+)/);
                   3517:                 }
                   3518:                 $url = $link;
                   3519:             }
                   3520:             $URLback = $url;
                   3521:             if ($usehttp) {
                   3522:                 $URLback .= (($URLback =~ /\?/) ? '&amp;':'?').'usehttp=1';
                   3523:             }
1.627.2.30  raeburn  3524:             unless ($plainurl =~ /\.page$/) {
                   3525:                 $URLback .= (($URLback =~ /\?/) ? '&amp;':'?').'symb='.&escape($symb.$anchor);
                   3526:             }
1.627.2.21  raeburn  3527:         }
1.276     sakharuk 3528:     }
1.556     foxr     3529:     #
                   3530:     # Final adjustment of the font size:
                   3531:     #
                   3532: 
                   3533:     $result = set_font_size($result);
1.375     foxr     3534: 
1.612     foxr     3535:     # Insert any babel headers required.
                   3536: 
                   3537:     $result       = &collect_languages($result);
                   3538: 
                   3539: 
1.627.2.38! raeburn  3540: #-- writing .tex file in prtspool
1.525     www      3541:     my $temp_file;
                   3542:     my $identifier = &Apache::loncommon::get_cgi_id();
                   3543:     my $filename = "/home/httpd/prtspool/$env{'user.name'}_$env{'user.domain'}_printout_$identifier.tex";
1.627.2.38! raeburn  3544:     if (!($#print_array>0)) {
1.525     www      3545:        unless ($temp_file = Apache::File->new('>'.$filename)) {
                   3546: 	  $r->log_error("Couldn't open $filename for output $!");
1.627.2.38! raeburn  3547: 	  return SERVER_ERROR;
1.525     www      3548:        }
                   3549:        print $temp_file $result;
                   3550:        my $begin=index($result,'\begin{document}',0);
1.627.2.38! raeburn  3551:        my $inc=substr($result,0,$begin+16);
1.515     foxr     3552:     } else {
1.525     www      3553:        my $begin=index($result,'\begin{document}',0);
                   3554:        my $inc=substr($result,0,$begin+16);
                   3555:        for (my $i=0;$i<=$#print_array;$i++) {
                   3556: 	  if ($i==0) {
                   3557: 	      $print_array[$i]=$result;
                   3558: 	  } else {
                   3559: 	      $print_array[$i].='\end{document}';
1.627.2.38! raeburn  3560: 	      $print_array[$i] =
1.525     www      3561: 		&latex_corrections($number_of_columns,$print_array[$i],
1.627.2.38! raeburn  3562: 				   $selectionmade,
1.525     www      3563: 				   $helper->{'VARS'}->{'ANSWER_TYPE'});
1.627.2.38! raeburn  3564: 
1.525     www      3565: 	      my $anobegin=index($print_array[$i],'\setcounter{page}',0);
                   3566: 	      substr($print_array[$i],0,$anobegin)='';
                   3567: 	      $print_array[$i]=$inc.$print_array[$i];
                   3568: 	  }
                   3569: 	  my $temp_file;
                   3570: 	  my $newfilename=$filename;
                   3571: 	  my $num=$i+1;
1.627.2.38! raeburn  3572: 	  $newfilename =~s/\.tex$//;
1.525     www      3573: 	  $newfilename=sprintf("%s_%03d.tex",$newfilename, $num);
                   3574: 	  unless ($temp_file = Apache::File->new('>'.$newfilename)) {
                   3575: 	      $r->log_error("Couldn't open $newfilename for output $!");
1.627.2.38! raeburn  3576: 	      return SERVER_ERROR;
1.525     www      3577: 	  }
                   3578: 	  print $temp_file $print_array[$i];
                   3579:        }
                   3580:     }
                   3581:     my $student_names='';
                   3582:     if ($#print_array>0) {
                   3583:         for (my $i=0;$i<=$#print_array;$i++) {
                   3584:   	  $student_names.=$student_names[$i].'_ENDPERSON_';
1.515     foxr     3585: 	}
1.525     www      3586:     } else {
                   3587: 	if ($#student_names>-1) {
                   3588: 	   $student_names=$student_names[0].'_ENDPERSON_';
1.515     foxr     3589: 	} else {
1.525     www      3590:            my $fullname = &get_name($env{'user.name'},$env{'user.domain'});
                   3591: 	   $student_names=join(':',$env{'user.name'},$env{'user.domain'},
1.515     foxr     3592: 				    $env{'request.course.sec'},$fullname).
                   3593: 					'_ENDPERSON_'.'_END_';
                   3594: 	}
1.525     www      3595:      }
1.515     foxr     3596: 	
1.525     www      3597:      # logic for now is too complex to trace if this has been defined
                   3598:      #  yet.
                   3599:      my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
                   3600:      my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
                   3601:      &Apache::lonnet::appenv({'cgi.'.$identifier.'.file'   => $filename,
1.515     foxr     3602: 				'cgi.'.$identifier.'.layout'  => $laystyle,
                   3603: 				'cgi.'.$identifier.'.numcol'  => $numberofcolumns,
                   3604: 				'cgi.'.$identifier.'.paper'  => $papersize,
                   3605: 				'cgi.'.$identifier.'.selection' => $selectionmade,
                   3606: 				'cgi.'.$identifier.'.tableofcontents' => $helper->{'VARS'}->{'TABLE_CONTENTS'},
                   3607: 				'cgi.'.$identifier.'.tableofindex' => $helper->{'VARS'}->{'TABLE_INDEX'},
                   3608: 				'cgi.'.$identifier.'.role' => $perm{'pav'},
                   3609: 				'cgi.'.$identifier.'.numberoffiles' => $#print_array,
                   3610: 				'cgi.'.$identifier.'.studentnames' => $student_names,
1.627.2.20  raeburn  3611: 				'cgi.'.$identifier.'.backref' => &escape($URLback),});
1.525     www      3612:     &Apache::lonnet::appenv({"cgi.$identifier.user"    => $env{'user.name'},
1.515     foxr     3613: 				"cgi.$identifier.domain"  => $env{'user.domain'},
1.627.2.38! raeburn  3614: 				"cgi.$identifier.courseid" => $cnum,
        !          3615: 				"cgi.$identifier.coursedom" => $cdom,
1.520     raeburn  3616: 				"cgi.$identifier.resources" => $resources_printed});
1.515     foxr     3617: 	
1.525     www      3618:     my $end_page = &Apache::loncommon::end_page();
1.529     raeburn  3619:     my $continue_text = &mt('Continue');
1.525     www      3620:     # If there's been an unrecoverable SSI error, report it to the user
                   3621:     if ($ssi_error) {
                   3622:         my $helpurl = &Apache::loncommon::top_nav_help('Helpdesk');
1.554     bisitz   3623:         $r->print('<br /><p class="LC_error">'.&mt('An unrecoverable network error occurred:').'</p><p>'.
1.526     www      3624:                   &mt('At least one of the resources you chose to print could not be rendered due to an unrecoverable error when communicating with a server:').
                   3625:                   '<br />'.$ssi_last_error_resource.'<br />'.$ssi_last_error.
                   3626:                   '</p><p>'.&mt('You can continue using the link provided below, but make sure to carefully inspect your output file! The errors will be marked in the file.').'<br />'.
1.528     raeburn  3627:                   &mt('You may be able to reprint the individual resources for which this error occurred, as the issue may be temporary.').
1.525     www      3628:                   '<br />'.&mt('If the error persists, please contact the [_1] for assistance.',$helpurl).'</p><p>'.
                   3629:                   &mt('We apologize for the inconvenience.').'</p>'.
1.528     raeburn  3630:                   '<a href="/cgi-bin/printout.pl?'.$identifier.'">'.$continue_text.'</a>'.$end_page);
1.525     www      3631:     } else {
1.515     foxr     3632: 	$r->print(<<FINALEND);
1.317     albertel 3633: <br />
1.288     albertel 3634: <meta http-equiv="Refresh" content="0; url=/cgi-bin/printout.pl?$identifier" />
1.528     raeburn  3635: <a href="/cgi-bin/printout.pl?$identifier">$continue_text</a>
1.431     albertel 3636: $end_page
1.140     sakharuk 3637: FINALEND
1.528     raeburn  3638:     }                                     # endif ssi errors.
1.140     sakharuk 3639: }
                   3640: 
1.288     albertel 3641: 
                   3642: sub get_CODE {
1.381     albertel 3643:     my ($all_codes,$num,$seed,$size,$type)=@_;
1.288     albertel 3644:     my $max='1'.'0'x$size;
                   3645:     my $newcode;
                   3646:     while(1) {
1.392     albertel 3647: 	$newcode=sprintf("%0".$size."d",int(rand($max)));
1.288     albertel 3648: 	if (!exists($$all_codes{$newcode})) {
                   3649: 	    $$all_codes{$newcode}=1;
1.381     albertel 3650: 	    if ($type eq 'number' ) {
                   3651: 		return $newcode;
                   3652: 	    } else {
                   3653: 		return &num_to_letters($newcode);
                   3654: 	    }
1.288     albertel 3655: 	}
                   3656:     }
                   3657: }
1.140     sakharuk 3658: 
1.284     albertel 3659: sub print_resources {
1.360     albertel 3660:     my ($r,$helper,$person,$type,$moreenv,$master_seq,$remove_latex_header,
1.422     albertel 3661: 	$LaTeXwidth)=@_;
1.627.2.38! raeburn  3662:     my $current_output = '';
1.375     foxr     3663:     my $printed = '';
1.284     albertel 3664:     my ($username,$userdomain,$usersection) = split /:/,$person;
                   3665:     my $fullname = &get_name($username,$userdomain);
1.492     foxr     3666:     my $namepostfix = "\\\\";	# Both anon and not anon should get the same vspace.
1.600     foxr     3667: 
1.619     foxr     3668: 
1.591     foxr     3669:     #
                   3670:     # Figure out if we need to filter the output by
                   3671:     # the incomplete problems for that person
                   3672:     #
                   3673:     my $print_type = $helper->{'VARS'}->{'PRINT_TYPE'};
                   3674:     my $print_incomplete = 0;
                   3675:     if (($print_type eq 'map_incomplete_problems_people_seq')   ||
                   3676: 	($print_type eq 'incomplete_problems_selpeople_course')) {
                   3677: 	$print_incomplete = 1;
                   3678:     }
1.596     raeburn  3679:     if ($person eq 'anonymous') {
1.613     ramirez  3680: 	$namepostfix .=&mt('Name:')." ";
1.288     albertel 3681: 	$fullname = "CODE - ".$moreenv->{'CODE'};
                   3682:     }
1.590     foxr     3683: 
1.444     foxr     3684:     #  Fullname may have special latex characters that need \ prefixing:
                   3685:     #
                   3686: 
1.350     foxr     3687:     my $i           = 0;
1.591     foxr     3688:     my $actually_printed = 0;	# Count of resources printed.
1.627.2.38! raeburn  3689:     #goes through all resources, checks if they are available for
        !          3690:     #current student, and produces output
1.428     albertel 3691: 
                   3692:     &Apache::lonxml::clear_problem_counter();
1.364     albertel 3693:     my %page_breaks  = &get_page_breaks($helper);
1.476     albertel 3694:     my $columns_in_format = (split(/\|/,$helper->{'VARS'}->{'FORMAT'}))[1];
1.440     foxr     3695:     #
1.627.2.38! raeburn  3696:     #   end each student with a
1.440     foxr     3697:     #   Special that allows the post processor to even out the page
                   3698:     #   counts later.  Nasty problem this... it would be really
                   3699:     #   nice to put the special in as a postscript comment
1.441     foxr     3700:     #   e.g. \special{ps:\ENDOFSTUDENTSTAMP}  unfortunately,
1.440     foxr     3701:     #   The special gets passed the \ and dvips puts it in the output file
1.627.2.38! raeburn  3702:     #   so we will just rely on printout.pl to strip ENDOFSTUDENTSTAMP from the
1.441     foxr     3703:     #   postscript.  Each ENDOFSTUDENTSTAMP will go on a line by itself.
1.440     foxr     3704:     #
1.591     foxr     3705: 
1.568     foxr     3706:     my $syllabus_first = 0;
1.619     foxr     3707:     my $current_assignment = "";
                   3708:     my $assignment;
                   3709:     my $courseidinfo = &get_course();
1.627.2.13  raeburn  3710:     my $possprint = scalar(@{$master_seq});
1.627.2.1  raeburn  3711:     if (defined($courseidinfo)) { $courseidinfo=' - '.$courseidinfo }
                   3712:     if ($usersection ne '') {$courseidinfo.=' - Sec. '.$usersection}
1.619     foxr     3713: 
1.284     albertel 3714:     foreach my $curresline (@{$master_seq})  {
1.351     foxr     3715: 	if (defined $page_breaks{$curresline}) {
1.350     foxr     3716: 	    if($i != 0) {
                   3717: 		$current_output.= "\\newpage\n";
                   3718: 	    }
                   3719: 	}
1.569     foxr     3720: 	$current_output .= &get_extra_vspaces($helper, $curresline);
1.350     foxr     3721: 	$i++;
1.619     foxr     3722: 	my ($map,$id,$res_url) = &Apache::lonnet::decode_symb($curresline);
                   3723: 
                   3724: 	# See if we need to emit a new header:
                   3725: 
1.627.2.38! raeburn  3726: 	if ( !($type eq 'problems' &&
1.609     www      3727: 	       ($curresline!~ m/$LONCAPA::assess_page_re/)) ) {
1.591     foxr     3728: 	    if ($print_incomplete && !&incomplete($username, $userdomain, $res_url)) {
                   3729: 		next;
                   3730: 	    }
                   3731: 	    $actually_printed++; # we're going to print one.
1.627.2.10  raeburn  3732: 
1.284     albertel 3733: 	    if (&Apache::lonnet::allowed('bre',$res_url)) {
1.414     albertel 3734: 		if ($res_url!~m|^ext/|
1.413     albertel 3735: 		    && $res_url=~/\.(problem|exam|quiz|assess|survey|form|library|page|xml|html|htm|xhtml|xhtm)$/) {
1.375     foxr     3736: 		    $printed .= $curresline.':';
1.627.2.38! raeburn  3737: 		    &Apache::lonxml::remember_problem_counter();
1.428     albertel 3738: 
1.526     www      3739: 		    my $rendered = &get_student_view_with_retries($curresline,$ssi_retry_count,$username,$userdomain,$env{'request.course.id'},'tex',$moreenv);
1.582     raeburn  3740:                     if ($res_url =~ /\.page$/) {
                   3741:                         if ($remove_latex_header eq 'NO') {
                   3742:                             if (!($rendered =~ /\\begin\{document\}/)) {
                   3743:                                 $rendered = &print_latex_header().$rendered;
                   3744:                             }
                   3745:                         }
1.591     foxr     3746: ;
1.582     raeburn  3747:                         if ($remove_latex_header eq 'YES') {
                   3748:                             $rendered = &latex_header_footer_remove($rendered);
                   3749:                         } else {
1.627.2.17  raeburn  3750:                             $rendered =~ s/\\end\{document}\d*//;
1.582     raeburn  3751:                         }
                   3752:                     }
1.305     sakharuk 3753: 		    if(($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') ||
                   3754: 		       ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only')) {
1.380     foxr     3755: 			#   Use a copy of the hash so we don't pervert it on future loop passes.
                   3756: 			my %answerenv = %{$moreenv};
                   3757: 			$answerenv{'answer_output_mode'}='tex';
1.591     foxr     3758: 
                   3759: 
1.380     foxr     3760: 			$answerenv{'latex_type'}=$helper->{'VARS'}->{'LATEX_TYPE'};
1.428     albertel 3761: 			
                   3762: 			&Apache::lonxml::restore_problem_counter();
                   3763: 
1.380     foxr     3764: 			my $ansrendered = &Apache::loncommon::get_student_answers($curresline,$username,$userdomain,$env{'request.course.id'},%answerenv);
1.428     albertel 3765: 
1.305     sakharuk 3766: 			if ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'no') {
1.627.2.17  raeburn  3767: 			    $rendered=~s/(\\keephidden\{ENDOFPROBLEM})/$ansrendered$1/;
1.305     sakharuk 3768: 			} else {
1.423     foxr     3769: 			    my $header =&print_latex_header($helper->{'VARS'}->{'LATEX_TYPE'});
1.582     raeburn  3770:                             unless ($helper->{'VARS'}->{'ANSWER_TYPE'} eq 'only') {
1.627.2.17  raeburn  3771:                                 $header =~ s/\\begin\{document}//;     #<<<<<
1.582     raeburn  3772:                             }
1.477     albertel 3773: 			    my $title = &Apache::lonnet::gettitle($curresline);
                   3774: 			    $title = &Apache::lonxml::latex_special_symbols($title);
                   3775: 			    my $body   ='\vskip 0 mm \noindent\textbf{'.$title.'}\vskip 0 mm ';
                   3776: 			    $body     .=&path_to_problem($res_url,$LaTeXwidth);
1.423     foxr     3777: 			    $body     .='\vskip 1 mm '.$ansrendered;
1.627.2.36  raeburn  3778: 			    $body     = &encapsulate_minipage($body,$answerenv{'problem_split'});
1.423     foxr     3779: 			    $rendered = $header.$body;
1.305     sakharuk 3780: 			}
                   3781: 		    }
1.511     foxr     3782: 		    if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   3783: 			my $url = &Apache::lonnet::clutter($res_url);
                   3784: 			my $annotation = &annotate($url);
1.627.2.17  raeburn  3785: 			$rendered =~  s/(\\keephidden\{ENDOFPROBLEM})/$annotation$1/;
1.511     foxr     3786: 		    }
1.562     foxr     3787: 		    my $junk;
1.305     sakharuk 3788: 		    if ($remove_latex_header eq 'YES') {
                   3789: 			$rendered = &latex_header_footer_remove($rendered);
                   3790: 		    } else {
1.627.2.17  raeburn  3791: 			$rendered =~ s/\\end\{document}//;
1.305     sakharuk 3792: 		    }
1.627.2.38! raeburn  3793: 		    $current_output .= $rendered;
1.456     raeburn  3794: 		} elsif ($res_url=~/\/(smppg|syllabus|aboutme|bulletinboard)$/) {
1.568     foxr     3795: 		    if ($i == 1) {
                   3796: 			$syllabus_first = 1;
                   3797: 		    }
1.375     foxr     3798: 		    $printed .= $curresline.':';
1.528     raeburn  3799: 		    my $rendered = &get_student_view_with_retries($curresline,$ssi_retry_count,$username,$userdomain,$env{'request.course.id'},'tex',$moreenv);
1.511     foxr     3800: 		    if ($helper->{'VARS'}->{'PRINT_ANNOTATIONS'} eq 'yes') {
                   3801: 			my $url = &Apache::lonnet::clutter($res_url);
                   3802: 			my $annotation = &annotate($url);
1.627.2.17  raeburn  3803: 			$annotation    =~ s/(\\end\{document})/$annotation$1/;
1.511     foxr     3804: 		    }
1.305     sakharuk 3805: 		    if ($remove_latex_header eq 'YES') {
                   3806: 			$rendered = &latex_header_footer_remove($rendered);
1.284     albertel 3807: 		    } else {
1.627.2.17  raeburn  3808: 			$rendered =~ s/\\end\{document}//;
1.284     albertel 3809: 		    }
1.421     foxr     3810: 		    $current_output .= $rendered.'\vskip 0.5mm\noindent\makebox[\textwidth/$number_of_columns][b]{\hrulefill}\strut \vskip 0 mm \strut ';
1.627.2.34  raeburn  3811: 		} elsif($res_url =~ /\.pdf$/) {
1.552     foxr     3812: 		    my $url = &Apache::lonnet::clutter($res_url);
                   3813: 		    my $rendered  = &include_pdf($url);
                   3814: 		    if ($remove_latex_header ne 'NO') {
                   3815: 			$rendered = &latex_header_footer_remove($rendered);
                   3816: 		    }
                   3817: 		    $current_output .= $rendered;
1.284     albertel 3818: 		} else {
1.414     albertel 3819: 		    my $rendered = &unsupported($res_url,$helper->{'VARS'}->{'LATEX_TYPE'},$curresline);
1.305     sakharuk 3820: 		    if ($remove_latex_header ne 'NO') {
                   3821: 			$rendered = &latex_header_footer_remove($rendered);
                   3822: 		    } else {
1.627.2.17  raeburn  3823: 			$rendered =~ s/\\end\{document}//;
1.305     sakharuk 3824: 		    }
                   3825: 		    $current_output .= $rendered;
1.284     albertel 3826: 		}
                   3827: 	    }
                   3828: 	    $remove_latex_header = 'YES';
1.590     foxr     3829: 	}
1.619     foxr     3830: 	$assignment = &Apache::lonxml::latex_special_symbols(
                   3831: 	    &Apache::lonnet::gettitle($map), 'header');
                   3832: 	if (($assignment ne $current_assignment) && ($assignment ne "")) {
                   3833: 	    my $header_line = &format_page_header($LaTeXwidth, $parmhash{'print_header_format'},
1.627.2.38! raeburn  3834: 						  $assignment, $courseidinfo,
1.627.2.1  raeburn  3835: 						  $fullname);
1.619     foxr     3836: 	    my $header_start = ($columns_in_format == 1) ? '\lhead'
                   3837: 		: '\fancyhead[LO]';
                   3838: 	    $header_line = $header_start.'{'.$header_line.'}';
                   3839: 	    $current_output = $current_output . $header_line;
                   3840: 	    $current_assignment = $assignment;
                   3841: 	}
                   3842: 
1.331     albertel 3843: 	if (&Apache::loncommon::connection_aborted($r)) { last; }
1.284     albertel 3844:     }
1.591     foxr     3845:     # If we are printing incomplete it's possible we don't have
                   3846:     # anything to print.  The print subsystem is not so good at handling
                   3847:     # that so we're going to generate a stub that says there are no
                   3848:     # incomplete resources for the person.
                   3849:     #
1.593     foxr     3850: 
1.591     foxr     3851:     if ($actually_printed == 0) {
1.627.2.13  raeburn  3852:         my $message = &mt('No resources to print');
                   3853:         if (!$possprint) {
                   3854:             if ($perm{'pav'} || $perm{'pfo'}) {
                   3855:                 $message = &mt('There are no unhidden resources to print.')."\n\n".
                   3856:                            &mt('The most likely reason is one of the following: ')."\n".
                   3857:                            '\begin{itemize}'."\n".
                   3858:                            '\item '.&mt("The 'Resource hidden from students' parameter is set for the folder being printed.")."\n".
                   3859:                            '\item '.&mt("'Hidden' is checked in the Course Editor individually for each resource in the folder being printed.")."\n".
                   3860:                            '\end{itemize}'."\n\n".
                   3861:                            &mt("Note: to print a bubblesheet exam which you want to hide from students, ".
                   3862:                                "use the Course Editor to check the 'Hidden' checkbox for the exam folder itself.")."\n";
                   3863:             }
                   3864:         } elsif ($print_incomplete) {
                   3865:             $message = &mt('No incomplete resources');
                   3866:         }
                   3867:         if ($message) {
1.627.2.36  raeburn  3868:             $current_output  = &encapsulate_minipage("\\vskip -10mm \n$message\n \\vskip 100 mm { }\n",$moreenv->{'problem_split'});
1.627.2.13  raeburn  3869:         }
1.593     foxr     3870: 	if ($remove_latex_header eq "NO") {
                   3871: 	    $current_output = &print_latex_header() . $current_output;
                   3872: 	} else {
                   3873: 	    $current_output = &latex_header_footer_remove($current_output);
                   3874: 	}
1.591     foxr     3875:     }
1.552     foxr     3876: 
1.583     raeburn  3877:     if ($syllabus_first) {
                   3878:         $current_output =~ s/\\\\ Last updated:/Last updated:/
                   3879:     }
1.627.2.12  raeburn  3880:     my $currentassignment=&Apache::lonxml::latex_special_symbols($helper->{VARS}->{'assignment'},'header');
                   3881:     my $header_line =
                   3882:         &format_page_header($LaTeXwidth, $parmhash{'print_header_format'},
                   3883:                             $currentassignment, $courseidinfo, $fullname);
                   3884:     my $header_start = ($columns_in_format == 1) ? '\lhead' : '\fancyhead[LO]';
                   3885:     my $newheader = $header_start.'{'.$header_line.'}';
1.583     raeburn  3886:     if ($current_output=~/\\documentclass/) {
1.627.2.17  raeburn  3887: 	$current_output =~ s/\\begin\{document}/\\setlength{\\topmargin}{1cm} \\begin{document}\\noindent\\parbox{\\minipagewidth}{\\noindent$newheader$namepostfix}\\vskip 5 mm /;
1.627.2.38! raeburn  3888: 
1.284     albertel 3889:     } else {
1.627.2.38! raeburn  3890: 	my $blankpages =
1.476     albertel 3891: 	    '\clearpage\strut\clearpage'x$helper->{'VARS'}->{'EMPTY_PAGES'};
1.619     foxr     3892: 	
1.476     albertel 3893: 	$current_output = '\strut\vspace*{-6 mm}\\newline'.
                   3894: 	    &copyright_line().' \newpage '.$blankpages.$end_of_student.
1.627.2.10  raeburn  3895: 	    '\setcounter{page}{1}\noindent\parbox{\minipagewidth}{\noindent'.
                   3896: 	    $newheader.$namepostfix. '} \vskip 5 mm '.$current_output;
1.627.2.38! raeburn  3897: 
1.284     albertel 3898:     }
1.440     foxr     3899:     #
                   3900:     #  Close the student bracketing.
                   3901:     #
1.375     foxr     3902:     return ($current_output,$fullname, $printed);
1.284     albertel 3903: 
                   3904: }
1.140     sakharuk 3905: 
1.614     raeburn  3906: sub printing_blocked {
                   3907:     my ($r,$blocktext) = @_;
                   3908:     my $title = &mt('Preparing Printout');
                   3909:     &Apache::lonhtmlcommon::clear_breadcrumbs();
                   3910:     &Apache::lonhtmlcommon::add_breadcrumb({href=>'/adm/printout',
                   3911:                                             text=> $title});
                   3912:     my $breadcrumbs = &Apache::lonhtmlcommon::breadcrumbs($title);
                   3913:     &Apache::loncommon::content_type($r,'text/html');
                   3914:     &Apache::loncommon::no_cache($r);
                   3915:     $r->send_http_header;
                   3916:     $r->print(&Apache::loncommon::start_page('Preparing Printout').
                   3917:               $breadcrumbs.
                   3918:               $blocktext.
                   3919:               &Apache::loncommon::end_page());
                   3920:     return;
                   3921: }
                   3922: 
1.3       sakharuk 3923: sub handler {
                   3924: 
                   3925:     my $r = shift;
1.614     raeburn  3926: 
                   3927:     if ($env{'request.course.id'}) {
                   3928:         my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
                   3929:         my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
1.627.2.31  raeburn  3930:         my $clientip = &Apache::lonnet::get_requestor_ip($r);
1.627.2.38! raeburn  3931:         my ($blocked,$blocktext) =
1.627.2.31  raeburn  3932:             &Apache::loncommon::blocking_status('printout',$clientip,$cnum,$cdom);
1.614     raeburn  3933:         if ($blocked) {
                   3934:             my $checkrole = "cm./$cdom/$cnum";
                   3935:             if ($env{'request.course.sec'} ne '') {
                   3936:                 $checkrole .= "/$env{'request.course.sec'}";
                   3937:             }
1.627.2.38! raeburn  3938:             unless ((&Apache::lonnet::allowed('evb',undef,undef,$checkrole)) &&
1.614     raeburn  3939:                     ($env{'request.role'} !~ m{^st\./$cdom/$cnum})) {
                   3940:                 &printing_blocked($r,$blocktext);
                   3941:                 return OK;
                   3942:             }
                   3943:         }
                   3944:     }
1.627.2.38! raeburn  3945: 
1.397     albertel 3946:     &init_perm();
                   3947:     my $helper = printHelper($r);
                   3948:     if (!ref($helper)) {
                   3949: 	return $helper;
1.60      sakharuk 3950:     }
1.416     foxr     3951: 
1.350     foxr     3952: 
1.627.2.38! raeburn  3953:     %parmhash=&Apache::lonnet::coursedescription($env{'request.course.id'});
1.350     foxr     3954: 
1.367     foxr     3955:     #  If a figure conversion queue file exists for this user.domain
                   3956:     # we delete it since it can only be bad (if it were good, printout.pl
                   3957:     # would have deleted it the last time around.
                   3958: 
1.373     albertel 3959:     my $conversion_queuefile = "/home/httpd/prtspool/$env{'user.name'}_$env{'user.domain'}_printout.dat";
1.367     foxr     3960:     if(-e $conversion_queuefile) {
                   3961: 	unlink $conversion_queuefile;
                   3962:     }
1.515     foxr     3963: 
1.184     sakharuk 3964:     &output_data($r,$helper,\%parmhash);
1.2       sakharuk 3965:     return OK;
1.614     raeburn  3966: }
1.2       sakharuk 3967: 
1.131     bowersj2 3968: use Apache::lonhelper;
1.130     sakharuk 3969: 
1.223     bowersj2 3970: sub addMessage {
                   3971:     my $text = shift;
                   3972:     my $paramHash = Apache::lonhelper::getParamHash();
                   3973:     $paramHash->{MESSAGE_TEXT} = $text;
                   3974:     Apache::lonhelper::message->new();
                   3975: }
                   3976: 
1.416     foxr     3977: 
1.238     bowersj2 3978: 
1.397     albertel 3979: sub init_perm {
                   3980:     undef(%perm);
                   3981:     $perm{'pav'}=&Apache::lonnet::allowed('pav',$env{'request.course.id'});
                   3982:     if (!$perm{'pav'}) {
                   3983: 	$perm{'pav'}=&Apache::lonnet::allowed('pav',
                   3984: 		  $env{'request.course.id'}.'/'.$env{'request.course.sec'});
                   3985:     }
1.465     albertel 3986:     $perm{'pfo'}=&Apache::lonnet::allowed('pfo',$env{'request.course.id'});
1.397     albertel 3987:     if (!$perm{'pfo'}) {
                   3988: 	$perm{'pfo'}=&Apache::lonnet::allowed('pfo',
                   3989: 		  $env{'request.course.id'}.'/'.$env{'request.course.sec'});
                   3990:     }
1.585     raeburn  3991:     $perm{'vgr'}=&Apache::lonnet::allowed('vgr',$env{'request.course.id'});
                   3992:     if (!$perm{'vgr'}) {
                   3993:         $perm{'vgr'}=&Apache::lonnet::allowed('vgr',
                   3994:                    $env{'request.course.id'}.'/'.$env{'request.course.sec'});
                   3995:     }
1.397     albertel 3996: }
                   3997: 
1.507     albertel 3998: sub get_randomly_ordered_warning {
                   3999:     my ($helper,$map) = @_;
                   4000: 
                   4001:     my $message;
                   4002: 
                   4003:     my $postdata = $env{'form.postdata'} || $helper->{VARS}{'postdata'};
                   4004:     my $navmap = Apache::lonnavmaps::navmap->new();
1.547     raeburn  4005:     if (defined($navmap)) {
                   4006:         my $res = $navmap->getResourceByUrl($map);
                   4007:         if ($res) {
1.627.2.38! raeburn  4008: 	    my $func =
1.547     raeburn  4009: 	        sub { return ($_[0]->is_map() && $_[0]->randomorder); };
                   4010: 	    my @matches = $navmap->retrieveResources($res, $func,1,1,1);
1.610     foxr     4011: 
1.547     raeburn  4012:         }
                   4013:     } else {
1.627.2.38! raeburn  4014:         $message = "Retrieval of information about ordering of resources failed.";
1.547     raeburn  4015:         return '<message type="warning">'.$message.'</message>';
1.507     albertel 4016:     }
                   4017:     return;
                   4018: }
                   4019: 
1.131     bowersj2 4020: sub printHelper {
1.115     bowersj2 4021:     my $r = shift;
                   4022: 
                   4023:     if ($r->header_only) {
1.373     albertel 4024:         if ($env{'browser.mathml'}) {
1.241     www      4025:             &Apache::loncommon::content_type($r,'text/xml');
1.131     bowersj2 4026:         } else {
1.241     www      4027:             &Apache::loncommon::content_type($r,'text/html');
1.131     bowersj2 4028:         }
                   4029:         $r->send_http_header;
                   4030:         return OK;
1.115     bowersj2 4031:     }
                   4032: 
1.131     bowersj2 4033:     # Send header, nocache
1.373     albertel 4034:     if ($env{'browser.mathml'}) {
1.241     www      4035:         &Apache::loncommon::content_type($r,'text/xml');
1.115     bowersj2 4036:     } else {
1.241     www      4037:         &Apache::loncommon::content_type($r,'text/html');
1.115     bowersj2 4038:     }
                   4039:     &Apache::loncommon::no_cache($r);
                   4040:     $r->send_http_header;
                   4041:     $r->rflush();
                   4042: 
1.131     bowersj2 4043:     # Unfortunately, this helper is so complicated we have to
                   4044:     # write it by hand
                   4045: 
                   4046:     Apache::loncommon::get_unprocessed_cgi($ENV{QUERY_STRING});
1.627.2.38! raeburn  4047: 
1.176     bowersj2 4048:     my $helper = Apache::lonhelper::helper->new("Printing Helper");
1.146     bowersj2 4049:     $helper->declareVar('symb');
1.627.2.38! raeburn  4050:     $helper->declareVar('postdata');
        !          4051:     $helper->declareVar('curseed');
        !          4052:     $helper->declareVar('probstatus');
1.156     bowersj2 4053:     $helper->declareVar('filename');
                   4054:     $helper->declareVar('construction');
1.178     sakharuk 4055:     $helper->declareVar('assignment');
1.262     sakharuk 4056:     $helper->declareVar('style_file');
1.340     foxr     4057:     $helper->declareVar('student_sort');
1.363     foxr     4058:     $helper->declareVar('FINISHPAGE');
1.366     foxr     4059:     $helper->declareVar('PRINT_TYPE');
1.372     foxr     4060:     $helper->declareVar("showallfoils");
1.483     foxr     4061:     $helper->declareVar("STUDENTS");
1.569     foxr     4062:     $helper->declareVar("EXTRASPACE");
1.518     foxr     4063: 
1.627.2.38! raeburn  4064: 
1.518     foxr     4065: 
                   4066: 
1.569     foxr     4067:     #  The page breaks and extra spaces
                   4068:     #  can get loaded initially from the course environment:
1.394     foxr     4069:     # But we only do this in the initial state so that they are allowed to change.
                   4070:     #
1.366     foxr     4071: 
1.627.2.38! raeburn  4072: 
1.363     foxr     4073:     &Apache::loncommon::restore_course_settings('print',
1.366     foxr     4074: 						{'pagebreaks'  => 'scalar',
1.569     foxr     4075: 						 'extraspace'  => 'scalar',
1.570     foxr     4076: 						 'extraspace_units' => 'scalar',
1.366     foxr     4077: 					         'lastprinttype' => 'scalar'});
1.627.2.38! raeburn  4078: 
1.483     foxr     4079:     # This will persistently load in the data we want from the
                   4080:     # very first screen.
1.627.2.38! raeburn  4081: 
1.394     foxr     4082:     if($helper->{VARS}->{PRINT_TYPE} eq $env{'form.lastprinttype'}) {
                   4083: 	if (!defined ($env{"form.CURRENT_STATE"})) {
1.627.2.38! raeburn  4084: 
1.394     foxr     4085: 	    $helper->{VARS}->{FINISHPAGE} = $env{'form.pagebreaks'};
1.569     foxr     4086: 	    $helper->{VARS}->{EXTRASPACE} = $env{'form.extraspace'};
1.570     foxr     4087: 	    $helper->{VARS}->{EXTRASPACE_UNITS} = $env{'form.extraspace_units'};
1.394     foxr     4088: 	} else {
                   4089: 	    my $state = $env{"form.CURRENT_STATE"};
                   4090: 	    if ($state eq "START") {
                   4091: 		$helper->{VARS}->{FINISHPAGE} = $env{'form.pagebreaks'};
1.569     foxr     4092: 		$helper->{VARS}->{EXTRASPACE} = $env{'form.extraspace'};
1.570     foxr     4093: 		$helper->{VARS}->{EXTRASPACE_UNITS} = $env{'form.extraspace_units'};
                   4094: 		
1.394     foxr     4095: 	    }
                   4096: 	}
1.366     foxr     4097:     }
1.481     albertel 4098: 
1.156     bowersj2 4099:     # Detect whether we're coming from construction space
1.602     www      4100:     if ($env{'form.postdata'}=~m{^/priv}) {
                   4101:         $helper->{VARS}->{'filename'} = $env{'form.postdata'};
1.156     bowersj2 4102:         $helper->{VARS}->{'construction'} = 1;
1.481     albertel 4103:     } else {
1.373     albertel 4104:         if ($env{'form.postdata'}) {
                   4105:             $helper->{VARS}->{'symb'} = &Apache::lonnet::symbread($env{'form.postdata'});
1.482     albertel 4106: 	    if ( $helper->{VARS}->{'symb'} eq '') {
                   4107: 		$helper->{VARS}->{'postdata'} = $env{'form.postdata'};
                   4108: 	    }
1.156     bowersj2 4109:         }
1.373     albertel 4110:         if ($env{'form.symb'}) {
                   4111:             $helper->{VARS}->{'symb'} = $env{'form.symb'};
1.156     bowersj2 4112:         }
1.373     albertel 4113:         if ($env{'form.url'}) {
1.156     bowersj2 4114:             $helper->{VARS}->{'symb'} = &Apache::lonnet::symbread($helper->{VARS}->{'postdata'});
                   4115:         }
1.157     bowersj2 4116:     }
1.481     albertel 4117: 
1.373     albertel 4118:     if ($env{'form.symb'}) {
                   4119:         $helper->{VARS}->{'symb'} = $env{'form.symb'};
1.146     bowersj2 4120:     }
1.373     albertel 4121:     if ($env{'form.url'}) {
1.140     sakharuk 4122:         $helper->{VARS}->{'symb'} = &Apache::lonnet::symbread($helper->{VARS}->{'postdata'});
1.153     sakharuk 4123: 
1.140     sakharuk 4124:     }
1.627.2.26  raeburn  4125:     if ($helper->{VARS}->{'symb'} ne '') {
                   4126:         $helper->{VARS}->{'symb'}=
                   4127: 	    &Apache::lonenc::check_encrypt($helper->{VARS}->{'symb'});
                   4128:     }
1.335     albertel 4129:     my ($resourceTitle,$sequenceTitle,$mapTitle) = &details_for_menu($helper);
1.178     sakharuk 4130:     if ($sequenceTitle ne '') {$helper->{VARS}->{'assignment'}=$sequenceTitle;}
1.481     albertel 4131: 
1.146     bowersj2 4132:     # Extract map
                   4133:     my $symb = $helper->{VARS}->{'symb'};
1.156     bowersj2 4134:     my ($map, $id, $url);
                   4135:     my $subdir;
1.483     foxr     4136:     my $is_published=0;		# True when printing from resource space.
1.627.2.38! raeburn  4137:     my $res_printable = 1;	# By default the current resource is printable.
1.627.2.25  raeburn  4138:     my $res_error;
1.589     foxr     4139:     my $userCanPrint = ($perm{'pav'} || $perm{'pfo'});
1.615     foxr     4140:     my $res_printstartdate;
                   4141:     my $res_printenddate;
1.616     foxr     4142:     my $map_open = 0;
                   4143:     my $map_close = 0xffffffff;
                   4144:     my $course_open = 0;
                   4145:     my $course_close = 0xffffffff;
1.156     bowersj2 4146: 
                   4147:     # Get the resource name from construction space
                   4148:     if ($helper->{VARS}->{'construction'}) {
1.627.2.38! raeburn  4149:         $resourceTitle = substr($helper->{VARS}->{'filename'},
1.156     bowersj2 4150:                                 rindex($helper->{VARS}->{'filename'}, '/')+1);
                   4151:         $subdir = substr($helper->{VARS}->{'filename'},
                   4152:                          0, rindex($helper->{VARS}->{'filename'}, '/') + 1);
1.481     albertel 4153:     } else {
1.562     foxr     4154: 	# From course space:
                   4155: 
1.482     albertel 4156: 	if ($symb ne '') {
                   4157: 	    ($map, $id, $url) = &Apache::lonnet::decode_symb($symb);
1.627.2.38! raeburn  4158: 	    $helper->{VARS}->{'postdata'} =
1.482     albertel 4159: 		&Apache::lonenc::check_encrypt(&Apache::lonnet::clutter($url));
1.627.2.25  raeburn  4160:             if (!$userCanPrint) {
                   4161: 	        my $navmap = Apache::lonnavmaps::navmap->new();
                   4162:                 if (ref($navmap)) {
                   4163: 	            my $res = $navmap->getBySymb($symb);
                   4164:                     if (ref($res)) {
                   4165: 	                $res_printable = $res->resprintable(); #printability in course context
                   4166: 	                ($res_printstartdate, $res_printenddate) = &get_print_dates($res);
                   4167: 	                ($course_open, $course_close) = &course_print_dates($res);
                   4168: 	                ($map_open, $map_close) = &map_print_dates($res);
                   4169:                     } else {
                   4170:                         $res_error = 1;
                   4171:                     }
                   4172:                 } else {
                   4173:                     $res_error = 1;
                   4174:                 }
                   4175:             }
1.482     albertel 4176: 	} else {
1.589     foxr     4177: 	    # Resource space.
                   4178: 
1.482     albertel 4179: 	    $url = $helper->{VARS}->{'postdata'};
1.483     foxr     4180: 	    $is_published=1;	# From resource space.
1.482     albertel 4181: 	}
                   4182: 	$url = &Apache::lonnet::clutter($url);
1.156     bowersj2 4183:         if (!$resourceTitle) { # if the resource doesn't have a title, use the filename
1.238     bowersj2 4184:             my $postdata = $helper->{VARS}->{'postdata'};
                   4185:             $resourceTitle = substr($postdata, rindex($postdata, '/') + 1);
1.156     bowersj2 4186:         }
                   4187:         $subdir = &Apache::lonnet::filelocation("", $url);
1.589     foxr     4188: 
                   4189: 
1.128     bowersj2 4190:     }
1.373     albertel 4191:     if (!$helper->{VARS}->{'curseed'} && $env{'form.curseed'}) {
                   4192: 	$helper->{VARS}->{'curseed'}=$env{'form.curseed'};
1.230     albertel 4193:     }
1.616     foxr     4194: 
1.373     albertel 4195:     if (!$helper->{VARS}->{'probstatus'} && $env{'form.problemtype'}) {
1.512     foxr     4196: 	$helper->{VARS}->{'probstatus'}=$env{'form.problemstatus'};
1.290     sakharuk 4197:     }
1.115     bowersj2 4198: 
1.192     bowersj2 4199:     my $userCanSeeHidden = Apache::lonnavmaps::advancedUser();
                   4200: 
1.481     albertel 4201:     Apache::lonhelper::registerHelperTags();
1.119     bowersj2 4202: 
1.131     bowersj2 4203:     # "Delete everything after the last slash."
1.119     bowersj2 4204:     $subdir =~ s|/[^/]+$||;
                   4205: 
1.131     bowersj2 4206:     # What can be printed is a very dynamic decision based on
                   4207:     # lots of factors. So we need to dynamically build this list.
                   4208:     # To prevent security leaks, states are only added to the wizard
                   4209:     # if they can be reached, which ensures manipulating the form input
                   4210:     # won't allow anyone to reach states they shouldn't have permission
                   4211:     # to reach.
                   4212: 
                   4213:     # printChoices is tracking the kind of printing the user can
                   4214:     # do, and will be used in a choices construction later.
                   4215:     # In the meantime we will be adding states and elements to
                   4216:     # the helper by hand.
                   4217:     my $printChoices = [];
                   4218:     my $paramHash;
1.130     sakharuk 4219: 
1.589     foxr     4220:     # If there is a current resource and it is printable
                   4221:     # Give that as a choice.
                   4222: 
                   4223:     if ($resourceTitle && $res_printable) {
1.458     www      4224:         push @{$printChoices}, ["<b><i>$resourceTitle</i></b> (".&mt('the resource you just saw on the screen').")", 'current_document', 'PAGESIZE'];
1.627.2.38! raeburn  4225:     }
1.156     bowersj2 4226: 
1.238     bowersj2 4227:     # Useful filter strings
1.589     foxr     4228: 
                   4229:     my $isPrintable = ' && $res->resprintable()';
                   4230: 
1.590     foxr     4231:     my $isProblem = '(($res->is_problem()||$res->contains_problem() ||$res->is_practice()))';
1.589     foxr     4232:     $isProblem .= $isPrintable unless $userCanPrint;
1.238     bowersj2 4233:     $isProblem .= ' && !$res->randomout()' if !$userCanSeeHidden;
1.589     foxr     4234:     my $isProblemOrMap = '($res->is_problem() || $res->contains_problem() || $res->is_sequence() || $res->is_practice())';
                   4235:     $isProblemOrMap .= $isPrintable unless $userCanPrint;
                   4236:     my $isNotMap = '(!$res->is_sequence())';
                   4237:     $isNotMap .= $isPrintable unless $userCanPrint;
1.238     bowersj2 4238:     $isNotMap .= ' && !$res->randomout()' if !$userCanSeeHidden;
1.589     foxr     4239:     my $isMap = '$res->is_map()';
                   4240:     $isMap .= $isPrintable unless $userCanPrint;
                   4241:     my $symbFilter = '$res->shown_symb() ';
1.342     albertel 4242:     my $urlValue = '$res->link()';
1.238     bowersj2 4243: 
                   4244:     $helper->declareVar('SEQUENCE');
                   4245: 
1.465     albertel 4246:     # If we're in a sequence...
1.416     foxr     4247: 
1.465     albertel 4248:     my $start_new_option;
                   4249:     if ($perm{'pav'}) {
1.627.2.38! raeburn  4250: 	$start_new_option =
1.627.2.7  raeburn  4251: 	    "<option text='".&mt('Start new page[_1]before selected','<br />').
1.569     foxr     4252: 	    "' variable='FINISHPAGE' />".
1.627.2.7  raeburn  4253: 	    "<option text='".&mt('Extra space[_1]before selected','<br />').
1.569     foxr     4254: 	    "' variable='EXTRASPACE' type='text' />" .
                   4255: 	    "<option " .
1.570     foxr     4256: 	    "' variable='POSSIBLE_RESOURCES' type='hidden' />".
1.627.2.7  raeburn  4257: 	    "<option text='".&mt('Space units[_1]check for mm','<br />').
1.570     foxr     4258: 	    "' variable='EXTRASPACE_UNITS' type='checkbox' />"
                   4259: 	    ;
1.465     albertel 4260:     }
1.238     bowersj2 4261: 
1.562     foxr     4262:     # If not construction space user can print the components of a page:
                   4263: 
                   4264:     my $page_ispage;
                   4265:     my $page_title;
                   4266:     if (!$helper->{VARS}->{'construction'}) {
                   4267: 	my $varspostdata = $helper->{VARS}->{'postdata'};
                   4268: 	my $varsassignment = $helper->{VARS}->{'assignment'};
                   4269: 	my $page_navmap         = Apache::lonnavmaps::navmap->new();
1.563     foxr     4270: 	if (defined($page_navmap)) {
                   4271: 	    my @page_resources      = $page_navmap->retrieveResources($url);
                   4272: 	    if(defined($page_resources[0])) {
                   4273: 		$page_ispage       = $page_resources[0]->is_page();
                   4274: 		$page_title     = $page_resources[0]->title();
                   4275: 		my $resourcesymb   = $page_resources[0]->symb();
                   4276: 		my ($pagemap, $pageid, $pageurl) = &Apache::lonnet::decode_symb($symb);
                   4277: 		if ($page_ispage) {
1.627.2.38! raeburn  4278: 		    push @{$printChoices},
        !          4279: 		    [&mt('Selected [_1]Problems[_2] from page [_3]', '<b>', '</b>', '<b><i>'.$page_title.'</i></b>'),
        !          4280: 		     'map_problems_in_page',
1.563     foxr     4281: 		     'CHOOSE_PROBLEMS_PAGE'];
1.627.2.38! raeburn  4282: 		    push @{$printChoices},
        !          4283: 		    [&mt('Selected [_1]Resources[_2] from page [_3]', '<b>', '</b>', '<b><i>'.$page_title.'</i></b>'),
        !          4284: 		     'map_resources_in_page',
1.563     foxr     4285: 		     'CHOOSE_RESOURCES_PAGE'];
                   4286: 		}
1.562     foxr     4287:         my $helperFragment = &generate_resource_chooser('CHOOSE_PROBLEMS_PAGE',
                   4288: 							'Select Problem(s) to print',
1.627.2.24  raeburn  4289: 							"multichoice='1' toponly='1' addstatus='1' closeallpages='1' modallink='1'",
1.562     foxr     4290: 							'RESOURCES',
                   4291: 							'PAGESIZE',
                   4292: 							$url,
                   4293: 							$isProblem, '',  $symbFilter,
                   4294: 							$start_new_option);
                   4295: 
                   4296: 
                   4297:       $helperFragment .= &generate_resource_chooser('CHOOSE_RESOURCES_PAGE',
                   4298: 						    'Select Resource(s) to print',
1.627.2.35  raeburn  4299: 						    'multichoice="1" toponly="1" addstatus="1" closeallpages="1" modallink="1" suppressNavmap="1"',
1.562     foxr     4300: 						    'RESOURCES',
                   4301: 						    'PAGESIZE',
                   4302: 						    $url,
                   4303: 						    $isNotMap, '', $symbFilter,
                   4304: 						    $start_new_option);
                   4305: 
1.627.2.38! raeburn  4306: 
1.562     foxr     4307: 
                   4308: 
                   4309: 
                   4310: 	&Apache::lonxml::xmlparse($r, 'helper', $helperFragment);
                   4311: 	
1.563     foxr     4312: 	    }
                   4313: 	}
1.562     foxr     4314:     }
                   4315: 
                   4316:     if (($helper->{'VAR'}->{'construction'} ne '1' ) &&
1.243     bowersj2 4317: 	$helper->{VARS}->{'postdata'} &&
                   4318: 	$helper->{VARS}->{'assignment'}) {
1.590     foxr     4319: 
                   4320: 	# BZ 5209 - Print incomplete problems from sequence:
                   4321: 	# the exact form of this depends on whether or not we are privileged or a mere
                   4322: 	# plebe of s student:
                   4323: 
                   4324: 	my $printSelector = 'map_incomplete_problems_seq';
                   4325: 	my $nextState     = 'CHOOSE_INCOMPLETE_SEQ';
                   4326: 	my $textSuffix    = '';
                   4327: 
1.616     foxr     4328: 	if ($userCanPrint)  {
1.590     foxr     4329: 	    $printSelector = 'map_incomplete_problems_people_seq';
                   4330: 	    $nextState     = 'CHOOSE_INCOMPLETE_PEOPLE_SEQ';
                   4331: 	    $textSuffix    = ' for selected students';
                   4332: 	    my $helperStates =
1.627.2.38! raeburn  4333: 		&create_incomplete_folder_selstud_helper($helper, $map);
1.590     foxr     4334: 	    &Apache::lonxml::xmlparse($r, 'helper', $helperStates);
                   4335: 	} else {
1.616     foxr     4336: 	    if (&printable($map_open, $map_close)) {
                   4337: 		my $helperStates = &create_incomplete_folder_helper($helper, $map); # Create needed states for student.
                   4338: 		&Apache::lonxml::xmlparse($r, 'helper', $helperStates);
                   4339: 	    } else {
                   4340: 		# TODO: Figure out how to break the news...this folder is not printable.
                   4341: 	    }
1.590     foxr     4342: 	}
                   4343: 
1.616     foxr     4344: 	if ($userCanPrint || &printable($map_open, $map_close)) {
                   4345: 	    push(@{$printChoices},
                   4346: 		 [&mt('Selected  [_1]Incomplete Problems[_2] from folder [_3]' . $textSuffix,
                   4347: 		      '<b>', '</b>',
                   4348: 		      '<b><i>'. $sequenceTitle . '</b></i>'),
                   4349: 		  $printSelector,
                   4350: 		  $nextState]);
                   4351: 	}
1.131     bowersj2 4352:         # Allow problems from sequence
1.616     foxr     4353: 	if ($userCanPrint || &printable($map_open, $map_close)) {
1.627.2.38! raeburn  4354: 	    push @{$printChoices},
        !          4355: 	    [&mt('Selected [_1]Problems[_2] from folder [_3]','<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>'),
        !          4356: 	     'map_problems',
1.562     foxr     4357: 	     'CHOOSE_PROBLEMS'];
1.616     foxr     4358: 	    # Allow all resources from sequence
1.627.2.38! raeburn  4359: 	    push @{$printChoices}, [&mt('Selected [_1]Resources[_2] from folder [_3]','<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>'),
        !          4360: 				    'map_problems_pages',
1.616     foxr     4361: 				    'CHOOSE_PROBLEMS_HTML'];
                   4362: 	    my $helperFragment = &generate_resource_chooser('CHOOSE_PROBLEMS',
                   4363: 							    'Select Problem(s) to print',
1.627.2.24  raeburn  4364: 							    'multichoice="1" toponly="1" addstatus="1" closeallpages="1" modallink="1"',
1.616     foxr     4365: 							    'RESOURCES',
                   4366: 							    'PAGESIZE',
                   4367: 							    $map,
1.621     foxr     4368: 							    $isProblem, '',
1.616     foxr     4369: 							    $symbFilter,
                   4370: 							    $start_new_option);
                   4371: 	    $helperFragment .= &generate_resource_chooser('CHOOSE_PROBLEMS_HTML',
                   4372: 							  'Select Resource(s) to print',
1.627.2.35  raeburn  4373: 							  'multichoice="1" toponly="1" addstatus="1" closeallpages="1" modallink="1" suppressNavmap="1"',
1.616     foxr     4374: 							  'RESOURCES',
                   4375: 							  'PAGESIZE',
                   4376: 							  $map,
                   4377: 							  $isNotMap, '',
                   4378: 							  $symbFilter,
                   4379: 							  $start_new_option);
1.627.2.38! raeburn  4380: 
1.616     foxr     4381: 	    &Apache::lonxml::xmlparse($r, 'helper', $helperFragment);
                   4382: 	} else {
                   4383: 	    # TODO: Figure out how to tell them the folder is not printable.
                   4384: 	}
1.121     bowersj2 4385:     }
1.627.2.38! raeburn  4386: 	# If the user has pfo (print for others) allow them to print all
1.616     foxr     4387: 	# problems and resources  in the entire course, optionally for selected students
                   4388: 	my $post_data = $helper->{VARS}->{'postdata'};
1.627.2.38! raeburn  4389: 
1.483     foxr     4390:     if ($perm{'pfo'} &&  !$is_published  &&
1.627.2.38! raeburn  4391:         ($post_data=~/\/res\// || $post_data =~/\/(syllabus|smppg|aboutme|bulletinboard)$/)) {
1.481     albertel 4392: 
1.590     foxr     4393: 	# BZ 5209 - incomplete problems from entire course:
                   4394: 
                   4395: 	push(@{$printChoices},
1.594     foxr     4396: 	     [&mtn('Selected <b>Incomplete Problems</b> from <b>entire course</b> for selected people'),
1.590     foxr     4397: 	      'incomplete_problems_selpeople_course', 'INCOMPLETE_PROBLEMS_COURSE_RESOURCES']);
                   4398: 	my $helperFragment = &create_incomplete_course_helper($helper); # Create needed states.
                   4399: 
                   4400: 	&Apache::lonxml::xmlparse($r, 'helper', $helperFragment);
                   4401: 
                   4402: 	#  Selected problems/resources from entire course:
                   4403: 
1.509     albertel 4404:         push @{$printChoices}, [&mtn('Selected <b>Problems</b> from <b>entire course</b>'), 'all_problems', 'ALL_PROBLEMS'];
                   4405: 	push @{$printChoices}, [&mtn('Selected <b>Resources</b> from <b>entire course</b>'), 'all_resources', 'ALL_RESOURCES'];
1.536     foxr     4406: 	push @{$printChoices}, [&mtn('Selected <b>Problems</b> from <b>entire course</b> for <b>selected people</b>'), 'all_problems_students', 'ALL_PROBLEMS_STUDENTS'];
1.562     foxr     4407: my $suffixXml = <<ALL_PROBLEMS;
1.536     foxr     4408:   <state name="STUDENTS1" title="Select People">
                   4409:       <message><b>Select sorting order of printout</b> </message>
                   4410:     <choices variable='student_sort'>
                   4411:       <choice computer='0'>Sort by section then student</choice>
                   4412:       <choice computer='1'>Sort by students across sections.</choice>
                   4413:     </choices>
                   4414:       <message><br /><hr /><br /> </message>
                   4415:       <student multichoice='1' variable="STUDENTS" nextstate="PRINT_FORMATTING" coursepersonnel="1"/>
                   4416:   </state>
1.284     albertel 4417: ALL_PROBLEMS
1.627.2.38! raeburn  4418:          &Apache::lonxml::xmlparse($r, 'helper',
1.562     foxr     4419: 				   &generate_resource_chooser('ALL_PROBLEMS',
1.627.2.7  raeburn  4420: 							      'Select Problem(s) to print',
1.627.2.24  raeburn  4421: 							      'multichoice="1" suppressEmptySequences="0" addstatus="1" closeallpages="1" modallink="1"',
1.562     foxr     4422: 							      'RESOURCES',
                   4423: 							      'PAGESIZE',
                   4424: 							      '',
                   4425: 							      $isProblemOrMap, $isNotMap,
                   4426: 							      $symbFilter,
                   4427: 							      $start_new_option) .
                   4428: 				   &generate_resource_chooser('ALL_RESOURCES',
                   4429: 							      'Select Resource(s) to print',
1.627.2.35  raeburn  4430: 							      'toponly="0" multichoice="1" suppressEmptySequences="0" addstatus="1" closeallpages="1" modallink="1" suppressNavmap="1"',
1.562     foxr     4431: 							      'RESOURCES',
                   4432: 							      'PAGESIZE',
                   4433: 							      '',
                   4434: 							      $isNotMap,'',$symbFilter,
                   4435: 							      $start_new_option) .
                   4436: 				   &generate_resource_chooser('ALL_PROBLEMS_STUDENTS',
                   4437: 							      'Select Problem(s) to print',
1.627.2.24  raeburn  4438: 							      'toponly="0" multichoice="1" suppressEmptySequences="0" addstatus="1" closeallpages="1" modallink="1"',
1.562     foxr     4439: 							      'RESOURCES',
                   4440: 							      'STUDENTS1',
                   4441: 							      '',
                   4442: 							      $isProblemOrMap,'' , $symbFilter,
                   4443: 							      $start_new_option) .
                   4444: 				     $suffixXml
                   4445: 				   );
1.132     bowersj2 4446: 
1.284     albertel 4447: 	if ($helper->{VARS}->{'assignment'}) {
1.562     foxr     4448: 
                   4449: 	    # If we were looking at a page, allow a selection of problems from the page
                   4450: 	    # either for selected students or for coded assignments.
                   4451: 
                   4452: 	    if ($page_ispage) {
                   4453: 		push @{$printChoices}, [&mt('Selected [_1]Problems[_2] from page [_3] for [_4]selected people[_5]',
                   4454: 					    '<b>', '</b>', '<b><i>'.$page_title.'</i></b>', '<b>', '</b>'),
                   4455: 					'problems_for_students_from_page', 'CHOOSE_TGT_STUDENTS_PAGE'];
                   4456: 		push @{$printChoices}, [&mt('Selected [_1]Problems[_2] from page [_3] for [_4]CODEd assignments[_5]',
                   4457: 					    '<b>', '</b>', '<b><i>'.$page_title.'</i></b>', '<b>', '</b>'),
                   4458: 					'problems_for_anon_page', 'CHOOSE_ANON1_PAGE'];
                   4459: 	    }
                   4460: 	    push @{$printChoices}, [&mt('Selected [_1]Problems[_2] from folder [_3] for [_4]selected people[_5]',
1.627.2.38! raeburn  4461: 					'<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>','<b>','</b>'),
1.562     foxr     4462: 				    'problems_for_students', 'CHOOSE_STUDENTS'];
                   4463: 	    push @{$printChoices}, [&mt('Selected [_1]Problems[_2] from folder [_3] for [_4]CODEd assignments[_5]',
1.627.2.38! raeburn  4464: 					'<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>','<b>','</b>'),
1.562     foxr     4465: 				    'problems_for_anon', 'CHOOSE_ANON1'];
1.284     albertel 4466: 	}
1.424     foxr     4467: 
1.627.2.38! raeburn  4468: 	my $randomly_ordered_warning =
1.591     foxr     4469:             &get_randomly_ordered_warning($helper, $map);
1.507     albertel 4470: 
1.424     foxr     4471: 	# resource_selector will hold a few states that:
                   4472: 	#   - Allow resources to be selected for printing.
                   4473: 	#   - Determine pagination between assignments.
                   4474: 	#   - Determine how many assignments should be bundled into a single PDF.
                   4475:         # TODO:
                   4476: 	#    Probably good to do things like separate this up into several vars, each
                   4477: 	#    with one state, and use REGEXPs at inclusion time to set state names
                   4478: 	#    and next states for better mix and match capability
                   4479: 	#
1.562     foxr     4480: 	my $resource_selector= &generate_resource_chooser('SELECT_PROBLEMS',
                   4481: 							  'Select resources to print',
1.627.2.35  raeburn  4482: 							  'multichoice="1" addstatus="1" closeallpages="1" modallink="1" suppressNavmap="1"',
1.627.2.38! raeburn  4483: 							  'RESOURCES',
1.562     foxr     4484: 							  'PRINT_FORMATTING',
1.571     droeschl 4485: 							  $map,
1.562     foxr     4486: 							  $isProblem, '', $symbFilter,
                   4487: 							  $start_new_option);
1.586     raeburn  4488: 	$resource_selector .=  &generate_format_selector($helper,
1.598     raeburn  4489:                                                          'How should results be printed?',
                   4490:                                                          'PRINT_FORMATTING').
1.586     raeburn  4491:                                &generate_resource_chooser('CHOOSE_STUDENTS_PAGE',
1.562     foxr     4492: 							'Select Problem(s) to print',
1.627.2.24  raeburn  4493: 							"multichoice='1' addstatus='1' closeallpages ='1' modallink='1'",
1.562     foxr     4494: 							'RESOURCES',
                   4495: 							'PRINT_FORMATTING',
                   4496: 							$url,
                   4497: 							$isProblem, '',  $symbFilter,
                   4498: 							$start_new_option);
                   4499: 
                   4500: 
                   4501: # Generate student choosers.
                   4502: 
                   4503: 
1.284     albertel 4504: 
1.562     foxr     4505:         &Apache::lonxml::xmlparse($r, 'helper',
                   4506: 				  &generate_student_chooser('CHOOSE_TGT_STUDENTS_PAGE',
                   4507: 							    'student_sort',
                   4508: 							    'STUDENTS',
                   4509: 							    'CHOOSE_STUDENTS_PAGE'));
1.627.2.38! raeburn  4510: 	&Apache::lonxml::xmlparse($r, 'helper',
1.562     foxr     4511: 				  &generate_student_chooser('CHOOSE_STUDENTS',
                   4512: 							    'student_sort',
                   4513: 							    'STUDENTS',
                   4514: 							    'SELECT_PROBLEMS'));
                   4515: 	&Apache::lonxml::xmlparse($r, 'helper', $resource_selector);
1.292     albertel 4516: 
1.373     albertel 4517: 	my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
                   4518: 	my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
1.292     albertel 4519:         my @names=&Apache::lonnet::getkeys('CODEs',$cdom,$cnum);
                   4520: 	my $namechoice='<choice></choice>';
1.337     albertel 4521: 	foreach my $name (sort {uc($a) cmp uc($b)} @names) {
1.294     albertel 4522: 	    if ($name =~ /^error: 2 /) { next; }
1.381     albertel 4523: 	    if ($name =~ /^type\0/) { next; }
1.292     albertel 4524: 	    $namechoice.='<choice computer="'.$name.'">'.$name.'</choice>';
                   4525: 	}
1.389     foxr     4526: 
                   4527: 
                   4528: 	my %code_values;
1.405     albertel 4529: 	my %codes_to_print;
1.411     albertel 4530: 	foreach my $key (@names) {
1.389     foxr     4531: 	    %code_values = &Apache::grades::get_codes($key, $cdom, $cnum);
1.405     albertel 4532: 	    foreach my $key (keys(%code_values)) {
                   4533: 		$codes_to_print{$key} = 1;
1.388     foxr     4534: 	    }
                   4535: 	}
1.389     foxr     4536: 
1.452     albertel 4537: 	my $code_selection;
1.405     albertel 4538: 	foreach my $code (sort {uc($a) cmp uc($b)} (keys(%codes_to_print))) {
1.389     foxr     4539: 	    my $choice  = $code;
                   4540: 	    if ($code =~ /^[A-Z]+$/) { # Alpha code
                   4541: 		$choice = &letters_to_num($code);
                   4542: 	    }
1.432     albertel 4543: 	    push(@{$helper->{DATA}{ALL_CODE_CHOICES}},[$code,$choice]);
1.388     foxr     4544: 	}
1.436     albertel 4545: 	if (%codes_to_print) {
1.627.2.38! raeburn  4546: 	    $code_selection .='
1.472     albertel 4547: 	    <message><b>Choose single CODE from list:</b></message>
1.448     albertel 4548: 		<message></td><td></message>
1.452     albertel 4549: 		<dropdown variable="CODE_SELECTED_FROM_LIST" multichoice="0" allowempty="0">
                   4550:                   <choice></choice>
1.448     albertel 4551:                   <exec>
                   4552:                      push(@{$state->{CHOICES}},@{$helper->{DATA}{ALL_CODE_CHOICES}});
                   4553:                   </exec>
1.452     albertel 4554: 		</dropdown>
1.468     foxr     4555: 	    <message></td></tr><tr><td></message>
1.436     albertel 4556:             '.$/;
1.448     albertel 4557: 
1.436     albertel 4558: 	}
1.432     albertel 4559: 
1.627.2.29  raeburn  4560:         my @lines = &Apache::lonnet::get_scantronformat_file();
1.381     albertel 4561: 	my $codechoice='';
1.542     raeburn  4562: 	foreach my $line (@lines) {
1.627.2.36  raeburn  4563:             next if (($line =~ /^\#/) || ($line eq ''));
1.381     albertel 4564: 	    my ($name,$description,$code_type,$code_length)=
                   4565: 		(split(/:/,$line))[0,1,2,4];
1.627.2.38! raeburn  4566: 	    if ($code_length > 0 &&
1.381     albertel 4567: 		$code_type =~/^(letter|number|-1)/) {
                   4568: 		$codechoice.='<choice computer="'.$name.'">'.$description.'</choice>';
                   4569: 	    }
                   4570: 	}
                   4571: 	if ($codechoice eq '') {
                   4572: 	    $codechoice='<choice computer="default">Default</choice>';
                   4573: 	}
1.627.2.38! raeburn  4574: 	my $anon1 = &generate_code_selector($helper,
1.562     foxr     4575: 					    'CHOOSE_ANON1',
                   4576: 					    'SELECT_PROBLEMS',
                   4577: 					    $codechoice,
                   4578: 					    $code_selection,
                   4579: 					    $namechoice) . $resource_selector;
1.627.2.38! raeburn  4580: 
        !          4581: 
1.562     foxr     4582:         &Apache::lonxml::xmlparse($r, 'helper',$anon1);
                   4583: 
                   4584: 	my $anon_page = &generate_code_selector($helper,
                   4585: 						'CHOOSE_ANON1_PAGE',
                   4586: 						'SELECT_PROBLEMS_PAGE',
                   4587: 						$codechoice,
                   4588: 						$code_selection,
                   4589: 						$namechoice) .
                   4590: 			&generate_resource_chooser('SELECT_PROBLEMS_PAGE',
                   4591: 						   'Select Problem(s) to print',
1.627.2.24  raeburn  4592: 						   "multichoice='1' addstatus='1' closeallpages ='1' modallink='1'",
1.562     foxr     4593: 						   'RESOURCES',
                   4594: 						   'PRINT_FORMATTING',
                   4595: 						   $url,
                   4596: 						   $isProblem, '',  $symbFilter,
                   4597: 						   $start_new_option);
                   4598: 	&Apache::lonxml::xmlparse($r, 'helper', $anon_page);
1.254     sakharuk 4599: 
1.272     sakharuk 4600: 
1.254     sakharuk 4601: 	if ($helper->{VARS}->{'assignment'}) {
1.590     foxr     4602: 
                   4603: 	    # Assignment printing:
                   4604: 
1.546     bisitz   4605: 	    push @{$printChoices}, [&mt('Selected [_1]Resources[_2] from folder [_3] for [_4]selected people[_5]','<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>','<b>','</b>'), 'resources_for_students', 'CHOOSE_STUDENTS1'];
                   4606: 	    push @{$printChoices}, [&mt('Selected [_1]Resources[_2] from folder [_3] for [_4]CODEd assignments[_5]','<b>','</b>','<b><i>'.$sequenceTitle.'</i></b>','<b>','</b>'), 'resources_for_anon', 'CHOOSE_ANON2'];
1.254     sakharuk 4607: 	}
1.627.2.38! raeburn  4608: 
1.284     albertel 4609: 
                   4610: 	$resource_selector=<<RESOURCE_SELECTOR;
1.424     foxr     4611:     <state name="SELECT_RESOURCES" title="Select Resources">
1.507     albertel 4612:     $randomly_ordered_warning
1.424     foxr     4613:     <nextstate>PRINT_FORMATTING</nextstate>
1.254     sakharuk 4614:     <message><br /><big><i><b>Select resources for the assignment</b></i></big><br /></message>
1.627.2.38! raeburn  4615:     <resource variable="RESOURCES" multichoice="1" addstatus="1"
1.627.2.24  raeburn  4616:               closeallpages="1" modallink="1">
1.254     sakharuk 4617:       <filterfunc>return $isNotMap;</filterfunc>
                   4618:       <mapurl>$map</mapurl>
                   4619:       <valuefunc>return $symbFilter;</valuefunc>
1.465     albertel 4620:       $start_new_option
1.254     sakharuk 4621:       </resource>
1.424     foxr     4622:     </state>
1.284     albertel 4623: RESOURCE_SELECTOR
                   4624: 
1.586     raeburn  4625:         $resource_selector .= &generate_format_selector($helper,
                   4626:                                                         'Format of the print job',
1.598     raeburn  4627:                                                         'PRINT_FORMATTING');
1.284     albertel 4628: 	&Apache::lonxml::xmlparse($r, 'helper', <<CHOOSE_STUDENTS1);
                   4629:   <state name="CHOOSE_STUDENTS1" title="Select Students and Resources">
1.340     foxr     4630:     <choices variable='student_sort'>
                   4631:       <choice computer='0'>Sort by section then student</choice>
                   4632:       <choice computer='1'>Sort by students across sections.</choice>
                   4633:     </choices>
1.437     foxr     4634:     <message><br /><hr /><br /></message>
1.426     foxr     4635:     <student multichoice='1' variable="STUDENTS" nextstate="SELECT_RESOURCES" coursepersonnel="1" />
1.340     foxr     4636: 
1.424     foxr     4637:     </state>
1.284     albertel 4638:     $resource_selector
1.254     sakharuk 4639: CHOOSE_STUDENTS1
                   4640: 
1.284     albertel 4641: 	&Apache::lonxml::xmlparse($r, 'helper', <<CHOOSE_ANON2);
1.472     albertel 4642:   <state name="CHOOSE_ANON2" title="Select CODEd Assignments">
1.424     foxr     4643:     <nextstate>SELECT_RESOURCES</nextstate>
1.472     albertel 4644:     <message><h4>Fill out one of the forms below</h4></message>
                   4645:     <message><br /><hr /> <br /></message>
                   4646:     <message><h3>Generate new CODEd Assignments</h3></message>
                   4647:     <message><table><tr><td><b>Number of CODEd assignments to print:</b></td><td></message>
1.579     foxr     4648:     <string variable="NUMBER_TO_PRINT_TOTAL" maxlength="5" size="5"  noproceed="1">
1.362     albertel 4649:        <validator>
                   4650: 	if (((\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}+0) < 1) &&
1.386     foxr     4651: 	    !\$helper->{'VARS'}{'REUSE_OLD_CODES'}                &&
1.388     foxr     4652: 	    !\$helper->{'VARS'}{'SINGLE_CODE'}                   &&
                   4653: 	    !\$helper->{'VARS'}{'CODE_SELECTED_FROM_LIST'}) {
1.362     albertel 4654: 	    return "You need to specify the number of assignments to print";
                   4655: 	}
1.578     foxr     4656:         if (((\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}+0) >= 1)  &&
                   4657:              (\$helper->{'VARS'}{'SINGLE_CODE'} ne '') ) {
                   4658:             return 'Specifying number of codes to print and a specific code is not compatible';
                   4659:         }
1.362     albertel 4660: 	return undef;
                   4661:        </validator>
                   4662:     </string>
                   4663:     <message></td></tr><tr><td></message>
1.501     albertel 4664:     <message><b>Names to save the CODEs under for later:</b></message>
1.412     albertel 4665:     <message></td><td></message>
                   4666:     <string variable="ANON_CODE_STORAGE_NAME" maxlength="50" size="20" />
                   4667:     <message></td></tr><tr><td></message>
1.599     raeburn  4668:     <message><b>Bubblesheet type:</b></message>
1.412     albertel 4669:     <message></td><td></message>
                   4670:     <dropdown variable="CODE_OPTION" multichoice="0" allowempty="0">
                   4671:     $codechoice
                   4672:     </dropdown>
1.472     albertel 4673:     <message></td></tr><tr><td></table></message>
                   4674:     <message><br /><hr /><h3>Print a Specific CODE </h3><br /><table></message>
                   4675:     <message><tr><td><b>Enter a CODE to print:</b></td><td></message>
1.412     albertel 4676:     <string variable="SINGLE_CODE" size="10">
1.386     foxr     4677:         <validator>
                   4678: 	   if(!\$helper->{'VARS'}{'NUMBER_TO_PRINT_TOTAL'}           &&
1.388     foxr     4679: 	      !\$helper->{'VARS'}{'REUSE_OLD_CODES'}                 &&
                   4680: 	      !\$helper->{'VARS'}{'CODE_SELECTED_FROM_LIST'}) {
1.386     foxr     4681: 	      return &Apache::lonprintout::is_code_valid(\$helper->{'VARS'}{'SINGLE_CODE'},
                   4682: 						      \$helper->{'VARS'}{'CODE_OPTION'});
1.577     foxr     4683: 	  } elsif (\$helper->{'VARS'}{'SINGLE_CODE'} ne ''){
1.578     foxr     4684: 	      return 'Specifying a code name is incompatible specifying number of codes.';
1.386     foxr     4685: 	   } else {
                   4686: 	       return undef;	# Other forces control us.
                   4687: 	   }
                   4688:         </validator>
                   4689:     </string>
1.472     albertel 4690:     <message></td></tr><tr><td></message>
1.432     albertel 4691:         $code_selection
1.472     albertel 4692:     <message></td></tr></table></message>
                   4693:     <message><hr /><h3>Reprint a Set of Saved CODEs</h3><table><tr><td></message>
                   4694:     <message><b>Select saved CODEs:</b></message>
1.381     albertel 4695:     <message></td><td></message>
1.294     albertel 4696:     <dropdown variable="REUSE_OLD_CODES">
                   4697:         $namechoice
                   4698:     </dropdown>
1.412     albertel 4699:     <message></td></tr></table></message>
1.424     foxr     4700:   </state>
1.284     albertel 4701:     $resource_selector
                   4702: CHOOSE_ANON2
1.481     albertel 4703:     }
                   4704: 
1.121     bowersj2 4705:     # FIXME: That RE should come from a library somewhere.
1.483     foxr     4706:     if (($perm{'pav'} 
1.482     albertel 4707: 	&& $subdir ne $Apache::lonnet::perlvar{'lonDocRoot'}.'/res/'
                   4708: 	&& (defined($helper->{'VARS'}->{'construction'})
                   4709: 	    ||
                   4710: 	    (&Apache::lonnet::allowed('bre',$subdir) eq 'F'
                   4711: 	     && 
                   4712: 	     $helper->{VARS}->{'postdata'}=~/\.(problem|exam|quiz|assess|survey|form|library|page|xml|html|htm|xhtml|xhtm)/)
1.483     foxr     4713: 	    )) 
                   4714: 	&& $helper->{VARS}->{'assignment'} eq ""
1.482     albertel 4715: 	) {
                   4716: 	my $pretty_dir = &Apache::lonnet::hreflocation($subdir);
1.546     bisitz   4717:         push @{$printChoices}, [&mt('Selected [_1]Problems[_2] from current subdirectory [_3]','<b>','</b>','<b><i>'.$pretty_dir.'</i></b>','<b>','</b>'), 'problems_from_directory', 'CHOOSE_FROM_SUBDIR'];
1.139     bowersj2 4718:         my $xmlfrag = <<CHOOSE_FROM_SUBDIR;
1.482     albertel 4719:   <state name="CHOOSE_FROM_SUBDIR" title="Select File(s) from <b><small>$pretty_dir</small></b> to print">
1.458     www      4720: 
1.138     bowersj2 4721:     <files variable="FILES" multichoice='1'>
1.144     bowersj2 4722:       <nextstate>PAGESIZE</nextstate>
1.138     bowersj2 4723:       <filechoice>return '$subdir';</filechoice>
1.139     bowersj2 4724: CHOOSE_FROM_SUBDIR
1.627.2.38! raeburn  4725: 
1.238     bowersj2 4726:         # this is broken up because I really want interpolation above,
                   4727:         # and I really DON'T want it below
1.139     bowersj2 4728:         $xmlfrag .= <<'CHOOSE_FROM_SUBDIR';
1.225     bowersj2 4729:       <filefilter>return Apache::lonhelper::files::not_old_version($filename) &&
                   4730: 	  $filename =~ m/\.(problem|exam|quiz|assess|survey|form|library)$/;
1.131     bowersj2 4731:       </filefilter>
1.138     bowersj2 4732:       </files>
1.131     bowersj2 4733:     </state>
                   4734: CHOOSE_FROM_SUBDIR
1.139     bowersj2 4735:         &Apache::lonxml::xmlparse($r, 'helper', $xmlfrag);
1.131     bowersj2 4736:     }
1.238     bowersj2 4737: 
                   4738:     # Allow the user to select any sequence in the course, feed it to
                   4739:     # another resource selector for that sequence
1.483     foxr     4740:     if (!$helper->{VARS}->{'construction'} && !$is_published) {
1.509     albertel 4741: 	push @$printChoices, [&mtn("Selected <b>Resources</b> from <b>selected folder</b> in course"),
1.249     sakharuk 4742: 			      'select_sequences', 'CHOOSE_SEQUENCE'];
1.244     bowersj2 4743: 	my $escapedSequenceName = $helper->{VARS}->{'SEQUENCE'};
                   4744: 	#Escape apostrophes and backslashes for Perl
                   4745: 	$escapedSequenceName =~ s/\\/\\\\/g;
                   4746: 	$escapedSequenceName =~ s/'/\\'/g;
1.239     bowersj2 4747: 	&Apache::lonxml::xmlparse($r, 'helper', <<CHOOSE_FROM_ANY_SEQUENCE);
1.238     bowersj2 4748:   <state name="CHOOSE_SEQUENCE" title="Select Sequence To Print From">
                   4749:     <message>Select the sequence to print resources from:</message>
                   4750:     <resource variable="SEQUENCE">
                   4751:       <nextstate>CHOOSE_FROM_ANY_SEQUENCE</nextstate>
1.616     foxr     4752:       <filterfunc>return &Apache::lonprintout::printable_sequence(\$res);</filterfunc>
1.238     bowersj2 4753:       <valuefunc>return $urlValue;</valuefunc>
1.447     foxr     4754:       <choicefunc>return \$res->hasResource(\$res,sub { return !\$_[0]->is_sequence() },0,0);
1.391     foxr     4755: 	</choicefunc>
1.238     bowersj2 4756:       </resource>
                   4757:     </state>
                   4758:   <state name="CHOOSE_FROM_ANY_SEQUENCE" title="Select Resources To Print">
                   4759:     <message>(mark desired resources then click "next" button) <br /></message>
1.435     foxr     4760:     <resource variable="RESOURCES" multichoice="1" toponly='1' addstatus="1"
1.627.2.35  raeburn  4761:               closeallpages="1" modallink="1" suppressNavmap="1">
1.238     bowersj2 4762:       <nextstate>PAGESIZE</nextstate>
1.466     albertel 4763:       <filterfunc>return $isNotMap</filterfunc>
1.244     bowersj2 4764:       <mapurl evaluate='1'>return '$escapedSequenceName';</mapurl>
1.238     bowersj2 4765:       <valuefunc>return $symbFilter;</valuefunc>
1.465     albertel 4766:       $start_new_option
1.238     bowersj2 4767:       </resource>
                   4768:     </state>
                   4769: CHOOSE_FROM_ANY_SEQUENCE
1.239     bowersj2 4770: }
1.481     albertel 4771: 
1.627.2.26  raeburn  4772:     my $numchoices = 0;
                   4773:     if (ref($printChoices) eq 'ARRAY') {
                   4774:         $numchoices = @{$printChoices};
                   4775:     }
                   4776:     # Early out if nothing to print
                   4777:     if (!$numchoices) {
                   4778:         $r->print(&Apache::loncommon::start_page('Printing Helper').
                   4779:                   '<h2>'.&mt('Unable to determine print context').'</h2>'.
                   4780:                   '<p>'.&mt('Please display a resource, and then click the "Print" button/icon').'</p>');
                   4781:         my $prtspool=$r->dir_config('lonPrtDir');
                   4782:         my $footer = &recently_generated($prtspool);
                   4783:         $r->print($footer.&Apache::loncommon::end_page());
                   4784:         return OK;
                   4785:     }
                   4786: 
1.131     bowersj2 4787:     # Generate the first state, to select which resources get printed.
1.223     bowersj2 4788:     Apache::lonhelper::state->new("START", "Select Printing Options:");
1.615     foxr     4789:     if (!$res_printable) {
1.627.2.9  raeburn  4790:         my $noprintmsg;
1.627.2.25  raeburn  4791:         if ($res_error) {
                   4792:             $noprintmsg = &mt('Print availability for current resource could not be determined');
                   4793:         } else {
                   4794:             my $now = time;
                   4795:             my $shownprintstart = &Apache::lonlocal::locallocaltime($res_printstartdate);
                   4796:             my $shownprintend = &Apache::lonlocal::locallocaltime($res_printenddate);
                   4797:             if (($res_printenddate) && ($res_printenddate < $now)) {
1.627.2.9  raeburn  4798:                 $noprintmsg = &mt('Printing for current resource no longer available (ended: [_1])',
                   4799:                                   $shownprintend);
1.627.2.25  raeburn  4800:             } else {
                   4801:                 if (($res_printstartdate) && ($res_printstartdate > $now)) {
                   4802:                     if (($res_printenddate) && ($res_printenddate > $now) && ($res_printenddate > $res_printstartdate)) {
                   4803:                         $noprintmsg = &mt('Printing for current resource is only possible between [_1] and [_2]',
                   4804:                                           $shownprintstart,$shownprintend);
                   4805:                     } elsif (!$res_printenddate) {
                   4806:                         $noprintmsg = &mt('Printing for current resource will only be possible starting [_1]',
                   4807:                                           $shownprintstart);
                   4808:                     } else {
                   4809:                         $noprintmsg = &mt('Printing for current resource is unavailable');
                   4810:                     }
1.627.2.9  raeburn  4811:                 }
                   4812:             }
                   4813:         }
                   4814: 
                   4815:         if ($noprintmsg) {
                   4816:             $paramHash = Apache::lonhelper::getParamHash();
                   4817:             $paramHash->{MESSAGE_TEXT} =
                   4818:                 '<p class="LC_info">'.$noprintmsg.'</p>';
                   4819:             Apache::lonhelper::message->new();
                   4820:         }
1.615     foxr     4821:     }
                   4822:     $paramHash = Apache::lonhelper::getParamHash();
1.131     bowersj2 4823:     $paramHash = Apache::lonhelper::getParamHash();
1.155     sakharuk 4824:     $paramHash->{MESSAGE_TEXT} = "";
1.131     bowersj2 4825:     Apache::lonhelper::message->new();
                   4826:     $paramHash = Apache::lonhelper::getParamHash();
                   4827:     $paramHash->{'variable'} = 'PRINT_TYPE';
                   4828:     $paramHash->{CHOICES} = $printChoices;
                   4829:     Apache::lonhelper::choices->new();
1.161     bowersj2 4830: 
1.223     bowersj2 4831:     my $startedTable = 0; # have we started an HTML table yet? (need
                   4832:                           # to close it later)
                   4833: 
1.585     raeburn  4834:     if (($perm{'pav'} and $perm{'vgr'}) or 
1.170     sakharuk 4835: 	($helper->{VARS}->{'construction'} eq '1')) {
1.544     bisitz   4836: 	&addMessage('<br />'
                   4837:                    .'<h3>'.&mt('Print Options').'</h3>'
                   4838:                    .&Apache::lonhtmlcommon::start_pick_box()
                   4839:                    .&Apache::lonhtmlcommon::row_title(
                   4840:                        '<label for="ANSWER_TYPE_forminput">'
                   4841:                       .&mt('Print Answers')
                   4842:                       .'</label>'
                   4843:                     )
                   4844:         );
1.161     bowersj2 4845:         $paramHash = Apache::lonhelper::getParamHash();
1.627.2.38! raeburn  4846: 	$paramHash->{'variable'} = 'ANSWER_TYPE';
        !          4847: 	$helper->declareVar('ANSWER_TYPE');
1.161     bowersj2 4848:         $paramHash->{CHOICES} = [
1.242     sakharuk 4849:                                    ['Without Answers', 'yes'],
                   4850:                                    ['With Answers', 'no'],
1.368     albertel 4851:                                    ['Only Answers', 'only']
1.289     sakharuk 4852:                                 ];
1.210     sakharuk 4853:         Apache::lonhelper::dropdown->new();
1.544     bisitz   4854: 	&addMessage(&Apache::lonhtmlcommon::row_closure());
1.223     bowersj2 4855: 	$startedTable = 1;
1.556     foxr     4856: 
                   4857: #
                   4858: #  Select font size.
                   4859: #
                   4860: 
                   4861:             $helper->declareVar('fontsize');
                   4862:             &addMessage(&Apache::lonhtmlcommon::row_title(&mt('Font Size')));
                   4863:             my $xmlfrag = << "FONT_SELECTION";
                   4864: 
1.627.2.38! raeburn  4865: 
1.627.2.27  raeburn  4866:             <dropdown variable='fontsize' multichoice='0' allowempty='0'>
1.556     foxr     4867:             <defaultvalue>
                   4868: 		  return 'normalsize';
                   4869:             </defaultvalue>
                   4870:             <choice computer='tiny'>Tiny</choice>
                   4871:             <choice computer='sub/superscriptsize'>Script Size</choice>
                   4872:             <choice computer='footnotesize'>Footnote Size</choice>
                   4873:             <choice computer='small'>Small</choice>
                   4874:             <choice computer='normalsize'>Normal (default)</choice>
                   4875:             <choice computer='large'>larger than normal</choice>
                   4876:             <choice computer='Large'>Even larger than normal</choice>
                   4877:             <choice computer='LARGE'>Still larger than normal</choice>
                   4878:             <choice computer='huge'>huge font size</choice>
                   4879:             <choice computer='Huge'>Largest possible size</choice>
                   4880:             </dropdown>
                   4881: FONT_SELECTION
                   4882:             &Apache::lonxml::xmlparse($r, 'helper', $xmlfrag);
                   4883:             &addMessage(&Apache::lonhtmlcommon::row_closure(1));
1.161     bowersj2 4884:     }
1.209     sakharuk 4885: 
1.397     albertel 4886:     if ($perm{'pav'}) {
1.223     bowersj2 4887: 	if (!$startedTable) {
1.497     www      4888: 	    addMessage("<hr width='33%' /><table><tr><td align='right'>".
                   4889:                        '<label for="LATEX_TYPE_forminput">'.
                   4890:                        &mt('LaTeX mode').
                   4891:                        "</label>: </td><td>");
1.223     bowersj2 4892: 	    $startedTable = 1;
                   4893: 	} else {
1.544     bisitz   4894: 	    &addMessage(&Apache::lonhtmlcommon::row_title(
                   4895:                            '<label for="LATEX_TYPE_forminput">'
                   4896:                            .&mt('LaTeX mode')
                   4897:                            .'</label>'
                   4898:                         )
                   4899:             );
1.223     bowersj2 4900: 	}
1.203     sakharuk 4901:         $paramHash = Apache::lonhelper::getParamHash();
1.627.2.38! raeburn  4902: 	$paramHash->{'variable'} = 'LATEX_TYPE';
        !          4903: 	$helper->declareVar('LATEX_TYPE');
        !          4904: 	if ($helper->{VARS}->{'construction'} eq '1') {
1.203     sakharuk 4905: 	    $paramHash->{CHOICES} = [
1.627.2.38! raeburn  4906: 				     ['standard LaTeX mode', 'standard'],
1.223     bowersj2 4907: 				     ['LaTeX batchmode', 'batchmode'], ];
1.203     sakharuk 4908: 	} else {
                   4909: 	    $paramHash->{CHOICES} = [
1.223     bowersj2 4910: 				     ['LaTeX batchmode', 'batchmode'],
                   4911: 				     ['standard LaTeX mode', 'standard'] ];
1.203     sakharuk 4912: 	}
1.210     sakharuk 4913:         Apache::lonhelper::dropdown->new();
1.627.2.38! raeburn  4914: 
1.544     bisitz   4915: 	&addMessage(&Apache::lonhtmlcommon::row_closure()
                   4916:                    .&Apache::lonhtmlcommon::row_title(
                   4917:                         '<label for="TABLE_CONTENTS_forminput">'
                   4918:                        .&mt('Print Table of Contents')
                   4919:                        .'</label>'
                   4920:                     )
                   4921:         );
1.209     sakharuk 4922:         $paramHash = Apache::lonhelper::getParamHash();
1.627.2.38! raeburn  4923: 	$paramHash->{'variable'} = 'TABLE_CONTENTS';
        !          4924: 	$helper->declareVar('TABLE_CONTENTS');
1.209     sakharuk 4925:         $paramHash->{CHOICES} = [
1.223     bowersj2 4926:                                    ['No', 'no'],
                   4927:                                    ['Yes', 'yes'] ];
1.210     sakharuk 4928:         Apache::lonhelper::dropdown->new();
1.544     bisitz   4929: 	&addMessage(&Apache::lonhtmlcommon::row_closure());
1.627.2.38! raeburn  4930: 
1.220     sakharuk 4931: 	if (not $helper->{VARS}->{'construction'}) {
1.545     bisitz   4932: 	    &addMessage(&Apache::lonhtmlcommon::row_title(
                   4933:                             '<label for="TABLE_INDEX_forminput">'
                   4934:                            .&mt('Print Index')
                   4935:                            .'</label>'
                   4936:                         )
                   4937:             );
1.220     sakharuk 4938: 	    $paramHash = Apache::lonhelper::getParamHash();
1.627.2.38! raeburn  4939: 	    $paramHash->{'variable'} = 'TABLE_INDEX';
        !          4940: 	    $helper->declareVar('TABLE_INDEX');
1.220     sakharuk 4941: 	    $paramHash->{CHOICES} = [
1.223     bowersj2 4942: 				     ['No', 'no'],
                   4943: 				     ['Yes', 'yes'] ];
1.220     sakharuk 4944: 	    Apache::lonhelper::dropdown->new();
1.545     bisitz   4945:             &addMessage(&Apache::lonhtmlcommon::row_closure());
                   4946:             &addMessage(&Apache::lonhtmlcommon::row_title(
                   4947:                             '<label for="PRINT_DISCUSSIONS_forminput">'
                   4948:                            .&mt('Print Discussions')
                   4949:                            .'</label>'
                   4950:                         )
                   4951:             );
1.309     sakharuk 4952: 	    $paramHash = Apache::lonhelper::getParamHash();
1.627.2.38! raeburn  4953: 	    $paramHash->{'variable'} = 'PRINT_DISCUSSIONS';
        !          4954: 	    $helper->declareVar('PRINT_DISCUSSIONS');
1.309     sakharuk 4955: 	    $paramHash->{CHOICES} = [
                   4956: 				     ['No', 'no'],
                   4957: 				     ['Yes', 'yes'] ];
                   4958: 	    Apache::lonhelper::dropdown->new();
1.545     bisitz   4959:             &addMessage(&Apache::lonhtmlcommon::row_closure());
1.372     foxr     4960: 
1.511     foxr     4961: 	    # Prompt for printing annotations too.
                   4962: 		
1.545     bisitz   4963: 	    &addMessage(&Apache::lonhtmlcommon::row_title(
                   4964:                             '<label for="PRINT_ANNOTATIONS_forminput">'
                   4965:                            .&mt('Print Annotations')
                   4966:                            .'</label>'
                   4967:                         )
                   4968:             );
1.511     foxr     4969: 	    $paramHash = Apache::lonhelper::getParamHash();
                   4970: 	    $paramHash->{'variable'} = "PRINT_ANNOTATIONS";
                   4971: 	    $helper->declareVar("PRINT_ANNOTATIONS");
                   4972: 	    $paramHash->{CHOICES} = [
                   4973: 				     ['No', 'no'],
                   4974: 				     ['Yes', 'yes']];
                   4975: 	    Apache::lonhelper::dropdown->new();
1.545     bisitz   4976:             &addMessage(&Apache::lonhtmlcommon::row_closure());
1.511     foxr     4977: 
1.545     bisitz   4978:             &addMessage(&Apache::lonhtmlcommon::row_title(&mt('Foils')));
1.397     albertel 4979: 	    $paramHash = Apache::lonhelper::getParamHash();
                   4980: 	    $paramHash->{'multichoice'} = "true";
                   4981: 	    $paramHash->{'allowempty'}  = "true";
                   4982: 	    $paramHash->{'variable'}   = "showallfoils";
1.555     bisitz   4983: 	    $paramHash->{'CHOICES'} = [ [&mt('Show All Foils'), "1"] ];
1.397     albertel 4984: 	    Apache::lonhelper::choices->new();
1.545     bisitz   4985:             &addMessage(&Apache::lonhtmlcommon::row_closure(1));
1.220     sakharuk 4986: 	}
1.219     sakharuk 4987: 
1.627.2.38! raeburn  4988: 	if ($helper->{'VARS'}->{'construction'}) {
1.505     albertel 4989: 	    my $stylevalue='$Apache::lonnet::env{"construct.style"}';
1.497     www      4990:             my $randseedtext=&mt("Use random seed");
                   4991:             my $stylefiletext=&mt("Use style file");
1.506     albertel 4992:             my $selectfiletext=&mt("Select style file");
1.497     www      4993: 
1.544     bisitz   4994: 	    my $xmlfrag .= '<message>'
                   4995:             .&Apache::lonhtmlcommon::row_title('<label for="curseed_forminput">'
                   4996:                                               .$randseedtext
                   4997:                                               .'</label>'
                   4998:              )
                   4999:             .'</message>
                   5000:             <string variable="curseed" size="15" maxlength="15">
                   5001:                 <defaultvalue>
                   5002:                    return '.$helper->{VARS}->{'curseed'}.';
                   5003:                 </defaultvalue>'
                   5004:             .'</string>'
                   5005:             .'<message>'
                   5006:             .&Apache::lonhtmlcommon::row_closure()
                   5007:             .&Apache::lonhtmlcommon::row_title('<label for="style_file">'
                   5008:                                               .$stylefiletext
                   5009:                                               .'</label>'
                   5010:              )
                   5011:             .'</message>
1.504     albertel 5012:              <string variable="style_file" size="40">
1.544     bisitz   5013:                 <defaultvalue>
                   5014:                     return '.$stylevalue.';
                   5015:                 </defaultvalue>
                   5016:              </string><message>&nbsp;'
                   5017: .qq|<a href="javascript:openbrowser('helpform','style_file_forminput','sty')">|
                   5018: .$selectfiletext.'</a>'
                   5019:             .&Apache::lonhtmlcommon::row_closure()
1.555     bisitz   5020:             .&Apache::lonhtmlcommon::row_title(&mt('Show All Foils'))
1.544     bisitz   5021:             .'</message>
1.371     foxr     5022: 	     <choices allowempty="1" multichoice="true" variable="showallfoils">
1.544     bisitz   5023:                 <choice computer="1">&nbsp;</choice>
                   5024:              </choices>'
                   5025: 	    .'<message>'
                   5026:             .&Apache::lonhtmlcommon::row_closure()
                   5027:             .'</message>';
1.230     albertel 5028:             &Apache::lonxml::xmlparse($r, 'helper', $xmlfrag);
1.512     foxr     5029: 
                   5030: 
1.544     bisitz   5031:             &addMessage(&Apache::lonhtmlcommon::row_title(&mt('Problem Type')));
1.512     foxr     5032: 	    #
                   5033: 	    # Initial value from construction space:
                   5034: 	    #
                   5035: 	    if (!$helper->{VARS}->{'probstatus'} && $env{'form.problemtype'}) {
                   5036: 		$helper->{VARS}->{'probstatus'} = $env{'form.problemtype'};	# initial value
                   5037: 	    }
1.518     foxr     5038: 	    $xmlfrag = << "PROBTYPE";
                   5039: 		<dropdown variable="probstatus" multichoice="0" allowempty="0">
                   5040: 		   <defaultvalue>
                   5041: 		      return "$helper->{VARS}->{'probstatus'}";
                   5042:                    </defaultvalue>
                   5043: 		   <choice computer="problem">Homework Problem</choice>
                   5044: 		   <choice computer="exam">Exam Problem</choice>
                   5045: 		   <choice computer="survey">Survey question</choice>
1.572     raeburn  5046:                    ,choice computer="anonsurvey"Anonymous survey question</choice>
1.518     foxr     5047: 		</dropdown>
                   5048: PROBTYPE
                   5049:             &Apache::lonxml::xmlparse($r, 'helper', $xmlfrag);
1.544     bisitz   5050:             &addMessage(&Apache::lonhtmlcommon::row_closure(1));
1.512     foxr     5051: 
1.556     foxr     5052: 
                   5053: 
1.544     bisitz   5054:         }
1.223     bowersj2 5055:     }
1.264     sakharuk 5056: 
                   5057: 
                   5058: 
1.218     sakharuk 5059: 
1.223     bowersj2 5060:     if ($startedTable) {
1.544     bisitz   5061:         &addMessage(&Apache::lonhtmlcommon::end_pick_box());
1.215     sakharuk 5062:     }
1.161     bowersj2 5063: 
1.131     bowersj2 5064:     Apache::lonprintout::page_format_state->new("FORMAT");
                   5065: 
1.144     bowersj2 5066:     # Generate the PAGESIZE state which will offer the user the margin
                   5067:     # choices if they select one column
                   5068:     Apache::lonhelper::state->new("PAGESIZE", "Set Margins");
                   5069:     Apache::lonprintout::page_size_state->new('pagesize', 'FORMAT', 'FINAL');
                   5070: 
                   5071: 
1.131     bowersj2 5072:     $helper->process();
                   5073: 
1.416     foxr     5074: 
1.131     bowersj2 5075:     # MANUAL BAILOUT CONDITION:
                   5076:     # If we're in the "final" state, bailout and return to handler
                   5077:     if ($helper->{STATE} eq 'FINAL') {
                   5078:         return $helper;
1.627.2.38! raeburn  5079:     }
1.130     sakharuk 5080: 
1.584     raeburn  5081:     my $footer;
1.395     www      5082:     if ($helper->{STATE} eq 'START') {
1.627.2.38! raeburn  5083:         my $prtspool=$r->dir_config('lonPrtDir');
1.584     raeburn  5084: 	$footer = &recently_generated($prtspool);
1.395     www      5085:     }
1.584     raeburn  5086:     $r->print($helper->display($footer));
1.333     albertel 5087:     &Apache::lonhelper::unregisterHelperTags();
1.115     bowersj2 5088: 
                   5089:     return OK;
                   5090: }
                   5091: 
1.1       www      5092: 
                   5093: 1;
1.119     bowersj2 5094: 
                   5095: package Apache::lonprintout::page_format_state;
                   5096: 
                   5097: =pod
                   5098: 
1.131     bowersj2 5099: =head1 Helper element: page_format_state
                   5100: 
                   5101: See lonhelper.pm documentation for discussion of the helper framework.
1.119     bowersj2 5102: 
1.627.2.38! raeburn  5103: Apache::lonprintout::page_format_state is an element that gives the
        !          5104: user an opportunity to select the page layout they wish to print
        !          5105: with: Number of columns, portrait/landscape, and paper size. If you
        !          5106: want to change the paper size choices, change the @paperSize array
1.131     bowersj2 5107: contents in this package.
1.119     bowersj2 5108: 
1.131     bowersj2 5109: page_format_state is always directly invoked in lonprintout.pm, so there
                   5110: is no tag interface. You actually pass parameters to the constructor.
1.119     bowersj2 5111: 
                   5112: =over 4
                   5113: 
1.131     bowersj2 5114: =item * B<new>(varName): varName is where the print information will be stored in the format FIXME.
1.119     bowersj2 5115: 
                   5116: =back
                   5117: 
                   5118: =cut
                   5119: 
1.131     bowersj2 5120: use Apache::lonhelper;
1.119     bowersj2 5121: 
                   5122: no strict;
1.131     bowersj2 5123: @ISA = ("Apache::lonhelper::element");
1.119     bowersj2 5124: use strict;
1.266     sakharuk 5125: use Apache::lonlocal;
1.373     albertel 5126: use Apache::lonnet;
1.119     bowersj2 5127: 
                   5128: my $maxColumns = 2;
1.376     albertel 5129: # it'd be nice if these all worked
1.627.2.38! raeburn  5130: #my @paperSize = ("letter [8 1/2x11 in]", "legal [8 1/2x14 in]",
1.376     albertel 5131: #                 "tabloid (ledger) [11x17 in]", "executive [7 1/2x10 in]",
1.627.2.38! raeburn  5132: #                 "a2 [420x594 mm]", "a3 [297x420 mm]", "a4 [210x297 mm]",
1.376     albertel 5133: #                 "a5 [148x210 mm]", "a6 [105x148 mm]" );
1.627.2.38! raeburn  5134: my @paperSize = ("letter [8 1/2x11 in]", "legal [8 1/2x14 in]",
1.376     albertel 5135: 		 "a4 [210x297 mm]");
1.119     bowersj2 5136: 
                   5137: # Tentative format: Orientation (L = Landscape, P = portrait) | Colnum |
                   5138: #                   Paper type
                   5139: 
1.627.2.38! raeburn  5140: sub new {
1.131     bowersj2 5141:     my $self = Apache::lonhelper::element->new();
1.119     bowersj2 5142: 
1.135     bowersj2 5143:     shift;
                   5144: 
1.131     bowersj2 5145:     $self->{'variable'} = shift;
1.134     bowersj2 5146:     my $helper = Apache::lonhelper::getHelper();
1.135     bowersj2 5147:     $helper->declareVar($self->{'variable'});
1.131     bowersj2 5148:     bless($self);
1.119     bowersj2 5149:     return $self;
                   5150: }
                   5151: 
                   5152: sub render {
                   5153:     my $self = shift;
1.131     bowersj2 5154:     my $helper = Apache::lonhelper::getHelper();
1.119     bowersj2 5155:     my $result = '';
1.131     bowersj2 5156:     my $var = $self->{'variable'};
1.266     sakharuk 5157:     my $PageLayout=&mt('Page layout');
                   5158:     my $NumberOfColumns=&mt('Number of columns');
                   5159:     my $PaperType=&mt('Paper type');
1.506     albertel 5160:     my $landscape=&mt('Landscape');
                   5161:     my $portrait=&mt('Portrait');
1.627.2.38! raeburn  5162: 
1.556     foxr     5163: 
1.544     bisitz   5164:     $result.='<h3>'.&mt('Layout Options').'</h3>'
                   5165:             .&Apache::loncommon::start_data_table()
                   5166:             .&Apache::loncommon::start_data_table_header_row()
                   5167:             .'<th>'.$PageLayout.'</th>'
                   5168:             .'<th>'.$NumberOfColumns.'</th>'
                   5169:             .'<th>'.$PaperType.'</th>'
                   5170:             .&Apache::loncommon::end_data_table_header_row()
                   5171:             .&Apache::loncommon::start_data_table_row()
                   5172:     .'<td>'
                   5173:     .'<label><input type="radio" name="'.${var}.'.layout" value="L" />'.$landscape.'</label><br />'
                   5174:     .'<label><input type="radio" name="'.${var}.'.layout" value="P" checked="checked" />'.$portrait.'</label>'
                   5175:     .'</td>';
1.119     bowersj2 5176: 
1.544     bisitz   5177:     $result.='<td align="center">'
                   5178:             .'<select name="'.${var}.'.cols">';
1.119     bowersj2 5179: 
                   5180:     my $i;
                   5181:     for ($i = 1; $i <= $maxColumns; $i++) {
1.144     bowersj2 5182:         if ($i == 2) {
1.553     bisitz   5183:             $result .= '<option value="'.$i.'" selected="selected">'.$i.'</option>'."\n";
1.119     bowersj2 5184:         } else {
1.553     bisitz   5185:             $result .= '<option value="'.$i.'">'.$i.'</option>'."\n";
1.119     bowersj2 5186:         }
                   5187:     }
                   5188: 
                   5189:     $result .= "</select></td><td>\n";
                   5190:     $result .= "<select name='${var}.paper'>\n";
                   5191: 
1.373     albertel 5192:     my %parmhash=&Apache::lonnet::coursedescription($env{'request.course.id'});
1.398     albertel 5193:     my $DefaultPaperSize=lc($parmhash{'default_paper_size'});
                   5194:     $DefaultPaperSize=~s/\s//g;
1.304     sakharuk 5195:     if ($DefaultPaperSize eq '') {$DefaultPaperSize='letter';}
1.119     bowersj2 5196:     $i = 0;
                   5197:     foreach (@paperSize) {
1.326     sakharuk 5198: 	$_=~/(\w+)/;
                   5199: 	my $papersize=$1;
1.304     sakharuk 5200:         if ($paperSize[$i]=~/$DefaultPaperSize/) {
1.553     bisitz   5201:             $result .= '<option selected="selected" value="'.$papersize.'">'.$paperSize[$i].'</option>'."\n";
1.119     bowersj2 5202:         } else {
1.553     bisitz   5203:             $result .= '<option value="'.$papersize.'">'.$paperSize[$i].'</option>'."\n";
1.119     bowersj2 5204:         }
                   5205:         $i++;
                   5206:     }
1.539     onken    5207:     $result .= <<HTML;
                   5208:         </select>
                   5209:     </td>
                   5210: HTML
1.544     bisitz   5211:     $result.=&Apache::loncommon::end_data_table_row()
                   5212:             .&Apache::loncommon::end_data_table();
1.539     onken    5213: 
1.119     bowersj2 5214:     return $result;
1.135     bowersj2 5215: }
                   5216: 
                   5217: sub postprocess {
                   5218:     my $self = shift;
                   5219: 
                   5220:     my $var = $self->{'variable'};
1.136     bowersj2 5221:     my $helper = Apache::lonhelper->getHelper();
1.627.2.38! raeburn  5222:     $helper->{VARS}->{$var} =
1.373     albertel 5223:         $env{"form.$var.layout"} . '|' . $env{"form.$var.cols"} . '|' .
1.539     onken    5224:         $env{"form.$var.paper"} . '|' . $env{"form.$var.pdfFormFields"};
1.135     bowersj2 5225:     return 1;
1.119     bowersj2 5226: }
                   5227: 
                   5228: 1;
1.144     bowersj2 5229: 
                   5230: package Apache::lonprintout::page_size_state;
                   5231: 
                   5232: =pod
                   5233: 
                   5234: =head1 Helper element: page_size_state
                   5235: 
                   5236: See lonhelper.pm documentation for discussion of the helper framework.
                   5237: 
1.627.2.38! raeburn  5238: Apache::lonprintout::page_size_state is an element that gives the
1.144     bowersj2 5239: user the opportunity to further refine the page settings if they
                   5240: select a single-column page.
                   5241: 
                   5242: page_size_state is always directly invoked in lonprintout.pm, so there
                   5243: is no tag interface. You actually pass parameters to the constructor.
                   5244: 
                   5245: =over 4
                   5246: 
                   5247: =item * B<new>(varName): varName is where the print information will be stored in the format FIXME.
                   5248: 
                   5249: =back
                   5250: 
                   5251: =cut
                   5252: 
                   5253: use Apache::lonhelper;
1.373     albertel 5254: use Apache::lonnet;
1.144     bowersj2 5255: no strict;
                   5256: @ISA = ("Apache::lonhelper::element");
                   5257: use strict;
                   5258: 
                   5259: 
                   5260: 
1.627.2.38! raeburn  5261: sub new {
1.144     bowersj2 5262:     my $self = Apache::lonhelper::element->new();
                   5263: 
                   5264:     shift; # disturbs me (probably prevents subclassing) but works (drops
                   5265:            # package descriptor)... - Jeremy
                   5266: 
                   5267:     $self->{'variable'} = shift;
                   5268:     my $helper = Apache::lonhelper::getHelper();
                   5269:     $helper->declareVar($self->{'variable'});
                   5270: 
1.627.2.38! raeburn  5271:     # The variable name of the format element, so we can look into
1.144     bowersj2 5272:     # $helper->{VARS} to figure out whether the columns are one or two
                   5273:     $self->{'formatvar'} = shift;
                   5274: 
1.463     foxr     5275: 
1.144     bowersj2 5276:     $self->{NEXTSTATE} = shift;
                   5277:     bless($self);
1.467     foxr     5278: 
1.144     bowersj2 5279:     return $self;
                   5280: }
                   5281: 
                   5282: sub render {
                   5283:     my $self = shift;
                   5284:     my $helper = Apache::lonhelper::getHelper();
                   5285:     my $result = '';
                   5286:     my $var = $self->{'variable'};
                   5287: 
1.467     foxr     5288: 
                   5289: 
1.144     bowersj2 5290:     if (defined $self->{ERROR_MSG}) {
1.464     albertel 5291:         $result .= '<br /><span class="LC_error">' . $self->{ERROR_MSG} . '</span><br />';
1.144     bowersj2 5292:     }
                   5293: 
1.438     foxr     5294:     my $format = $helper->{VARS}->{$self->{'formatvar'}};
1.463     foxr     5295: 
                   5296:     # Use format to get sensible defaults for the margins:
                   5297: 
                   5298: 
                   5299:     my ($laystyle, $cols, $papersize) = split(/\|/, $format);
                   5300:     ($papersize)                      = split(/ /, $papersize);
                   5301: 
1.559     foxr     5302:     $laystyle = &Apache::lonprintout::map_laystyle($laystyle);
1.463     foxr     5303: 
                   5304: 
                   5305: 
1.464     albertel 5306:     my %size;
                   5307:     ($size{'width_and_units'},
                   5308:      $size{'height_and_units'},
                   5309:      $size{'margin_and_units'})=
                   5310: 	 &Apache::lonprintout::page_format($papersize, $laystyle, $cols);
1.627.2.38! raeburn  5311: 
1.464     albertel 5312:     foreach my $dimension ('width','height','margin') {
                   5313: 	($size{$dimension},$size{$dimension.'_unit'}) =
                   5314: 	    split(/ +/, $size{$dimension.'_and_units'},2);
                   5315:        	
                   5316: 	foreach my $unit ('cm','in') {
                   5317: 	    $size{$dimension.'_options'} .= '<option ';
                   5318: 	    if ($size{$dimension.'_unit'} eq $unit) {
                   5319: 		$size{$dimension.'_options'} .= 'selected="selected" ';
                   5320: 	    }
                   5321: 	    $size{$dimension.'_options'} .= '>'.$unit.'</option>';
                   5322: 	}
1.438     foxr     5323:     }
                   5324: 
1.470     foxr     5325:     # Adjust margin for LaTeX margin: .. requires units == cm or in.
                   5326: 
                   5327:     if ($size{'margin_unit'} eq 'in') {
                   5328: 	$size{'margin'} += 1;
                   5329:     }  else {
                   5330: 	$size{'margin'} += 2.54;
                   5331:     }
1.548     bisitz   5332:     my %lt = &Apache::lonlocal::texthash(
                   5333:         'format' => 'How should each column be formatted?',
                   5334:         'width'  => 'Width',
                   5335:         'height' => 'Height',
                   5336:         'margin' => 'Left Margin'
                   5337:     );
                   5338: 
                   5339:     $result .= '<p>'.$lt{'format'}.'</p>'
                   5340:               .&Apache::lonhtmlcommon::start_pick_box()
                   5341:               .&Apache::lonhtmlcommon::row_title($lt{'width'})
                   5342:               .'<input type="text" name="'.$var.'.width" value="'.$size{'width'}.'" size="4" />'
                   5343:               .'<select name="'.$var.'.widthunit">'
                   5344:               .$size{'width_options'}
                   5345:               .'</select>'
                   5346:               .&Apache::lonhtmlcommon::row_closure()
                   5347:               .&Apache::lonhtmlcommon::row_title($lt{'height'})
                   5348:               .'<input type="text" name="'.$var.'.height" value="'.$size{'height'}.'" size="4" />'
                   5349:               .'<select name="'.$var.'.heightunit">'
                   5350:               .$size{'height_options'}
                   5351:               .'</select>'
                   5352:               .&Apache::lonhtmlcommon::row_closure()
                   5353:               .&Apache::lonhtmlcommon::row_title($lt{'margin'})
                   5354:               .'<input type="text" name="'.$var.'.lmargin" value="'.$size{'margin'}.'" size="4" />'
                   5355:               .'<select name="'.$var.'.lmarginunit">'
                   5356:               .$size{'margin_options'}
                   5357:               .'</select>'
                   5358:               .&Apache::lonhtmlcommon::row_closure(1)
                   5359:               .&Apache::lonhtmlcommon::end_pick_box();
                   5360:     # <p>Hint: Some instructors like to leave scratch space for the student by
                   5361:     # making the width much smaller than the width of the page.</p>
1.144     bowersj2 5362: 
                   5363:     return $result;
                   5364: }
                   5365: 
1.470     foxr     5366: 
1.144     bowersj2 5367: sub preprocess {
                   5368:     my $self = shift;
                   5369:     my $helper = Apache::lonhelper::getHelper();
                   5370: 
                   5371:     my $format = $helper->{VARS}->{$self->{'formatvar'}};
1.467     foxr     5372: 
                   5373:     #  If the user does not have 'pav' privilege, set default widths and
                   5374:     #  on to the next state right away.
                   5375:     #
                   5376:     if (!$perm{'pav'}) {
                   5377: 	my $var = $self->{'variable'};
                   5378: 	my $format = $helper->{VARS}->{$self->{'formatvar'}};
                   5379: 	
                   5380: 	my ($laystyle, $cols, $papersize) = split(/\|/, $format);
                   5381: 	($papersize)                      = split(/ /, $papersize);
                   5382: 	
                   5383: 	
1.560     foxr     5384: 	$laystyle = &Apache::lonprintout::map_laystyle($laystyle);
1.559     foxr     5385: 
1.467     foxr     5386: 	#  Figure out some good defaults for the print out and set them:
                   5387: 	
                   5388: 	my %size;
                   5389: 	($size{'width'},
                   5390: 	 $size{'height'},
                   5391: 	 $size{'lmargin'})=
                   5392: 	     &Apache::lonprintout::page_format($papersize, $laystyle, $cols);
                   5393: 	
                   5394: 	foreach my $dim ('width', 'height', 'lmargin') {
                   5395: 	    my ($value, $units) = split(/ /, $size{$dim});
1.627.2.38! raeburn  5396: 
1.467     foxr     5397: 	    $helper->{VARS}->{"$var.".$dim}      = $value;
                   5398: 	    $helper->{VARS}->{"$var.".$dim.'unit'} = $units;
1.627.2.38! raeburn  5399: 
1.467     foxr     5400: 	}
                   5401: 	
                   5402: 
                   5403: 	# Transition to the next state
                   5404: 
                   5405: 	$helper->changeState($self->{NEXTSTATE});
                   5406:     }
1.627.2.38! raeburn  5407: 
1.144     bowersj2 5408:     return 1;
                   5409: }
                   5410: 
                   5411: sub postprocess {
                   5412:     my $self = shift;
                   5413: 
                   5414:     my $var = $self->{'variable'};
                   5415:     my $helper = Apache::lonhelper->getHelper();
1.627.2.38! raeburn  5416:     my $width = $helper->{VARS}->{$var .'.width'} = $env{"form.${var}.width"};
        !          5417:     my $height = $helper->{VARS}->{$var .'.height'} = $env{"form.${var}.height"};
        !          5418:     my $lmargin = $helper->{VARS}->{$var .'.lmargin'} = $env{"form.${var}.lmargin"};
        !          5419:     $helper->{VARS}->{$var .'.widthunit'} = $env{"form.${var}.widthunit"};
        !          5420:     $helper->{VARS}->{$var .'.heightunit'} = $env{"form.${var}.heightunit"};
        !          5421:     $helper->{VARS}->{$var .'.lmarginunit'} = $env{"form.${var}.lmarginunit"};
1.144     bowersj2 5422: 
                   5423:     my $error = '';
                   5424: 
1.627.2.38! raeburn  5425:     # /^-?[0-9]+(\.[0-9]*)?$/ -> optional minus, at least on digit, followed
1.144     bowersj2 5426:     # by an optional period, followed by digits, ending the string
                   5427: 
1.464     albertel 5428:     if ($width !~  /^-?[0-9]*(\.[0-9]*)?$/) {
1.144     bowersj2 5429:         $error .= "Invalid width; please type only a number.<br />\n";
                   5430:     }
1.464     albertel 5431:     if ($height !~  /^-?[0-9]*(\.[0-9]*)?$/) {
1.144     bowersj2 5432:         $error .= "Invalid height; please type only a number.<br />\n";
                   5433:     }
1.464     albertel 5434:     if ($lmargin !~  /^-?[0-9]*(\.[0-9]*)?$/) {
1.144     bowersj2 5435:         $error .= "Invalid left margin; please type only a number.<br />\n";
1.470     foxr     5436:     } else {
                   5437: 	# Adjust for LaTeX 1.0 inch margin:
                   5438: 
                   5439: 	if ($env{"form.${var}.lmarginunit"} eq "in") {
                   5440: 	    $helper->{VARS}->{$var.'.lmargin'} = $lmargin - 1;
                   5441: 	} else {
                   5442: 	    $helper->{VARS}->{$var.'.lmargin'} = $lmargin - 2.54;
                   5443: 	}
1.144     bowersj2 5444:     }
                   5445: 
                   5446:     if (!$error) {
                   5447:         Apache::lonhelper::getHelper()->changeState($self->{NEXTSTATE});
                   5448:         return 1;
                   5449:     } else {
                   5450:         $self->{ERROR_MSG} = $error;
                   5451:         return 0;
                   5452:     }
                   5453: }
                   5454: 
1.1       www      5455: __END__
1.6       sakharuk 5456: 

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