Annotation of loncom/interface/londropadd.pm, revision 1.141

1.1       www         1: # The LearningOnline Network with CAPA
                      2: # Handler to drop and add students in courses 
                      3: #
1.141   ! albertel    4: # $Id: londropadd.pm,v 1.140 2006/04/26 14:52:22 albertel Exp $
1.17      albertel    5: #
                      6: # Copyright Michigan State University Board of Trustees
                      7: #
                      8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
                      9: #
                     10: # LON-CAPA is free software; you can redistribute it and/or modify
                     11: # it under the terms of the GNU General Public License as published by
                     12: # the Free Software Foundation; either version 2 of the License, or
                     13: # (at your option) any later version.
                     14: #
                     15: # LON-CAPA is distributed in the hope that it will be useful,
                     16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
                     17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     18: # GNU General Public License for more details.
                     19: #
                     20: # You should have received a copy of the GNU General Public License
                     21: # along with LON-CAPA; if not, write to the Free Software
                     22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
                     23: #
                     24: # /home/httpd/html/adm/gpl.txt
                     25: #
                     26: # http://www.lon-capa.org/
                     27: #
1.1       www        28: #
1.50      matthew    29: ###############################################################
1.82      www        30: ##############################################################
1.1       www        31: 
                     32: package Apache::londropadd;
                     33: 
                     34: use strict;
1.127     albertel   35: use Apache::lonnet;
1.24      albertel   36: use Apache::loncommon();
1.50      matthew    37: use Apache::lonhtmlcommon();
1.1       www        38: use Apache::Constants qw(:common :http REDIRECT);
1.60      matthew    39: use Spreadsheet::WriteExcel;
1.110     matthew    40: use Apache::lonstathelpers();
1.86      www        41: use Apache::lonlocal;
1.1       www        42: 
1.50      matthew    43: ###############################################################
                     44: ###############################################################
1.10      www        45: sub header {
1.138     albertel   46:     my $start_page=&Apache::loncommon::start_page('Enrollment Manager');
1.27      matthew    47:     return(<<ENDHEAD);
1.138     albertel   48: $start_page
1.40      matthew    49: <form method="post" enctype="multipart/form-data"  
                     50:       action="/adm/dropadd" name="studentform">
1.1       www        51: ENDHEAD
1.10      www        52: }
                     53: 
1.50      matthew    54: ###############################################################
                     55: ###############################################################
                     56: # Drop student from all sections of a course, except optional $csec
1.26      matthew    57: sub modifystudent {
1.33      matthew    58:     my ($udom,$unam,$courseid,$csec,$desiredhost)=@_;
1.26      matthew    59:     # if $csec is undefined, drop the student from all the courses matching
                     60:     # this one.  If $csec is defined, drop them from all other sections of 
                     61:     # this course and add them to section $csec
1.25      matthew    62:     $courseid=~s/\_/\//g;
                     63:     $courseid=~s/^(\w)/\/$1/;
1.26      matthew    64:     my %roles = &Apache::lonnet::dump('roles',$udom,$unam);
                     65:     my ($tmp) = keys(%roles);
                     66:     # Bail out if we were unable to get the students roles
1.35      matthew    67:     return "$1" if ($tmp =~ /^(con_lost|error|no_such_host)/i);
1.26      matthew    68:     # Go through the roles looking for enrollment in this course
1.35      matthew    69:     my $result = '';
1.26      matthew    70:     foreach my $course (keys(%roles)) {
1.35      matthew    71:         if ($course=~/^$courseid(?:\/)*(?:\s+)*(\w+)*\_st$/) {
1.26      matthew    72:             # We are in this course
1.25      matthew    73:             my $section=$1;
1.26      matthew    74:             $section='' if ($course eq $courseid.'_st');
1.87      matthew    75:             if (defined($csec) && $section eq $csec) {
1.71      matthew    76:                 $result .= 'ok:';
                     77:             } elsif ( ((!$section) && (!$csec)) || ($section ne $csec) ) {
1.27      matthew    78:                 my (undef,$end,$start)=split(/\_/,$roles{$course});
1.25      matthew    79:                 my $now=time;
1.50      matthew    80:                 # if this is an active role 
1.27      matthew    81:                 if (!($start && ($now<$start)) || !($end && ($now>$end))) {
1.25      matthew    82:                     my $reply=&Apache::lonnet::modifystudent
1.70      matthew    83:                         # dom  name  id mode pass     f     m     l     g
                     84:                         ($udom,$unam,'',  '',  '',undef,undef,undef,undef,
1.33      matthew    85:                          $section,time,undef,undef,$desiredhost);
1.35      matthew    86:                     $result .= $reply.':';
1.25      matthew    87:                 }
1.10      www        88:             }
                     89:         }
1.20      harris41   90:     }
1.35      matthew    91:     if ($result eq '') {
1.62      matthew    92:         $result = 'Unable to find section for this student';
1.37      matthew    93:     } else {
                     94:         $result =~ s/(ok:)+/ok/g;
1.35      matthew    95:     }
                     96:     return $result;
1.10      www        97: }
                     98: 
1.50      matthew    99: ###############################################################
                    100: ###############################################################
                    101: # build a domain and server selection form
1.31      matthew   102: sub domain_form {
                    103:     my ($defdom) = @_;
                    104:     # Set up domain and server selection forms
                    105:     #
                    106:     # Get the domains
                    107:     my @domains = &Apache::loncommon::get_domains();
                    108:     # build up the menu information to be passed to 
                    109:     # &Apache::loncommon::linked_select_forms
                    110:     my %select_menus;
                    111:     foreach my $dom (@domains) {
                    112:         # set up the text for this domain
                    113:         $select_menus{$dom}->{'text'}= $dom;
                    114:         # we want a choice of 'default' as the default in the second menu
                    115:         $select_menus{$dom}->{'default'}= 'default';
                    116:         $select_menus{$dom}->{'select2'}->{'default'} = 'default';
                    117:         # Now build up the other items in the second menu
1.45      matthew   118:         my %servers = &Apache::loncommon::get_library_servers($dom);
1.31      matthew   119:         foreach my $server (keys(%servers)) {
                    120:             $select_menus{$dom}->{'select2'}->{$server} 
                    121:                                             = "$server $servers{$server}";
                    122:         }
                    123:     }
                    124:     my $result  = &Apache::loncommon::linked_select_forms
                    125:         ('studentform',' with home server ',$defdom,
                    126:          'lcdomain','lcserver',\%select_menus);
                    127:     return $result;
                    128: }
                    129: 
1.50      matthew   130: ###############################################################
                    131: ###############################################################
                    132: #  Menu Phase One
                    133: sub print_main_menu {
1.132     raeburn   134:     my ($r,$enrl_permission,$view_permission,$grp_manage_permission,
                    135:         $grp_view_permission)=@_;
1.121     matthew   136:     #
1.127     albertel  137:     my ($cdom,$cnum) = split/_/,$env{'request.course.id'};
1.121     matthew   138:     my @menu = 
                    139:         ( 
1.122     matthew   140:           { text => 'Upload a class list', 
1.121     matthew   141:             help => 'Course_Create_Class_List',
                    142:             action => 'upload',
                    143:             permission => $enrl_permission,
                    144:             },
                    145:           { text => 'Enroll a single student', 
                    146:             help => 'Course_Add_Student',
                    147:             action => 'enrollstudent',
                    148:             permission => $enrl_permission,
                    149:             },
                    150:           { text => 'Modify student data', 
                    151:             help => 'Course_Modify_Student_Data',
                    152:             action => 'modifystudent',
                    153:             permission => $enrl_permission,
                    154:             },
                    155:           { text => 'View Class List', 
                    156:             help => 'Course_View_Class_List',
                    157:             action => 'classlist',
                    158:             permission => $view_permission,
                    159:             },
                    160:           { text => 'Drop Students', 
                    161:             help => 'Course_Drop_Student',
                    162:             action => 'drop',
                    163:             permission => $enrl_permission,
                    164:             },
                    165:           { text => 'Automated Enrollment Manager', 
                    166:             permission => &Apache::lonnet::auto_run($cnum,$cdom),
                    167:             url  => '/adm/populate',
                    168:             },
1.137     raeburn   169:           { text => 'Create a new group',
                    170:             help => 'Course_Create_Group',
                    171:             permission => $grp_manage_permission,
                    172:             url => '/adm/coursegroups?refpage=enrl&action=create',
                    173:             },
                    174:           { text => 'Modify an existing group',
                    175:             help => 'Course_Modify_Group',
                    176:             permission => $grp_manage_permission,
                    177:             url => '/adm/coursegroups?refpage=enrl&action=modify',
                    178:             },
                    179:           { text => 'Delete an existing group',
                    180:             help => 'Course_Delete_Group',
                    181:             permission => $grp_manage_permission,
                    182:             url => '/adm/coursegroups?refpage=enrl&action=delete',
                    183:             },
                    184:           { text => 'Enter an existing group',
                    185:             help => 'Course_Display_Group',
                    186:             permission => $grp_view_permission,
                    187:             url => '/adm/coursegroups?refpage=enrl&action=view',
                    188:             },
1.121     matthew   189:           );
                    190:     my $menu_html = '';
                    191:     foreach my $menu_item (@menu) {
                    192:         next if (! $menu_item->{'permission'});
                    193:         $menu_html.='<p>';
                    194:         $menu_html.='<font size="+1">';
                    195:         if (exists($menu_item->{'url'})) {
                    196:             $menu_html.=qq{<a href="$menu_item->{'url'}">};
                    197:         } else {
                    198:             $menu_html.=
                    199:                 qq{<a href="/adm/dropadd?action=$menu_item->{'action'}">};
                    200:         }
                    201:         $menu_html.= &mt($menu_item->{'text'}).'</a></font>';
                    202:         if (exists($menu_item->{'help'})) {
                    203:             $menu_html.=
                    204:                 &Apache::loncommon::help_open_topic($menu_item->{'help'});
                    205:         }
                    206:         $menu_html.='</p>'.$/;
1.113     raeburn   207:     }
1.121     matthew   208:     $r->print($menu_html);
                    209:     return;
1.10      www       210: }
                    211: 
1.50      matthew   212: ###############################################################
                    213: ###############################################################
1.89      matthew   214: sub hidden_input {
                    215:     my ($name,$value) = @_;
                    216:     return '<input type="hidden" name="'.$name.'" value="'.$value.'" />'."\n";
                    217: }
                    218: 
1.50      matthew   219: sub print_upload_manager_header {
1.23      albertel  220:     my ($r,$datatoken,$distotal,$krbdefdom)=@_;
1.24      albertel  221:     my $javascript;
1.99      matthew   222:     #
1.127     albertel  223:     if (! exists($env{'form.upfile_associate'})) {
                    224:         $env{'form.upfile_associate'} = 'forward';
1.50      matthew   225:     }
1.127     albertel  226:     if ($env{'form.associate'} eq 'Reverse Association') {
                    227:         if ( $env{'form.upfile_associate'} ne 'reverse' ) {
                    228:             $env{'form.upfile_associate'} = 'reverse';
1.50      matthew   229:         } else {
1.127     albertel  230:             $env{'form.upfile_associate'} = 'forward';
1.50      matthew   231:         }
                    232:     }
1.127     albertel  233:     if ($env{'form.upfile_associate'} eq 'reverse') {
1.50      matthew   234: 	$javascript=&upload_manager_javascript_reverse_associate();
1.24      albertel  235:     } else {
1.50      matthew   236: 	$javascript=&upload_manager_javascript_forward_associate();
1.24      albertel  237:     }
1.99      matthew   238:     #
                    239:     # Deal with restored settings
                    240:     my $password_choice = '';
1.127     albertel  241:     if (exists($env{'form.ipwd_choice'}) &&
                    242:         $env{'form.ipwd_choice'} ne '') {
1.99      matthew   243:         # If a column was specified for password, assume it is for an
                    244:         # internal password.  This is a bug waiting to be filed (could be
                    245:         # local or krb auth instead of internal) but I do not have the 
                    246:         # time to mess around with this now.
                    247:         $password_choice = 'int';        
                    248:     }
                    249:     #
                    250:     my $javascript_validations=&javascript_validations('auth',$krbdefdom,
                    251:                                     $password_choice);
1.127     albertel  252:     my $checked=(($env{'form.noFirstLine'})?' checked="1"':'');
1.88      matthew   253:     $r->print('<h3>'.&mt('Uploading Class List')."</h3>\n".
                    254:               "<hr>\n".
                    255:               '<h3>'.&mt('Identify fields')."</h3>\n");
                    256:     $r->print("<p>\n".
                    257:               &mt('Total number of records found in file: [_1].',$distotal).
                    258:               "\n".
                    259:               "</p><hr>\n");
1.94      sakharuk  260:     $r->print(&mt('Enter as many fields as you can. The system will inform you and bring you back to this page if the data selected is insufficient to enroll students in your class.')."<hr>\n");
1.89      matthew   261:     $r->print(&hidden_input('action','upload').
                    262:               &hidden_input('state','got_file').
                    263:               &hidden_input('associate','').
                    264:               &hidden_input('datatoken',$datatoken).
1.127     albertel  265:               &hidden_input('fileupload',$env{'form.fileupload'}).
                    266:               &hidden_input('upfiletype',$env{'form.upfiletype'}).
                    267:               &hidden_input('upfile_associate',$env{'form.upfile_associate'}));
1.89      matthew   268:     $r->print('<input type="button" value="Reverse Association" '.
                    269:               'name="'.&mt('Reverse Association').'" '.
                    270:               'onClick="javascript:this.form.associate.value=\'Reverse Association\';submit(this.form);" />');
1.131     albertel  271:     $r->print('<label><input type="checkbox" name="noFirstLine" $checked />'.
                    272:               &mt('Ignore First Line').'</label>');
1.89      matthew   273:     $r->print("<hr />\n".
                    274:               '<script type="text/javascript" language="Javascript">'."\n".
                    275:               $javascript."\n".$javascript_validations.'</script>');
1.24      albertel  276: }
                    277: 
1.50      matthew   278: ###############################################################
                    279: ###############################################################
1.24      albertel  280: sub javascript_validations {
1.96      raeburn   281:     my ($mode,$krbdefdom,$curr_authtype,$curr_authfield)=@_;
1.89      matthew   282:     my $authheader;
                    283:     if ($mode eq 'auth') {
                    284:         my %param = ( formname => 'studentform',
1.99      matthew   285:                       kerb_def_dom => $krbdefdom,
                    286:                       curr_authtype => $curr_authtype);
1.89      matthew   287:         $authheader = &Apache::loncommon::authform_header(%param);
1.91      raeburn   288:     } elsif ($mode eq 'createcourse') {
                    289:         my %param = ( formname => 'ccrs',
1.99      matthew   290:                   kerb_def_dom => $krbdefdom,
                    291:                       curr_authtype => $curr_authtype );
1.91      raeburn   292:         $authheader = &Apache::loncommon::authform_header(%param);
1.96      raeburn   293:     } elsif ($mode eq 'modifycourse') {
                    294:         my %param = ( formname => 'cmod',
                    295:                   kerb_def_dom => $krbdefdom,
                    296:                   mode => 'modifycourse',
                    297:                   curr_authtype => $curr_authtype,
                    298:                   curr_autharg => $curr_authfield );
                    299:         $authheader = &Apache::loncommon::authform_header(%param);
1.89      matthew   300:     }
1.96      raeburn   301: 
1.91      raeburn   302:     
1.89      matthew   303:     my %alert = &Apache::lonlocal::texthash
                    304:         (username => 'You need to specify the username field.',
                    305:          authen   => 'You must choose an authentication type.',
                    306:          krb      => 'You need to specify the Kerberos domain.',
                    307:          ipass    => 'You need to specify the initial password.',
                    308:          name     => 'The optional name field was not specified.',
                    309:          snum     => 'The optional student number field was not specified.',
                    310:          section  => 'The optional section or group field was not specified.', 
                    311:          email    => 'The optional email address field was not specified.',
                    312:          continue => 'Continue enrollment?',
                    313:          );
                    314:     
                    315: #    my $pjump_def = &Apache::lonhtmlcommon::pjump_javascript_definition();
                    316:     my $function_name =(<<END);
1.73      www       317: function verify_message (vf,founduname,foundpwd,foundname,foundid,foundsec,foundemail) {
1.89      matthew   318: END
1.97      raeburn   319:     my $auth_checks;
1.96      raeburn   320:     if ($mode eq 'createcourse') {
                    321:         $auth_checks .= (<<END);
1.97      raeburn   322:     if (vf.autoadds[0].checked == true) {
1.96      raeburn   323:         if (current.radiovalue == null || current.radiovalue == 'nochange') {
                    324:             alert('$alert{'authen'}');
                    325:             return;
                    326:         }
                    327:     }
                    328: END
                    329:     } else {
1.91      raeburn   330:         $auth_checks .= (<<END);
1.97      raeburn   331:     var foundatype=0;
1.3       www       332:     if (founduname==0) {
1.89      matthew   333: 	alert('$alert{'username'}');
1.3       www       334:         return;
                    335:     }
1.61      matthew   336:     // alert('current.radiovalue = '+current.radiovalue);
1.119     albertel  337:     if (current.radiovalue == null || current.radiovalue == '' || current.radiovalue == 'nochange') {
1.28      matthew   338:         // They did not check any of the login radiobuttons.
1.89      matthew   339:         alert('$alert{'authen'}');
1.28      matthew   340:         return;
                    341:     }
1.96      raeburn   342: END
                    343:     }
1.97      raeburn   344:     if ($mode eq 'createcourse') {
                    345:         $auth_checks .= "
                    346:     if ( (vf.autoadds[0].checked == true) &&
                    347:          (vf.elements[current.argfield].value == null || vf.elements[current.argfield].value == '') ) {
                    348: ";
                    349:     } elsif ($mode eq 'modifycourse') {
                    350:         $auth_checks .= " 
                    351:     if (vf.elements[current.argfield].value == null || vf.elements[current.argfield].value == '') {
                    352: ";
                    353:     }
1.96      raeburn   354:     if ( ($mode eq 'createcourse') || ($mode eq 'modifycourse') ) {
1.97      raeburn   355:         $auth_checks .= (<<END);
                    356:         var alertmsg = '';
                    357:         switch (current.radiovalue) {
                    358:             case 'krb':
                    359:                 alertmsg = '$alert{'krb'}';
                    360:                 break;
                    361:             default:
                    362:                 alertmsg = '';
1.96      raeburn   363:         }
1.97      raeburn   364:         if (alertmsg != '') {
                    365:             alert(alertmsg);
                    366:             return;
1.96      raeburn   367:         }
                    368:     }
                    369: END
                    370:     } else {
                    371:         $auth_checks .= (<<END);
1.28      matthew   372:     foundatype=1;
1.29      matthew   373:     if (current.argfield == null || current.argfield == '') {
1.28      matthew   374:         var alertmsg = '';
1.29      matthew   375:         switch (current.value) {
1.28      matthew   376:             case 'krb': 
1.89      matthew   377:                 alertmsg = '$alert{'krb'}';
1.28      matthew   378:                 break;
                    379:             case 'loc':
                    380:             case 'fsys':
1.89      matthew   381:                 alertmsg = '$alert{'ipass'}';
1.28      matthew   382:                 break;
                    383:             case 'fsys':
                    384:                 alertmsg = '';
                    385:                 break;
                    386:             default: 
                    387:                 alertmsg = '';
1.3       www       388:         }
1.28      matthew   389:         if (alertmsg != '') {
                    390:             alert(alertmsg);
1.3       www       391:             return;
                    392:         }
                    393:     }
1.89      matthew   394: END
1.96      raeburn   395:     }
1.91      raeburn   396:     my $optional_checks = '';
1.96      raeburn   397:     if ( ($mode eq 'createcourse') || ($mode eq 'modifycourse') ) {
1.91      raeburn   398:         $optional_checks = (<<END);
                    399:     vf.submit();
                    400: }
                    401: END
                    402:     } else {
                    403:         $optional_checks = (<<END);
1.89      matthew   404:     var message='';
                    405:     if (foundname==0) { 
                    406:         message='$alert{'name'}';
1.24      albertel  407:     }
1.89      matthew   408:     if (foundid==0) { 
                    409:         if (message!='') { 
                    410:             message+='\\n'; 
                    411:         }
                    412:         message+='$alert{'snum'}';
                    413:     }
                    414:     if (foundsec==0) {
                    415:         if (message!='') {
                    416:             message+='\\n';
                    417:         } 
                    418:         message+='$alert{'section'}';
                    419:     }
                    420:     if (foundemail==0) {
                    421:         if (message!='') {
                    422:             message+='\\n';
                    423:         }
                    424:         message+='$alert{'email'}';
1.74      matthew   425:     }
                    426:     if (message!='') {
1.89      matthew   427:         message+= '\\n$alert{'continue'}';
                    428:         if (confirm(message)) {
                    429:             vf.state.value='enrolling';
                    430:             vf.submit();
                    431:         }
1.74      matthew   432:     } else {
1.89      matthew   433:         vf.state.value='enrolling';
                    434:         vf.submit();
1.74      matthew   435:     }
                    436: }
1.89      matthew   437: END
1.91      raeburn   438:     }
1.89      matthew   439:     my $result = $function_name;
1.96      raeburn   440:     if ( ($mode eq 'auth') || ($mode eq 'createcourse') || ($mode eq 'modifycourse')  ) {
1.89      matthew   441:         $result .= $auth_checks;
                    442:     }
                    443:     $result .= $optional_checks;
1.96      raeburn   444:     if ( ($mode eq 'auth') || ($mode eq 'createcourse') || ($mode eq 'modifycourse')  ) {
1.89      matthew   445:         $result .= $authheader;
                    446:     }
                    447:     return $result;
1.74      matthew   448: }
                    449: 
1.50      matthew   450: ###############################################################
                    451: ###############################################################
                    452: sub upload_manager_javascript_forward_associate {
1.24      albertel  453:     return(<<ENDPICK);
                    454: function verify(vf) {
                    455:     var founduname=0;
                    456:     var foundpwd=0;
                    457:     var foundname=0;
                    458:     var foundid=0;
                    459:     var foundsec=0;
1.73      www       460:     var foundemail=0;
1.24      albertel  461:     var tw;
                    462:     for (i=0;i<=vf.nfields.value;i++) {
                    463:         tw=eval('vf.f'+i+'.selectedIndex');
                    464:         if (tw==1) { founduname=1; }
                    465:         if ((tw>=2) && (tw<=6)) { foundname=1; }
                    466:         if (tw==7) { foundid=1; }
                    467:         if (tw==8) { foundsec=1; }
                    468:         if (tw==9) { foundpwd=1; }
1.73      www       469:         if (tw==10) { foundemail=1; }
1.24      albertel  470:     }
1.73      www       471:     verify_message(vf,founduname,foundpwd,foundname,foundid,foundsec,foundemail);
1.24      albertel  472: }
                    473: 
1.49      matthew   474: //
                    475: // vf = this.form
                    476: // tf = column number
                    477: //
                    478: // values of nw
                    479: //
                    480: // 0 = none
                    481: // 1 = username
                    482: // 2 = names (lastname, firstnames)
                    483: // 3 = fname (firstname)
                    484: // 4 = mname (middlename)
                    485: // 5 = lname (lastname)
                    486: // 6 = gen   (generation)
                    487: // 7 = id
                    488: // 8 = section
                    489: // 9 = ipwd  (password)
1.73      www       490: // 10 = email address
                    491: 
1.24      albertel  492: function flip(vf,tf) {
                    493:    var nw=eval('vf.f'+tf+'.selectedIndex');
                    494:    var i;
1.49      matthew   495:    // make sure no other columns are labeled the same as this one
1.24      albertel  496:    for (i=0;i<=vf.nfields.value;i++) {
                    497:       if ((i!=tf) && (eval('vf.f'+i+'.selectedIndex')==nw)) {
                    498:           eval('vf.f'+i+'.selectedIndex=0;')
                    499:       }
                    500:    }
1.49      matthew   501:    // If we set this to 'lastname, firstnames', clear out all the ones
                    502:    // set to 'fname','mname','lname','gen' (3,4,5,6) currently.
1.24      albertel  503:    if (nw==2) {
                    504:       for (i=0;i<=vf.nfields.value;i++) {
                    505:          if ((eval('vf.f'+i+'.selectedIndex')>=3) &&
                    506:              (eval('vf.f'+i+'.selectedIndex')<=6)) {
                    507:              eval('vf.f'+i+'.selectedIndex=0;')
                    508:          }
                    509:       }
                    510:    }
1.49      matthew   511:    // If we set this to one of 'fname','mname','lname','gen' (3,4,5,6),
                    512:    // clear out any that are set to 'lastname, firstnames' (2)
1.24      albertel  513:    if ((nw>=3) && (nw<=6)) {
                    514:       for (i=0;i<=vf.nfields.value;i++) {
                    515:          if (eval('vf.f'+i+'.selectedIndex')==2) {
                    516:              eval('vf.f'+i+'.selectedIndex=0;')
                    517:          }
                    518:       }
                    519:    }
1.49      matthew   520:    // If we set the password, make the password form below correspond to 
                    521:    // the new value.
1.24      albertel  522:    if (nw==9) {
1.28      matthew   523:        changed_radio('int',document.studentform);
                    524:        set_auth_radio_buttons('int',document.studentform);
                    525:        vf.intarg.value='';
                    526:        vf.krbarg.value='';
1.24      albertel  527:        vf.locarg.value='';
                    528:    }
                    529: }
                    530: 
                    531: function clearpwd(vf) {
                    532:     var i;
                    533:     for (i=0;i<=vf.nfields.value;i++) {
                    534:         if (eval('vf.f'+i+'.selectedIndex')==9) {
                    535:             eval('vf.f'+i+'.selectedIndex=0;')
                    536:         }
                    537:     }
                    538: }
                    539: 
                    540: ENDPICK
                    541: }
                    542: 
1.50      matthew   543: ###############################################################
                    544: ###############################################################
                    545: sub upload_manager_javascript_reverse_associate {
1.24      albertel  546:     return(<<ENDPICK);
                    547: function verify(vf) {
                    548:     var founduname=0;
                    549:     var foundpwd=0;
                    550:     var foundname=0;
                    551:     var foundid=0;
                    552:     var foundsec=0;
                    553:     var tw;
                    554:     for (i=0;i<=vf.nfields.value;i++) {
                    555:         tw=eval('vf.f'+i+'.selectedIndex');
                    556:         if (i==0 && tw!=0) { founduname=1; }
                    557:         if (((i>=1) && (i<=5)) && tw!=0 ) { foundname=1; }
                    558:         if (i==6 && tw!=0) { foundid=1; }
                    559:         if (i==7 && tw!=0) { foundsec=1; }
                    560:         if (i==8 && tw!=0) { foundpwd=1; }
                    561:     }
                    562:     verify_message(vf,founduname,foundpwd,foundname,foundid,foundsec);
                    563: }
                    564: 
                    565: function flip(vf,tf) {
                    566:    var nw=eval('vf.f'+tf+'.selectedIndex');
                    567:    var i;
                    568:    // picked the all one one name field, reset the other name ones to blank
                    569:    if (tf==1 && nw!=0) {
                    570:       for (i=2;i<=5;i++) {
                    571:          eval('vf.f'+i+'.selectedIndex=0;')
                    572:       }
                    573:    }
                    574:    //picked one of the piecewise name fields, reset the all in
                    575:    //one field to blank
                    576:    if ((tf>=2) && (tf<=5) && (nw!=0)) {
                    577:       eval('vf.f1.selectedIndex=0;')
                    578:    }
                    579:    // intial password specified, pick internal authentication
                    580:    if (tf==8 && nw!=0) {
1.28      matthew   581:        changed_radio('int',document.studentform);
                    582:        set_auth_radio_buttons('int',document.studentform);
                    583:        vf.krbarg.value='';
                    584:        vf.intarg.value='';
1.24      albertel  585:        vf.locarg.value='';
                    586:    }
                    587: }
                    588: 
                    589: function clearpwd(vf) {
                    590:     var i;
                    591:     if (eval('vf.f8.selectedIndex')!=0) {
                    592:         eval('vf.f8.selectedIndex=0;')
                    593:     }
                    594: }
1.2       www       595: ENDPICK
1.23      albertel  596: }
1.10      www       597: 
1.50      matthew   598: ###############################################################
                    599: ###############################################################
                    600: sub print_upload_manager_footer {
1.23      albertel  601:     my ($r,$i,$keyfields,$defdom,$today,$halfyear)=@_;
1.64      albertel  602: 
                    603:     my ($krbdef,$krbdefdom) =
                    604:         &Apache::loncommon::get_kerberos_defaults($defdom);
                    605:     my %param = ( formname => 'document.studentform',
                    606:                   kerb_def_dom => $krbdefdom,
                    607:                   kerb_def_auth => $krbdef
                    608:                   );
1.127     albertel  609:     if (exists($env{'form.ipwd_choice'}) &&
                    610:         defined($env{'form.ipwd_choice'}) &&
                    611:         $env{'form.ipwd_choice'} ne '') {
1.99      matthew   612:         $param{'curr_authtype'} = 'int';
                    613:     }
1.28      matthew   614:     my $krbform = &Apache::loncommon::authform_kerberos(%param);
                    615:     my $intform = &Apache::loncommon::authform_internal(%param);
                    616:     my $locform = &Apache::loncommon::authform_local(%param);
1.31      matthew   617:     my $domform = &domain_form($defdom);
1.68      matthew   618:     my $date_table = &date_setting_table();
1.90      matthew   619:     my $Str = "</table>\n";
                    620:     $Str .= &hidden_input('nfields',$i);
                    621:     $Str .= &hidden_input('keyfields',$keyfields);
                    622:     $Str .= '<h3>'.&mt('Login Type')."</h3>\n";
                    623:     $Str .= "<p>\n".
                    624:         &mt('Note: this will not take effect if the user already exists').
1.130     www       625: 	&Apache::loncommon::help_open_topic('Auth_Options').
1.90      matthew   626:         "</p><p>\n";
                    627:     $Str .= $krbform."\n</p><p>\n".
                    628:         $intform."\n</p><p>\n".
                    629:         $locform."\n</p>\n";
                    630:     $Str .= '<h3>'.&mt('LON-CAPA Domain for Students')."</h3>\n";
                    631:     $Str .= "<p>\n".&mt('LON-CAPA domain: [_1]',$domform)."\n</p>\n";
                    632:     $Str .= "<h3>".&mt('Starting and Ending Dates')."</h3>\n";
                    633:     $Str .= "<p>\n".$date_table."</p>\n";
                    634:     $Str .= "<h3>".&mt('Full Update')."</h3>\n";
1.131     albertel  635:     $Str .= '<label><input type="checkbox" name="fullup" value="yes">'.
1.90      matthew   636:         ' '.&mt('Full update (also print list of users not enrolled anymore)').
1.131     albertel  637:         "</label></p>\n";
1.90      matthew   638:     $Str .= "<h3>".&mt('Student Number')."</h3>\n";
1.131     albertel  639:     $Str .= "<p>\n".'<label><input type="checkbox" name="forceid" value="yes">';
1.90      matthew   640:     $Str .= &mt('Disable ID/Student Number Safeguard and Force Change '.
                    641:                 'of Conflicting IDs (only do if you know what you are doing)').
1.131     albertel  642:                 "</label>\n</p><p>\n";
1.90      matthew   643:     $Str .= '<input type="button" onClick="javascript:verify(this.form)" '.
1.95      albertel  644:         'value="Update Class List" />'."<br />\n";
1.90      matthew   645:     $Str .= &mt('Note: for large courses, this operation may be time '.
                    646:                 'consuming');
                    647:     $r->print($Str);
                    648:     return;
1.23      albertel  649: }
1.24      albertel  650: 
1.90      matthew   651: ###############################################################
                    652: ###############################################################
1.50      matthew   653: sub print_upload_manager_form {
1.23      albertel  654:     my $r=shift;
1.99      matthew   655: 
1.82      www       656:     my $firstLine;
1.24      albertel  657:     my $datatoken;
1.127     albertel  658:     if (!$env{'form.datatoken'}) {
1.90      matthew   659:         $datatoken=&Apache::loncommon::upfile_store($r);
1.24      albertel  660:     } else {
1.127     albertel  661:         $datatoken=$env{'form.datatoken'};
1.90      matthew   662:         &Apache::loncommon::load_tmp_file($r);
1.24      albertel  663:     }
                    664:     my @records=&Apache::loncommon::upfile_record_sep();
1.127     albertel  665:     if($env{'form.noFirstLine'}){
1.90      matthew   666:         $firstLine=shift(@records);
                    667:     }
1.23      albertel  668:     my $total=$#records;
                    669:     my $distotal=$total+1;
                    670:     my $today=time;
                    671:     my $halfyear=$today+15552000;
1.99      matthew   672:     #
                    673:     # Restore memorized settings
                    674:     &Apache::loncommon::restore_course_settings
                    675:         ('enrollment_upload',{ 'username_choice' => 'scalar', # column settings
                    676:                                'names_choice' => 'scalar',
                    677:                                'fname_choice' => 'scalar',
                    678:                                'mname_choice' => 'scalar',
                    679:                                'lname_choice' => 'scalar',
                    680:                                'gen_choice' => 'scalar',
                    681:                                'id_choice' => 'scalar',
                    682:                                'sec_choice' => 'scalar',
                    683:                                'ipwd_choice' => 'scalar',
                    684:                                'email_choice' => 'scalar',
                    685:                            });
                    686:     #
                    687:     # Determine kerberos parameters as appropriate
1.127     albertel  688:     my $defdom=$env{'course.'.$env{'request.course.id'}.'.domain'};
1.64      albertel  689:     my ($krbdef,$krbdefdom) =
                    690:         &Apache::loncommon::get_kerberos_defaults($defdom);
1.99      matthew   691:     #
1.50      matthew   692:     &print_upload_manager_header($r,$datatoken,$distotal,$krbdefdom);
1.24      albertel  693:     my $i;
                    694:     my $keyfields;
1.23      albertel  695:     if ($total>=0) {
1.99      matthew   696:         my @field=
1.127     albertel  697:             (['username',&mt('Username'),     $env{'form.username_choice'}],
                    698:              ['names',&mt('Last Name, First Names'),$env{'form.names_choice'}],
                    699:              ['fname',&mt('First Name'),      $env{'form.fname_choice'}],
                    700:              ['mname',&mt('Middle Names/Initials'),$env{'form.mname_choice'}],
                    701:              ['lname',&mt('Last Name'),       $env{'form.lname_choice'}],
                    702:              ['gen',  &mt('Generation'),      $env{'form.gen_choice'}],
                    703:              ['id',   &mt('ID/Student Number'),$env{'form.id_choice'}],
                    704:              ['sec',  &mt('Group/Section'),   $env{'form.sec_choice'}],
                    705:              ['ipwd', &mt('Initial Password'),$env{'form.ipwd_choice'}],
                    706:              ['email',&mt('EMail Address'),   $env{'form.email_choice'}]);
                    707: 	if ($env{'form.upfile_associate'} eq 'reverse') {	
1.24      albertel  708: 	    &Apache::loncommon::csv_print_samples($r,\@records);
1.90      matthew   709: 	    $i=&Apache::loncommon::csv_print_select_table($r,\@records,
                    710:                                                           \@field);
                    711: 	    foreach (@field) { 
                    712:                 $keyfields.=$_->[0].','; 
                    713:             }
1.24      albertel  714: 	    chop($keyfields);
                    715: 	} else {
1.90      matthew   716: 	    unshift(@field,['none','']);
                    717: 	    $i=&Apache::loncommon::csv_samples_select_table($r,\@records,
                    718:                                                             \@field);
1.24      albertel  719: 	    my %sone=&Apache::loncommon::record_sep($records[0]);
                    720: 	    $keyfields=join(',',sort(keys(%sone)));
1.23      albertel  721: 	}
                    722:     }
1.50      matthew   723:     &print_upload_manager_footer($r,$i,$keyfields,$defdom,$today,$halfyear);
1.10      www       724: }
                    725: 
1.90      matthew   726: ###############################################################
                    727: ###############################################################
1.12      www       728: sub enroll_single_student {
                    729:     my $r=shift;
1.80      matthew   730:     # Remove non alphanumeric values from section
1.127     albertel  731:     $env{'form.csec'}=~s/\W//g;
1.68      matthew   732:     #
                    733:     # We do the dates first because the action of making them the defaul
1.107     www       734:     # in the course is entirely separate from the action of enrolling the
1.68      matthew   735:     # student.  Also, a failure in setting the dates as default is not fatal
                    736:     # to the process of enrolling / modifying a student.
                    737:     my ($startdate,$enddate) = &get_dates_from_form();
1.127     albertel  738:     if ($env{'form.makedatesdefault'}) {
1.68      matthew   739:         $r->print(&make_dates_default($startdate,$enddate));
                    740:     }
                    741: 
1.94      sakharuk  742:     $r->print('<h3>'.&mt('Enrolling Student').'</h3>');
1.127     albertel  743:     $r->print('<p>'.&mt('Enrolling').' '.$env{'form.cuname'}." \@ ".
                    744:               $env{'form.lcdomain'}.'</p>');
                    745:     if (($env{'form.cuname'})&&($env{'form.cuname'}!~/\W/)&&
                    746:         ($env{'form.lcdomain'})&&($env{'form.lcdomain'}!~/\W/)) {
1.31      matthew   747:         # Deal with home server selection
1.127     albertel  748:         my $domain=$env{'form.lcdomain'};
                    749:         my $desiredhost = $env{'form.lcserver'};
1.31      matthew   750:         if (lc($desiredhost) eq 'default') {
                    751:             $desiredhost = undef;
                    752:         } else {
1.45      matthew   753:             my %home_servers =&Apache::loncommon::get_library_servers($domain);
1.31      matthew   754:             if (! exists($home_servers{$desiredhost})) {
1.94      sakharuk  755:                 $r->print('<font color="#ff0000">'.&mt('Error').':</font>'.
                    756:                           &mt('Invalid home server specified'));
1.31      matthew   757:                 return;
                    758:             }
                    759:         }
1.94      sakharuk  760:         $r->print(" ".&mt('with server')." $desiredhost :") if (defined($desiredhost));
1.31      matthew   761:         # End of home server selection logic
1.12      www       762: 	my $amode='';
                    763:         my $genpwd='';
1.127     albertel  764:         if ($env{'form.login'} eq 'krb') {
1.47      albertel  765:            $amode='krb';
1.127     albertel  766: 	   $amode.=$env{'form.krbver'};
                    767:            $genpwd=$env{'form.krbarg'};
                    768:         } elsif ($env{'form.login'} eq 'int') {
1.26      matthew   769:            $amode='internal';
1.127     albertel  770:            $genpwd=$env{'form.intarg'};
                    771:         }  elsif ($env{'form.login'} eq 'loc') {
1.15      albertel  772: 	    $amode='localauth';
1.127     albertel  773: 	    $genpwd=$env{'form.locarg'};
1.15      albertel  774: 	    if (!$genpwd) { $genpwd=" "; }
                    775: 	}
1.127     albertel  776:         my $home = &Apache::lonnet::homeserver($env{'form.cuname'},
                    777:                                                    $env{'form.lcdomain'});
1.34      matthew   778:         if ((($amode) && ($genpwd)) || ($home ne 'no_host')) {
1.55      matthew   779:             # Clean out any old roles the student has in this class.
1.127     albertel  780:             &modifystudent($env{'form.lcdomain'},$env{'form.cuname'},
                    781:                            $env{'request.course.id'},$env{'form.csec'},
1.33      matthew   782:                             $desiredhost);
1.55      matthew   783:             my $login_result = &Apache::lonnet::modifystudent
1.127     albertel  784:                 ($env{'form.lcdomain'},$env{'form.cuname'},
                    785:                  $env{'form.cstid'},$amode,$genpwd,
                    786:                  $env{'form.cfirst'},$env{'form.cmiddle'},
                    787:                  $env{'form.clast'},$env{'form.cgen'},
                    788:                  $env{'form.csec'},$enddate,
                    789:                  $startdate,$env{'form.forceid'},
                    790:                  $desiredhost,$env{'form.emailaddress'});
1.55      matthew   791:             if ($login_result =~ /^ok/) {
                    792:                 $r->print($login_result);
1.94      sakharuk  793:                 $r->print("<p> ".&mt('If active, the new role will be available when the student next logs in to LON-CAPA.')."</p>");
1.55      matthew   794:             } else {
1.94      sakharuk  795:                 $r->print(&mt('unable to enroll').": ".$login_result);
1.55      matthew   796:             }
1.12      www       797: 	} else {
1.94      sakharuk  798:             $r->print('<p><font color="#ff0000">'.&mt('ERROR').'</font>&nbsp;');
1.79      matthew   799:             if ($amode =~ /^krb/) {
1.94      sakharuk  800:                 $r->print(&mt('Missing Kerberos domain information.').'  ');
1.79      matthew   801:             } else {
1.94      sakharuk  802:                 $r->print(&mt('Invalid login mode or password.').'  ');
1.79      matthew   803:             }
1.127     albertel  804:             $r->print('<b>'.&mt('Unable to enroll').' '.$env{'form.cuname'}.'.</b></p>');
1.79      matthew   805:         }
1.12      www       806:     } else {
1.94      sakharuk  807:         $r->print(&mt('Invalid username or domain'));
1.26      matthew   808:     }    
1.12      www       809: }
                    810: 
1.68      matthew   811: sub setup_date_selectors {
1.91      raeburn   812:     my ($starttime,$endtime,$mode) = @_;
1.68      matthew   813:     if (! defined($starttime)) {
                    814:         $starttime = time;
1.114     raeburn   815:         unless ($mode eq 'create_enrolldates' || $mode eq 'create_defaultdates') {
1.127     albertel  816:             if (exists($env{'course.'.$env{'request.course.id'}.
1.68      matthew   817:                             '.default_enrollment_start_date'})) {
1.127     albertel  818:                 $starttime = $env{'course.'.$env{'request.course.id'}.
1.68      matthew   819:                                   '.default_enrollment_start_date'};
1.91      raeburn   820:             }
1.68      matthew   821:         }
                    822:     }
                    823:     if (! defined($endtime)) {
                    824:         $endtime = time+(6*30*24*60*60); # 6 months from now, approx
1.91      raeburn   825:         unless ($mode eq 'createcourse') {
1.127     albertel  826:             if (exists($env{'course.'.$env{'request.course.id'}.
1.68      matthew   827:                             '.default_enrollment_end_date'})) {
1.127     albertel  828:                 $endtime = $env{'course.'.$env{'request.course.id'}.
1.68      matthew   829:                                 '.default_enrollment_end_date'};
1.91      raeburn   830:             }
1.68      matthew   831:         }
                    832:     }
                    833:     my $startdateform = &Apache::lonhtmlcommon::date_setter('studentform',
                    834:                                                             'startdate',
                    835:                                                             $starttime);
                    836:     my $enddateform = &Apache::lonhtmlcommon::date_setter('studentform',
                    837:                                                           'enddate',
                    838:                                                           $endtime);
1.114     raeburn   839:     if ($mode eq 'create_enrolldates') {
                    840:         $startdateform = &Apache::lonhtmlcommon::date_setter('ccrs',
                    841:                                                             'startenroll',
                    842:                                                             $starttime);
                    843:         $enddateform = &Apache::lonhtmlcommon::date_setter('ccrs',
                    844:                                                           'endenroll',
                    845:                                                           $endtime);
                    846:     }
                    847:     if ($mode eq 'create_defaultdates') {
1.91      raeburn   848:         $startdateform = &Apache::lonhtmlcommon::date_setter('ccrs',
1.114     raeburn   849:                                                             'startaccess',
1.91      raeburn   850:                                                             $starttime);
                    851:         $enddateform = &Apache::lonhtmlcommon::date_setter('ccrs',
1.114     raeburn   852:                                                           'endaccess',
1.91      raeburn   853:                                                           $endtime);
                    854:     }
1.68      matthew   855:     return ($startdateform,$enddateform);
                    856: }
                    857: 
                    858: sub get_dates_from_form {
                    859:     my $startdate = &Apache::lonhtmlcommon::get_date_from_form('startdate');
                    860:     my $enddate   = &Apache::lonhtmlcommon::get_date_from_form('enddate');
1.127     albertel  861:     if ($env{'form.no_end_date'}) {
1.68      matthew   862:         $enddate = 0;
                    863:     }
                    864:     return ($startdate,$enddate);
                    865: }
                    866: 
                    867: sub date_setting_table {
1.91      raeburn   868:     my ($starttime,$endtime,$mode) = @_;
                    869:     my ($startform,$endform)=&setup_date_selectors($starttime,$endtime,$mode);
1.68      matthew   870:     my $dateDefault = '<nobr>'.
1.131     albertel  871:         '<label><input type="checkbox" name="makedatesdefault" /> '.
                    872:         &mt('make these dates the default for future enrollment').'</label>';
1.114     raeburn   873:     if ($mode eq 'create_enrolldates' || $mode eq 'create_defaultdates') {
1.91      raeburn   874:         $dateDefault = '&nbsp;';
                    875:     }
1.131     albertel  876:     my $perpetual = '<nobr><label><input type="checkbox" name="no_end_date"';
1.68      matthew   877:     if (defined($endtime) && $endtime == 0) {
                    878:         $perpetual .= ' checked';
                    879:     }
1.131     albertel  880:     $perpetual.= ' /> '.&mt('no ending date').'</label></nobr>';
1.114     raeburn   881:     if ($mode eq 'create_enrolldates') {
                    882:         $perpetual = '&nbsp;';
                    883:     }
1.68      matthew   884:     my $result = '';
                    885:     $result .= "<table>\n";
1.94      sakharuk  886:     $result .= '<tr><td align="right">'.&mt('Starting Date').'</td>'.
1.68      matthew   887:         '<td>'.$startform.'</td>'.
                    888:         '<td>'.$dateDefault.'</td>'."</tr>\n";
1.94      sakharuk  889:     $result .= '<tr><td align="right">'.&mt('Ending Date').'</td>'.
1.68      matthew   890:         '<td>'.$endform.'</td>'.
                    891:         '<td>'.$perpetual.'</td>'."</tr>\n";
                    892:     $result .= "</table>\n";
                    893:     return $result;
                    894: }
                    895: 
                    896: sub make_dates_default {
                    897:     my ($startdate,$enddate) = @_;
                    898:     my $result = '';
1.127     albertel  899:     my $dom = $env{'course.'.$env{'request.course.id'}.'.domain'};
                    900:     my $crs = $env{'course.'.$env{'request.course.id'}.'.num'};
1.68      matthew   901:     my $put_result = &Apache::lonnet::put('environment',
                    902:             {'default_enrollment_start_date'=>$startdate,
                    903:              'default_enrollment_end_date'  =>$enddate},$dom,$crs);
                    904:     if ($put_result eq 'ok') {
                    905:         $result .= "Set default start and end dates for course<br />";
1.69      matthew   906:         #
                    907:         # Refresh the course environment
1.140     albertel  908:         &Apache::lonnet::coursedescription($env{'request.course.id'},
                    909: 					   {'freshen_cache' => 1});
1.68      matthew   910:     } else {
1.94      sakharuk  911:         $result .= &mt('Unable to set default dates for course').":".$put_result.
1.68      matthew   912:             '<br />';
                    913:     }
                    914:     return $result;
                    915: }
                    916: 
1.74      matthew   917: ##
                    918: ## Single student enrollment routines (some of them)
                    919: ##
                    920: sub get_student_username_domain_form {
                    921:     my $r = shift;
                    922:     my $domform = &Apache::loncommon::select_dom_form
1.127     albertel  923:         ($env{'course.'.$env{'request.course.id'}.'.domain'},'cudomain',0);
1.94      sakharuk  924:     my %lt=&Apache::lonlocal::texthash(
                    925: 		    'eos'  => "Enroll One Student",
                    926: 		    'usr'  => "Username",
                    927:                     'dom'  => "Domain",
                    928:                     'been' => "Begin Enrollment",
                    929: 				       );
1.74      matthew   930:     $r->print(<<END);
                    931: <input type="hidden" name="action" value="enrollstudent" />
                    932: <input type="hidden" name="state"  value="gotusername" />
1.94      sakharuk  933: <h3>$lt{'eos'}</h3>
1.74      matthew   934: <table>
1.94      sakharuk  935: <tr><th>$lt{'usr'}:</th>
1.74      matthew   936:     <td><input type="text" name="cuname"  size="15" /></td></tr>
1.94      sakharuk  937: <tr><th>$lt{'dom'}:</th>
1.74      matthew   938:     <td>$domform</td></tr>
                    939: <tr><th>&nbsp;</th>
                    940:     <td>
1.94      sakharuk  941:     <input type="submit" name="Begin Enrollment" value="$lt{'been'}" />
1.74      matthew   942:     </td></tr>
                    943: </table>
1.120     albertel  944: <script type="text/javascript">
                    945: // the if prevents the script error if the browser can not handle this
                    946: if ( document.studentform.cuname ) { document.studentform.cuname.focus(); }
                    947: </script>
1.74      matthew   948: END
                    949:     return;
                    950: }
                    951: 
1.50      matthew   952: sub print_enroll_single_student_form {
1.10      www       953:     my $r=shift;
1.94      sakharuk  954:     $r->print("<h3>".&mt('Enroll One Student')."</h3>");
1.74      matthew   955:     #
1.127     albertel  956:     my $username = $env{'form.cuname'};
                    957:     my $domain   = $env{'form.cudomain'};
1.123     albertel  958:     $username=~s/\W//gs;
                    959:     $domain=~s/\W//gs;
1.74      matthew   960:     my $home = &Apache::lonnet::homeserver($username,$domain);
                    961:     # $new_user flags whether we are creating a new user or using an old one
                    962:     my $new_user = 1;
                    963:     if ($home ne 'no_host') {
                    964:         $new_user = 0;
                    965:     }
                    966:     #
                    967:     my $user_data_html = '';
                    968:     my $javascript_validations = '';
                    969:     if ($new_user) {
1.127     albertel  970:         my $defdom=$env{'course.'.$env{'request.course.id'}.'.domain'};
1.74      matthew   971:         # Set up authentication forms
                    972:         my ($krbdef,$krbdefdom) =
1.75      matthew   973:             &Apache::loncommon::get_kerberos_defaults($domain);
1.89      matthew   974:         $javascript_validations=&javascript_validations('auth',$krbdefdom);
1.74      matthew   975:         my %param = ( formname => 'document.studentform',
                    976:                       kerb_def_dom => $krbdefdom,
                    977:                       kerb_def_auth => $krbdef
                    978:                       );
                    979:         my $krbform = &Apache::loncommon::authform_kerberos(%param);
                    980:         my $intform = &Apache::loncommon::authform_internal(%param);
                    981:         my $locform = &Apache::loncommon::authform_local(%param);
                    982:         #
                    983:         # Set up domain selection form
                    984:         my $homeserver_form = '';
                    985:         my %servers = &Apache::loncommon::get_library_servers($domain);
                    986:         $homeserver_form = '<select name="lcserver" size="1">'."\n".
                    987:             '<option value="default" selected>default</option>'."\n";
                    988:         while (my ($servername,$serverdescription) = each (%servers)) {
                    989:             $homeserver_form .= '<option value="'.$servername.'">'.
                    990:                 $serverdescription."</option>\n";
                    991:         }
                    992:         $homeserver_form .= "</select>\n";
                    993:         #
                    994:         #
1.94      sakharuk  995: 	my %lt=&Apache::lonlocal::texthash(
                    996: 		       'udf'  => "User Data for",
                    997:                        'fn'   => "First Name",
                    998:                        'mn'   => "Middle Name",
                    999:                        'ln'   => "Last Name",
                   1000:                        'gen'  => "Generation",
                   1001:                        'hs'   => "Home Server",
                   1002:                        'pswd' => "Password",
                   1003: 		       'psam' => "Please select an authentication mechanism",
1.124     www      1004:                        'mail' => "Email Address"
1.94      sakharuk 1005: 					   );
1.130     www      1006: 	my $authhelp=&Apache::loncommon::help_open_topic('Auth_Options');
1.74      matthew  1007:         $user_data_html = <<END;
1.94      sakharuk 1008: <h3>$lt{'udf'} $username\@$domain</h3>
1.74      matthew  1009: <table>
1.94      sakharuk 1010: <tr><th>$lt{'fn'}:</th>
1.74      matthew  1011:     <td><input type="text" name="cfirst"  size="15"></td></tr>
1.94      sakharuk 1012: <tr><th>$lt{'mn'}:</th>
1.74      matthew  1013:     <td><input type="text" name="cmiddle" size="15"></td></tr>
1.94      sakharuk 1014: <tr><th>$lt{'ln'}:</th>
1.74      matthew  1015:     <td><input type="text" name="clast"   size="15"></td></tr>
1.94      sakharuk 1016: <tr><th>$lt{'gen'}:</th>
1.74      matthew  1017:     <td><input type="text" name="cgen"    size="5"> </td></tr>
1.94      sakharuk 1018: <tr><th>$lt{'hs'}:</th>
1.74      matthew  1019:     <td>$homeserver_form</td></tr>
1.124     www      1020: <tr><th>$lt{'mail'}:</th>
                   1021:     <td><input type="text" name="emailaddress" size="20" /></td></tr>
1.74      matthew  1022: </table>
1.94      sakharuk 1023: <h3>$lt{'pswd'}</h3>
1.130     www      1024: $lt{'psam'}$authhelp
1.74      matthew  1025: <table>
                   1026: <p>
                   1027: $krbform
1.75      matthew  1028: <br />
1.74      matthew  1029: $intform
1.75      matthew  1030: <br />
1.74      matthew  1031: $locform
                   1032: </p>
                   1033: END
                   1034:     } else {
                   1035:         # User already exists.  Do not worry about authentication
                   1036:         my %uenv = &Apache::lonnet::dump('environment',$domain,$username);
1.89      matthew  1037:         $javascript_validations = &javascript_validations('noauth');
1.94      sakharuk 1038: 	my %lt=&Apache::lonlocal::texthash(
                   1039: 		       'udf'  => "User Data for",
                   1040:                        'fn'   => "First Name",
                   1041:                        'mn'   => "Middle Name",
                   1042:                        'ln'   => "Last Name",
                   1043:                        'gen'  => "Generation",
1.124     www      1044:                        'mail' => "Email Address",
1.94      sakharuk 1045: 					   );
1.74      matthew  1046:         $user_data_html = <<END;
1.94      sakharuk 1047: <h3>$lt{'udf'} $username\@$domain</h3>
1.74      matthew  1048: <input type="hidden" name="lcserver" value="default" />
                   1049: <table>
1.94      sakharuk 1050: <tr><th>$lt{'fn'}:</th>
1.74      matthew  1051:     <td>
                   1052:     <input type="text" name="cfirst" value="$uenv{'firstname'}" size="15" />
                   1053:     </td></tr>
1.94      sakharuk 1054: <tr><th>$lt{'mn'}:</th>
1.74      matthew  1055:     <td>
                   1056:     <input type="text" name="cmiddle" value="$uenv{'middlename'}" size="15" />
                   1057:     </td></tr>
1.94      sakharuk 1058: <tr><th>$lt{'ln'}:</th>
1.74      matthew  1059:     <td>
                   1060:     <input type="text" name="clast"value="$uenv{'lastname'}" size="15" />
                   1061:     </td></tr>
1.94      sakharuk 1062: <tr><th>$lt{'gen'}:</th>
1.74      matthew  1063:     <td>
                   1064:     <input type="text" name="cgen" value="$uenv{'generation'}" size="5" />
                   1065:     </td></tr>
1.124     www      1066: <tr><th>$lt{'mail'}:</th>
                   1067:     <td>
                   1068:     <input type="text" name="emailaddress" value="$uenv{'permanentemail'}" size="20" />
                   1069:     </td></tr>
1.74      matthew  1070: </table>
                   1071: END
                   1072:     }
1.68      matthew  1073:     my $date_table = &date_setting_table();
1.74      matthew  1074:         # Print it all out
1.94      sakharuk 1075:     my %lt=&Apache::lonlocal::texthash(
                   1076: 		   'cd'   => "Course Data",
                   1077:                    'gs'   => "Group/Section",
                   1078:                    'idsn' => "ID/Student Number",
                   1079:                    'disn' => "Disable ID/Student Number Safeguard and Force Change of Conflicting IDs (only do if you know what you are doing)",
                   1080:                    'eas'  => "Enroll as student",
                   1081: 				       );
1.50      matthew  1082:     $r->print(<<END);
1.74      matthew  1083: <input type="hidden" name="action" value="enrollstudent" />
                   1084: <input type="hidden" name="state"  value="done" />
                   1085: <input type="hidden" name="cuname" value="$username" />
                   1086: <input type="hidden" name="lcdomain" value="$domain" />
1.28      matthew  1087: <script type="text/javascript" language="Javascript">
1.12      www      1088: function verify(vf) {
                   1089:     var founduname=0;
                   1090:     var foundpwd=0;
                   1091:     var foundname=0;
                   1092:     var foundid=0;
                   1093:     var foundsec=0;
                   1094:     var tw;
1.26      matthew  1095:     if ((typeof(vf.cuname.value) !="undefined") && (vf.cuname.value!='') && 
1.31      matthew  1096: 	(typeof(vf.lcdomain.value)!="undefined") && (vf.lcdomain.value!='')) {
1.12      www      1097:         founduname=1;
                   1098:     }
1.14      harris41 1099:     if ((typeof(vf.cfirst.value)!="undefined") && (vf.cfirst.value!='') &&
1.26      matthew  1100: 	(typeof(vf.clast.value) !="undefined") && (vf.clast.value!='')) {
1.12      www      1101:         foundname=1;
                   1102:     }
1.14      harris41 1103:     if ((typeof(vf.csec.value)!="undefined") && (vf.csec.value!='')) {
1.12      www      1104:         foundsec=1;
                   1105:     }
1.14      harris41 1106:     if ((typeof(vf.cstid.value)!="undefined") && (vf.cstid.value!='')) {
1.12      www      1107: 	foundid=1;
                   1108:     }
                   1109:     if (founduname==0) {
                   1110: 	alert('You need to specify at least the username and domain fields');
                   1111:         return;
                   1112:     }
1.24      albertel 1113:     verify_message(vf,founduname,foundpwd,foundname,foundid,foundsec);
1.12      www      1114: }
                   1115: 
1.24      albertel 1116: $javascript_validations
1.12      www      1117: 
1.24      albertel 1118: function clearpwd(vf) {
                   1119:     //nothing else needs clearing
1.15      albertel 1120: }
                   1121: 
1.12      www      1122: </script>
1.11      www      1123: 
1.74      matthew  1124: $user_data_html
1.50      matthew  1125: 
1.94      sakharuk 1126: <h3>$lt{'cd'}</h3>
1.50      matthew  1127: 
1.94      sakharuk 1128: <p>$lt{'gs'}: <input type="text" name="csec" size="5" />
1.50      matthew  1129: <p>
1.68      matthew  1130: $date_table
1.50      matthew  1131: </p>
1.94      sakharuk 1132: <h3>$lt{'idsn'}</h3>
1.50      matthew  1133: <p>
1.94      sakharuk 1134: $lt{'idsn'}: <input type="text" name="cstid" size="10">
1.26      matthew  1135: </p><p>
1.131     albertel 1136: <label>
1.26      matthew  1137: <input type="checkbox" name="forceid" value="yes"> 
1.94      sakharuk 1138: $lt{'disn'}
1.131     albertel 1139: </label>
1.50      matthew  1140: </p><p>
1.94      sakharuk 1141: <input type="button" onClick="verify(this.form)" value="$lt{'eas'}">
1.26      matthew  1142: </p>
1.50      matthew  1143: END
                   1144:     return;
1.10      www      1145: }
                   1146: 
                   1147: # ========================================================= Menu Phase Two Drop
1.51      matthew  1148: sub print_drop_menu {
1.10      www      1149:     my $r=shift;
1.92      sakharuk 1150:     $r->print("<h3>".&mt('Drop Students')."</h3>");
1.127     albertel 1151:     my $cid=$env{'request.course.id'};
1.56      matthew  1152:     my ($classlist,$keylist) = &Apache::loncoursedata::get_classlist();
                   1153:     if (! defined($classlist)) {
1.94      sakharuk 1154:         $r->print(&mt('There are no students currently enrolled.')."\n");
1.51      matthew  1155:         return;
1.25      matthew  1156:     }
1.51      matthew  1157:     # Print out the available choices
1.56      matthew  1158:     &show_drop_list($r,$classlist,$keylist);
1.51      matthew  1159:     return;
1.11      www      1160: }
                   1161: 
1.40      matthew  1162: # ============================================== view classlist
1.50      matthew  1163: sub print_html_classlist {
1.103     matthew  1164:     my ($r,$mode) = @_;
1.127     albertel 1165:     if (! exists($env{'form.sortby'})) {
                   1166:         $env{'form.sortby'} = 'username';
1.57      matthew  1167:     }
1.127     albertel 1168:     if ($env{'form.Status'} !~ /^(Any|Expired|Active)$/) {
                   1169:         $env{'form.Status'} = 'Active';
1.57      matthew  1170:     }
                   1171:     my $status_select = &Apache::lonhtmlcommon::StatusOptions
1.127     albertel 1172:         ($env{'form.Status'});
                   1173:     my $cid=$env{'request.course.id'};
                   1174:     my $cdom=$env{'course.'.$cid.'.domain'};
                   1175:     my $cnum=$env{'course.'.$cid.'.num'};
1.103     matthew  1176:     #
                   1177:     # List course personnel
1.100     www      1178:     my %coursepersonnel=&Apache::lonnet::get_course_adv_roles($cdom.'/'.$cnum);
1.110     matthew  1179:     #
1.127     albertel 1180:     if (! defined($env{'form.output'}) ||
                   1181:         $env{'form.output'} !~ /^(csv|excel|html)$/ ) {
                   1182:         $env{'form.output'} = 'html';
1.110     matthew  1183:     }
                   1184:     #
1.139     albertel 1185:     $r->print('<br />'.&Apache::loncommon::start_data_table());
1.110     matthew  1186:     foreach my $role (sort keys %coursepersonnel) {
                   1187:         next if ($role =~ /^\s*$/);
1.139     albertel 1188: 	$r->print(&Apache::loncommon::start_data_table_row().
                   1189: 		  '<td>'.$role.'</td><td>');
1.110     matthew  1190:         foreach my $user (split(',',$coursepersonnel{$role})) {
                   1191: 	    my ($puname,$pudom)=split(':',$user);
1.100     www      1192: 	    $r->print(' '.&Apache::loncommon::aboutmewrapper(
1.110     matthew  1193:                                     &Apache::loncommon::plainname($puname,
                   1194:                                                                   $pudom),
                   1195:                                                              $puname,$pudom));
1.100     www      1196: 	}
1.139     albertel 1197:         $r->print('</td>'.&Apache::loncommon::end_data_table_row());
1.100     www      1198:     }
1.139     albertel 1199:     $r->print(&Apache::loncommon::end_data_table());
1.103     matthew  1200:     #
                   1201:     # Interface output
                   1202:     $r->print('<input type="hidden" name="action" value="'.
1.127     albertel 1203:               $env{'form.action'}.'" />');
1.103     matthew  1204:     $r->print("<p>\n");
1.127     albertel 1205:     if ($env{'form.action'} ne 'modifystudent') {
1.103     matthew  1206: 	my %lt=&Apache::lonlocal::texthash('csv' => "CSV",
                   1207:                                            'excel' => "Excel",
                   1208:                                            'html'  => 'HTML');
1.110     matthew  1209:         my $output_selector = '<select size="1" name="output" >';
1.103     matthew  1210:         foreach my $outputformat ('html','csv','excel') {
                   1211:             my $option = '<option value="'.$outputformat.'" ';
1.127     albertel 1212:             if ($outputformat eq $env{'form.output'}) {
1.104     matthew  1213:                 $option .= 'selected ';
1.103     matthew  1214:             }
                   1215:             $option .='>'.$lt{$outputformat}.'</option>';
                   1216:             $output_selector .= "\n".$option;
                   1217:         }
                   1218:         $output_selector .= '</select>';
                   1219:         $r->print(&mt('Output Format: [_1]',$output_selector).('&nbsp;'x3));
1.59      matthew  1220:     }
1.103     matthew  1221:     $r->print(&mt('Student Status: [_1]',$status_select)."\n");
1.105     matthew  1222:     $r->print('<input type="submit" value="'.&mt('Update Display').'" />'.
                   1223:               "\n</p>\n");
1.103     matthew  1224:     #
                   1225:     # Print the classlist
                   1226:     $r->print('<h2>'.&mt('Current Class List').'</h2>');
1.56      matthew  1227:     my ($classlist,$keylist)=&Apache::loncoursedata::get_classlist();
                   1228:     if (! defined($classlist)) {
1.94      sakharuk 1229:         $r->print(&mt('There are no students currently enrolled.')."\n");
1.40      matthew  1230:     } else {
                   1231:         # Print out the available choices
1.127     albertel 1232:         if ($env{'form.action'} eq 'modifystudent') {
1.110     matthew  1233:             &show_class_list($r,'view','modify',
1.127     albertel 1234:                              $env{'form.Status'},$classlist,$keylist);
1.110     matthew  1235:         } else {
1.127     albertel 1236:             &show_class_list($r,$env{'form.output'},'aboutme',
                   1237:                              $env{'form.Status'},$classlist,$keylist);
1.50      matthew  1238:         }
1.41      matthew  1239:     }
                   1240: }
                   1241: 
1.40      matthew  1242: # =================================================== Show student list to drop
                   1243: sub show_class_list {
1.110     matthew  1244:     my ($r,$mode,$linkto,$statusmode,$classlist,$keylist)=@_;
1.127     albertel 1245:     my $cid=$env{'request.course.id'};
1.60      matthew  1246:     #
                   1247:     # Variables for excel output
1.104     matthew  1248:     my ($excel_workbook, $excel_sheet, $excel_filename,$row,$format);
1.60      matthew  1249:     #
1.103     matthew  1250:     # Variables for csv output
                   1251:     my ($CSVfile,$CSVfilename);
                   1252:     #
1.127     albertel 1253:     my $sortby = $env{'form.sortby'};
1.114     raeburn  1254:     if ($sortby !~ /^(username|domain|section|fullname|id|start|end|type)$/) {
1.53      matthew  1255:         $sortby = 'username';
                   1256:     }
1.134     raeburn  1257:     if (! exists($env{'form.displayphotos'})) {
                   1258:         $env{'form.displayphotos'} = 'off';
                   1259:     }
                   1260:     my $displayphotos = $env{'form.displayphotos'};
                   1261: 
1.42      matthew  1262:     # Print out header 
1.114     raeburn  1263:     unless ($mode eq 'autoenroll') {
                   1264:         $r->print(<<END);
1.127     albertel 1265: <input type="hidden" name="state" value="$env{'form.state'}" />
1.114     raeburn  1266: END
                   1267:     }
1.103     matthew  1268:     $r->print(<<END);
                   1269: <input type="hidden" name="sortby" value="$sortby" />
1.134     raeburn  1270: <input type="hidden" name="displayphotos" value="$displayphotos" />
1.103     matthew  1271: END
1.114     raeburn  1272:     if ($mode eq 'html' || $mode eq 'view' || $mode eq 'autoenroll') {
1.50      matthew  1273:         if ($linkto eq 'aboutme') {
1.94      sakharuk 1274:             $r->print(&mt('Select a user name to view the users personal page.'));
1.50      matthew  1275:         } elsif ($linkto eq 'modify') {
1.94      sakharuk 1276:             $r->print(&mt('Select a user name to modify the students information'));
1.50      matthew  1277:         }
1.94      sakharuk 1278: 	my %lt=&Apache::lonlocal::texthash(
1.110     matthew  1279:                                            'usrn'   => "username",
                   1280:                                            'dom'    => "domain",
                   1281:                                            'sn'     => "student name",
                   1282:                                            'sec'    => "section",
                   1283:                                            'start'  => "start date",
                   1284:                                            'end'    => "end date",
1.134     raeburn  1285:                                            'type'   => "enroll type/action",
                   1286:                                            'photo'  => "photo",
1.94      sakharuk 1287: 					   );
1.114     raeburn  1288:         unless ($mode eq 'autoenroll') {
                   1289:             $r->print(<<END);
1.59      matthew  1290: <input type="hidden" name="sname"  value="" />
                   1291: <input type="hidden" name="sdom"   value="" />
1.114     raeburn  1292: END
                   1293:         }
1.136     raeburn  1294:         if ($env{'course.'.$env{'request.course.id'}.'.internal.showphoto'}) {
1.134     raeburn  1295:             $r->print('
                   1296: <script type="text/javascript">
                   1297: function photowindow(photolink) {
                   1298:     var title = "Photo_Viewer";
                   1299:     var options = "scrollbars=1,resizable=1,menubar=0";
                   1300:     options += ",width=240,height=240";
                   1301:     stdeditbrowser = open(photolink,title,options,"1");
                   1302:     stdeditbrowser.focus();
                   1303: }
                   1304: </script>
                   1305:            ');
                   1306:         }
1.115     raeburn  1307:         $r->print("
1.40      matthew  1308: <p>
1.139     albertel 1309: ".&Apache::loncommon::start_data_table()."
1.115     raeburn  1310: <tr>
                   1311:         ");
                   1312:         if ($mode eq 'autoenroll') {
                   1313:             $r->print("
                   1314:  <th><a href=\"javascript:document.studentform.sortby.value='type';document.studentform.submit();\">$lt{'type'}</a></th>
                   1315:             ");
                   1316:         } else {
                   1317:             $r->print("
                   1318: <th>Count</th>
                   1319:             ");
                   1320:         }
                   1321:         $r->print(<<END);
                   1322:     <th>
1.94      sakharuk 1323:        <a href="javascript:document.studentform.sortby.value='username';document.studentform.submit();">$lt{'usrn'}</a>
1.53      matthew  1324:     </th><th>
1.94      sakharuk 1325:        <a href="javascript:document.studentform.sortby.value='domain';document.studentform.submit();">$lt{'dom'}</a>
1.53      matthew  1326:     </th><th>
1.57      matthew  1327:        <a href="javascript:document.studentform.sortby.value='id';document.studentform.submit();">ID</a>
1.53      matthew  1328:     </th><th>
1.94      sakharuk 1329:        <a href="javascript:document.studentform.sortby.value='fullname';document.studentform.submit();">$lt{'sn'}</a>
1.53      matthew  1330:     </th><th>
1.94      sakharuk 1331:        <a href="javascript:document.studentform.sortby.value='section';document.studentform.submit();">$lt{'sec'}</a>
1.110     matthew  1332:     </th><th>
                   1333:        <a href="javascript:document.studentform.sortby.value='start';document.studentform.submit();">$lt{'start'}</a>
                   1334:     </th><th>
                   1335:        <a href="javascript:document.studentform.sortby.value='end';document.studentform.submit();">$lt{'end'}</a>
1.53      matthew  1336:     </th>
1.40      matthew  1337: END
1.136     raeburn  1338:         if ($env{'course.'.$env{'request.course.id'}.'.internal.showphoto'}) {
1.135     albertel 1339:             my %photo_options = &Apache::lonlocal::texthash(
1.134     raeburn  1340:                                                             'on' => 'Show',
                   1341:                                                             'off' => 'Hide',
                   1342:                                                             );
                   1343:             my $photochg = 'on';
                   1344:             if ($displayphotos eq 'on') {
                   1345:                 $photochg = 'off';
                   1346:             }
                   1347:             $r->print('    <th>'."\n".'     '. 
                   1348:             '<a href="javascript:document.studentform.displayphotos.value='.
                   1349:                       "'".$photochg."'".';document.studentform.submit();">'.
                   1350:                       $photo_options{$photochg}.'</a>&nbsp;'.$lt{'photo'}."\n".
                   1351:                       '    </th>'."\n");
                   1352:         }
                   1353:         $r->print("  </tr>\n");
1.41      matthew  1354:     } elsif ($mode eq 'csv') {
1.103     matthew  1355: 	#
                   1356: 	# Open a file
                   1357: 	$CSVfilename = '/prtspool/'.
1.127     albertel 1358: 	    $env{'user.name'}.'_'.$env{'user.domain'}.'_'.
1.103     matthew  1359:             time.'_'.rand(1000000000).'.csv';
                   1360: 	unless ($CSVfile = Apache::File->new('>/home/httpd'.$CSVfilename)) {
                   1361: 	    $r->log_error("Couldn't open $CSVfilename for output $!");
                   1362: 	    $r->print("Problems occured in writing the csv file.  ".
                   1363: 		      "This error has been logged.  ".
                   1364: 		      "Please alert your LON-CAPA administrator.");
                   1365: 	    $CSVfile = undef;
                   1366: 	}
                   1367: 	#
                   1368: 	# Write headers and data to file
1.58      matthew  1369:         if($statusmode eq 'Expired') {
1.103     matthew  1370:             print $CSVfile '"'.&mt('Students with expired roles').'"'."\n";
1.58      matthew  1371:         }
                   1372:         if ($statusmode eq 'Any') {
1.103     matthew  1373:             print $CSVfile '"'.join('","',map {
                   1374: 		&Apache::loncommon::csv_translate(&mt($_))
                   1375:                 } ("username","domain","ID","student name",
1.110     matthew  1376:                    "section","start date","end date","status")).'"'."\n";
1.58      matthew  1377:         } else {
1.103     matthew  1378:             print $CSVfile '"'.join('","',map {
                   1379: 		&Apache::loncommon::csv_translate(&mt($_))
                   1380:                 } ("username","domain","ID","student name",
1.110     matthew  1381:                    "section","start date","end date")).'"'."\n";
1.58      matthew  1382:         }
1.60      matthew  1383:     } elsif ($mode eq 'excel') {
                   1384:         # Create the excel spreadsheet
1.126     matthew  1385:         ($excel_workbook,$excel_filename,$format) = 
                   1386:             &Apache::loncommon::create_workbook($r);
                   1387:         return if (! defined($excel_workbook));
1.60      matthew  1388:         $excel_sheet = $excel_workbook->addworksheet('classlist');
                   1389:         #
1.76      albertel 1390:         my $description = 'Class List for '.
1.127     albertel 1391:             $env{'course.'.$env{'request.course.id'}.'.description'};
1.104     matthew  1392:         $excel_sheet->write($row++,0,$description,$format->{'h1'});
1.60      matthew  1393:         #
                   1394:         $excel_sheet->write($row++,0,["username","domain","ID",
1.110     matthew  1395:                                       "student name","section",
                   1396:                                       "start date","end date","status"],
                   1397:                             $format->{'bold'});
1.41      matthew  1398:     }
1.56      matthew  1399:     #
                   1400:     # Sort the students
                   1401:     my %index;
                   1402:     my $i;
                   1403:     foreach (@$keylist) {
                   1404:         $index{$_} = $i++;
                   1405:     }
                   1406:     my $index  = $index{$sortby};
                   1407:     my $second = $index{'username'};
                   1408:     my $third  = $index{'domain'};
1.53      matthew  1409:     my @Sorted_Students = sort {
1.56      matthew  1410:         lc($classlist->{$a}->[$index])  cmp lc($classlist->{$b}->[$index])
                   1411:             ||
                   1412:         lc($classlist->{$a}->[$second]) cmp lc($classlist->{$b}->[$second])
                   1413:             ||
                   1414:         lc($classlist->{$a}->[$third]) cmp lc($classlist->{$b}->[$third])
                   1415:         } (keys(%$classlist));
1.108     matthew  1416:     my $studentcount = 0;
1.115     raeburn  1417:     my $autocount = 0;
                   1418:     my $manualcount = 0;
                   1419:     my $unlockcount = 0;
                   1420:     my $lockcount = 0;
1.53      matthew  1421:     foreach my $student (@Sorted_Students) {
1.110     matthew  1422:         my $sdata = $classlist->{$student};
                   1423:         my $username = $sdata->[$index{'username'}];
                   1424:         my $domain   = $sdata->[$index{'domain'}];
                   1425:         my $section  = $sdata->[$index{'section'}];
                   1426:         my $name     = $sdata->[$index{'fullname'}];
                   1427:         my $id       = $sdata->[$index{'id'}];
                   1428:         my $status   = $sdata->[$index{'status'}];
                   1429:         my $start    = $sdata->[$index{'start'}];
                   1430:         my $end      = $sdata->[$index{'end'}];
1.115     raeburn  1431:         my $type     = $sdata->[$index{'type'}];
1.57      matthew  1432:         next if (($statusmode ne 'Any') && ($status ne $statusmode));
1.114     raeburn  1433:         if ($mode eq 'view' || $mode eq 'html' || $mode eq 'autoenroll') {
                   1434:             if (! defined($start) || $start == 0) {
                   1435:                 $start = &mt('none');
                   1436:             } else {
                   1437:                 $start = &Apache::lonlocal::locallocaltime($start);
                   1438:             }
                   1439:             if (! defined($end) || $end == 0) {
                   1440:                 $end = &mt('none');
                   1441:             } else {
                   1442:                 $end = &Apache::lonlocal::locallocaltime($end);
                   1443:             }
1.139     albertel 1444: 	    $r->print(&Apache::loncommon::start_data_table_row());
1.115     raeburn  1445:             if ($mode eq 'autoenroll') {
                   1446:                 my $lockedtype = $sdata->[$index{'lockedtype'}];
                   1447:                 $studentcount++;
                   1448:                 my $cellentry;
                   1449:                 if ($type eq 'auto') {
1.131     albertel 1450:                     $cellentry = '<b>'.&mt('auto').'</b>&nbsp;<label><input type="checkbox" name="chgauto" value="'.$username.':'.$domain.'" />&nbsp;Change</label>';
1.115     raeburn  1451:                     $autocount ++;
                   1452:                 } else {
1.131     albertel 1453:                     $cellentry = '<table border="0" cellspacing="0"><tr><td rowspan="2"><b>'.&mt('manual').'</b></td><td><nobr><label><input type="checkbox" name="chgmanual" value="'.$username.':'.$domain.'" />&nbsp;Change</label></nobr></td></tr><tr><td><nobr>';
1.115     raeburn  1454:                     $manualcount ++;
                   1455:                     if ($lockedtype) {
1.131     albertel 1456:                         $cellentry .= '<label><input type="checkbox" name="unlockchg" value="'.$username.':'.$domain.'" />&nbsp;'.&mt('Unlock').'</label>';
1.115     raeburn  1457:                         $unlockcount ++;
                   1458:                     } else {
1.131     albertel 1459:                         $cellentry .= '<label><input type="checkbox" name="lockchg" value="'.$username.':'.$domain.'" />&nbsp;'.&mt('Lock').'</label>';
1.115     raeburn  1460:                         $lockcount ++;
                   1461:                     }
1.118     raeburn  1462:                     $cellentry .= '</nobr></td></tr></table>';
1.115     raeburn  1463:                 }
                   1464:                 $r->print("<td>$cellentry<td>\n    ");
                   1465:             } else {
                   1466:                 $r->print("<td>".(++$studentcount)."</td><td>\n    ");
                   1467:             }
1.51      matthew  1468:             if ($linkto eq 'nothing') {
                   1469:                 $r->print($username);
                   1470:             } elsif ($linkto eq 'aboutme') {
                   1471:                 $r->print(&Apache::loncommon::aboutmewrapper($username,
                   1472:                                                              $username,
                   1473:                                                              $domain));
                   1474:             } elsif ($linkto eq 'modify') {
1.59      matthew  1475:                 $r->print('<a href="'.
                   1476:                           "javascript:document.studentform.sname.value='".
                   1477:                           $username.
                   1478:                           "';document.studentform.sdom.value='".$domain.
                   1479:                           "';document.studentform.state.value='selected".
                   1480:                           "';document.studentform.submit();".'">'.
1.53      matthew  1481:                           $username."</a>\n");
1.50      matthew  1482:             }
1.51      matthew  1483:             $r->print(<<"END");
1.50      matthew  1484:     </td>
1.51      matthew  1485:     <td>$domain</td>
                   1486:     <td>$id</td>
                   1487:     <td>$name</td>
                   1488:     <td>$section</td>
1.110     matthew  1489:     <td>$start</td>
                   1490:     <td>$end</td>
1.114     raeburn  1491: END
1.134     raeburn  1492:             if ($env{'course.'.$env{'request.course.id'}.
1.136     raeburn  1493: 			 '.internal.showphoto'}) {
1.134     raeburn  1494:                 if ($displayphotos eq 'on') {
1.135     albertel 1495:                     my $imgurl = 
                   1496: 			&Apache::lonnet::retrievestudentphoto($domain,
                   1497: 							      $username,'gif',
                   1498: 							      'thumbnail');
1.134     raeburn  1499:                 
                   1500:                     $r->print('    <td align="right"><a href="javascript:photowindow('."'".&Apache::lonnet::studentphoto($domain,$username,'jpg')."'".')"><img src="'.$imgurl.'" border="1"></a></td>');
                   1501:                 } else {
                   1502:                     $r->print('    <td>&nbsp;</td>  ');
                   1503:                 }
                   1504:             }
1.139     albertel 1505: 	    $r->print(&Apache::loncommon::end_data_table_row());
1.51      matthew  1506:         } elsif ($mode eq 'csv') {
1.103     matthew  1507:             next if (! defined($CSVfile));
1.51      matthew  1508:             # no need to bother with $linkto
1.114     raeburn  1509:             if (! defined($start) || $start == 0) {
                   1510:                 $start = &mt('none');
                   1511:             } else {
                   1512:                 $start = &Apache::lonlocal::locallocaltime($start);
                   1513:             }
                   1514:             if (! defined($end) || $end == 0) {
                   1515:                 $end = &mt('none');
                   1516:             } else {
                   1517:                 $end = &Apache::lonlocal::locallocaltime($end);
                   1518:             }
1.51      matthew  1519:             my @line = ();
1.110     matthew  1520:             foreach ($username,$domain,$id,$name,$section,$start,$end) {
1.51      matthew  1521:                 push @line,&Apache::loncommon::csv_translate($_);
1.58      matthew  1522:             }
                   1523:             if ($statusmode eq 'Any') {
                   1524:                 push @line,&Apache::loncommon::csv_translate($status);
1.41      matthew  1525:             }
1.103     matthew  1526:             print $CSVfile '"'.join('","',@line).'"'."\n";
1.60      matthew  1527:         } elsif ($mode eq 'excel') {
1.110     matthew  1528:             $excel_sheet->write($row,0,[$username,$domain,$id,
                   1529:                                           $name,$section]);
                   1530:             my $col = 5;
                   1531:             foreach my $time ($start,$end) {
1.129     matthew  1532:                 if (defined($time) && $time != 0) {
                   1533:                     $excel_sheet->write($row,$col++,
1.110     matthew  1534:                                    &Apache::lonstathelpers::calc_serial($time),
                   1535:                                     $format->{'date'});
1.129     matthew  1536:                 } else {
                   1537:                     $excel_sheet->write($row,$col++,'none');
                   1538:                 }                    
1.110     matthew  1539:             }
                   1540:             $excel_sheet->write($row,$col++,$status);
                   1541:             $row++;
1.40      matthew  1542:         }
                   1543:     }
1.114     raeburn  1544:     if ($mode eq 'view' || $mode eq 'html' || $mode eq 'autoenroll') {
1.139     albertel 1545: 	$r->print(&Apache::loncommon::end_data_table().'<br />');
1.60      matthew  1546:     } elsif ($mode eq 'excel') {
                   1547:         $excel_workbook->close();
                   1548:         $r->print('<p><a href="'.$excel_filename.'">'.
1.94      sakharuk 1549:                   &mt('Your Excel spreadsheet').'</a> '.&mt('is ready for download').'.</p>'."\n");
1.103     matthew  1550:     } elsif ($mode eq 'csv') {
                   1551:         close($CSVfile);
                   1552:         $r->print('<a href="'.$CSVfilename.'">'.
                   1553:                   &mt('Your CSV file').'</a> is ready for download.'.
                   1554:                   "\n");
                   1555:         $r->rflush();
1.60      matthew  1556:     }
1.114     raeburn  1557:     if ($mode eq 'autoenroll') {
1.115     raeburn  1558:         return ($studentcount,$autocount,$manualcount,$lockcount,$unlockcount);
1.114     raeburn  1559:     }
1.115     raeburn  1560:     return;
1.40      matthew  1561: }
                   1562: 
1.50      matthew  1563: 
                   1564: #
                   1565: # print out form for modification of a single students data
                   1566: #
                   1567: sub print_modify_student_form {
                   1568:     my $r = shift();
                   1569:     &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1.59      matthew  1570:                                             ['sdom','sname']);    
1.127     albertel 1571:     my $sname  = $env{'form.sname'};
                   1572:     my $sdom   = $env{'form.sdom'};
                   1573:     my $sortby = $env{'form.sortby'};
1.50      matthew  1574:     # determine the students name information
                   1575:     my %info=&Apache::lonnet::get('environment',
                   1576:                                   ['firstname','middlename',
1.52      matthew  1577:                                    'lastname','generation','id'],
1.50      matthew  1578:                                   $sdom, $sname);
                   1579:     my ($tmp) = keys(%info);
                   1580:     if ($tmp =~ /^(con_lost|error|no_such_host)/i) {
1.94      sakharuk 1581:         $r->print('<font color="#ff0000" size="+2">'.&mt('Error').'</font>'.
1.50      matthew  1582:                   '<p>'.
1.94      sakharuk 1583:                   &mt('Unable to retrieve environment data for').' '.$sname.
                   1584:                   &mt('in domain').' '.$sdom.'</p><p>'.
1.138     albertel 1585:                   &mt('Please contact your LON-CAPA administrator regarding this situation.').'</p>'.&Apache::loncommon::end_page());
1.50      matthew  1586:         return;
                   1587:     }
                   1588:     # determine the students starting and ending times and section
                   1589:     my ($starttime,$endtime,$section) = &get_enrollment_data($sname,$sdom);
1.87      matthew  1590:     if ($starttime =~ /^error/) {
1.94      sakharuk 1591:         $r->print('<h2>'&mt('Error').'</h2>');
1.87      matthew  1592:         $r->print('<p>'.$starttime.'</p>');
                   1593:         return;
                   1594:     }
1.101     matthew  1595:     #
1.50      matthew  1596:     # Deal with date forms
1.101     matthew  1597:     my $current_date_description = '';
                   1598:     my $textdate = '';
                   1599: 
                   1600:     if (! defined($starttime) || $starttime == 0) {
                   1601:         $current_date_description = &mt('Current Starting Date: not set').
                   1602:             '<br />';
                   1603:     } else {
                   1604:         $current_date_description = 
                   1605:             &mt('Current Starting Date: [_1]',
                   1606:                 &Apache::lonlocal::locallocaltime($starttime)).'<br />';
                   1607:     }
                   1608:     if (! defined($endtime) || $endtime == 0) {
                   1609:         $current_date_description.= &mt('Current Ending Date: not set').
                   1610:             '<br />';
                   1611:     } else {
                   1612:         $current_date_description.= 
                   1613:             &mt('Current Ending Date: [_1]',
                   1614:                 &Apache::lonlocal::locallocaltime($endtime)).'<br />';
                   1615: 
                   1616:     }
1.68      matthew  1617:     my $date_table = &date_setting_table($starttime,$endtime);
1.59      matthew  1618:     #
1.127     albertel 1619:     if (! exists($env{'form.Status'}) || 
                   1620:         $env{'form.Status'} !~ /^(Any|Expired|Active)$/) {
                   1621:         $env{'form.Status'} = 'crap';
1.59      matthew  1622:     }
1.94      sakharuk 1623:     # Make sure student is enrolled in course
                   1624:     my %lt=&Apache::lonlocal::texthash(
                   1625: 	           'mef'   => "Modify Enrollment for",
                   1626:                    'odcc'  => "Only domain coordinators can change a users password.",
                   1627:                    'sn'    => "Student Name",
                   1628:                    'fn'    => "First",
                   1629:                    'mn'    => "Middle",
                   1630:                    'ln'    => "Last",
                   1631:                    'gen'   => "Generation",
                   1632:                    'sid'   => "Student ID",
                   1633:                    'disn'  => "Disable ID/Student Number Safeguard and Force Change of Conflicting IDs (only do if you know what you are doing)",
                   1634:                    'sec'   => "Section",
                   1635:                    'sm'    => "Submit Modifications",
                   1636: 				       );
1.50      matthew  1637:     $r->print(<<END);
1.52      matthew  1638: <p>
                   1639: <font size="+1">
1.94      sakharuk 1640: $lt{'odcc'}
1.52      matthew  1641: </font>
                   1642: </p>
1.50      matthew  1643: <input type="hidden" name="slogin"  value="$sname"  />
                   1644: <input type="hidden" name="sdomain" value="$sdom" />
                   1645: <input type="hidden" name="action"  value="modifystudent" />
1.53      matthew  1646: <input type="hidden" name="state"   value="done" />
                   1647: <input type="hidden" name="sortby"  value="$sortby" />
1.127     albertel 1648: <input type="hidden" name="Status"  value="$env{'form.Status'}" />
1.94      sakharuk 1649: <h2>$lt{'mef'} $info{'firstname'} $info{'middlename'} 
1.50      matthew  1650: $info{'lastname'} $info{'generation'}, $sname\@$sdom</h2>
                   1651: <p>
1.94      sakharuk 1652: <b>$lt{'sn'}</b>
1.50      matthew  1653: <table>
1.94      sakharuk 1654: <tr><th>$lt{'fn'}</th><th>$lt{'mn'}</th><th>$lt{'ln'}</th><th>$lt{'gen'}</th></tr>
1.50      matthew  1655: <tr><td>
                   1656: <input type="text" name="firstname"  value="$info{'firstname'}"  /></td><td>
                   1657: <input type="text" name="middlename" value="$info{'middlename'}" /></td><td>
                   1658: <input type="text" name="lastname"   value="$info{'lastname'}"   /></td><td>
                   1659: <input type="text" name="generation" value="$info{'generation'}" /></td></tr>
                   1660: </table>
                   1661: </p><p>
1.94      sakharuk 1662: <b>$lt{'sid'}</b>: <input type="text" name="id" value="$info{'id'}" size="12"/>
1.52      matthew  1663: </p><p>
1.131     albertel 1664: <label>
1.53      matthew  1665: <input type="checkbox" name="forceid" > 
1.94      sakharuk 1666: $lt{'disn'}
1.131     albertel 1667: </label>
1.53      matthew  1668: </p><p>
1.101     matthew  1669: <b>$lt{'sec'}</b>: <input type="text" name="section" value="$section" size="14"/>
1.50      matthew  1670: </p>
1.101     matthew  1671: <p>$current_date_description</p>
1.68      matthew  1672: <p>$date_table</p>
1.94      sakharuk 1673: <input type="submit" value="$lt{'sm'}" />
1.50      matthew  1674: END
1.138     albertel 1675:     $r->print(&Apache::loncommon::end_page());
1.50      matthew  1676:     return;
                   1677: }
                   1678: 
                   1679: #
                   1680: # modify a single students section 
                   1681: #
                   1682: sub modify_single_student {
1.138     albertel 1683:     my ($r) = @_;
1.68      matthew  1684:     #
1.80      matthew  1685:     # Remove non alphanumeric values from the section
1.127     albertel 1686:     $env{'form.section'} =~ s/\W//g;
1.77      matthew  1687:     #
1.68      matthew  1688:     # Do the date defaults first
                   1689:     my ($starttime,$endtime) = &get_dates_from_form();
1.127     albertel 1690:     if ($env{'form.makedatesdefault'}) {
1.68      matthew  1691:         $r->print(&make_dates_default($starttime,$endtime));
                   1692:     }
1.59      matthew  1693:     # Get the 'sortby' and 'Status' variables so the user goes back to their
                   1694:     # previous screen
1.127     albertel 1695:     my $sortby = $env{'form.sortby'};
                   1696:     my $status = $env{'form.Status'};
1.53      matthew  1697:     #
                   1698:     # We always need this information
1.127     albertel 1699:     my $slogin     = $env{'form.slogin'};
                   1700:     my $sdom       = $env{'form.sdomain'};
1.53      matthew  1701:     #
                   1702:     # Get the old data
                   1703:     my %old=&Apache::lonnet::get('environment',
                   1704:                                  ['firstname','middlename',
                   1705:                                   'lastname','generation','id'],
                   1706:                                  $sdom, $slogin);
1.59      matthew  1707:     $old{'section'} = &Apache::lonnet::getsection($sdom,$slogin,
1.127     albertel 1708:                                                   $env{'request.course.id'});
1.53      matthew  1709:     my ($tmp) = keys(%old);
                   1710:     if ($tmp =~ /^(con_lost|error|no_such_host)/i) {
1.94      sakharuk 1711:         $r->print(&mt('There was an error determining the environment values for')." $slogin \@ $sdom.");
1.53      matthew  1712:         return;
                   1713:     }
                   1714:     undef $tmp;
                   1715:     #
                   1716:     # Get the new data
1.127     albertel 1717:     my $firstname  = $env{'form.firstname'};
                   1718:     my $middlename = $env{'form.middlename'};
                   1719:     my $lastname   = $env{'form.lastname'};
                   1720:     my $generation = $env{'form.generation'};
                   1721:     my $section    = $env{'form.section'};
                   1722:     my $courseid   = $env{'request.course.id'};
                   1723:     my $sid        = $env{'form.id'};
1.50      matthew  1724:     my $displayable_starttime = localtime($starttime);
                   1725:     my $displayable_endtime   = localtime($endtime);
1.53      matthew  1726:     # 
                   1727:     # check for forceid override
1.63      matthew  1728:     if ((defined($old{'id'})) && ($old{'id'} ne '') && 
1.127     albertel 1729:         ($sid ne $old{'id'}) && (! exists($env{'form.forceid'}))) {
1.94      sakharuk 1730:         $r->print("<font color=\"ff0000\">".&mt('You changed the students id but did not disable the ID change safeguard. The students id will not be changed.')."</font>");
1.53      matthew  1731:         $sid = $old{'id'};
                   1732:     }
                   1733:     #
1.50      matthew  1734:     # talk to the user about what we are going to do
1.94      sakharuk 1735:     my %lt=&Apache::lonlocal::texthash(
                   1736: 	           'mdu'   => "Modifying data for user",
                   1737:                    'si'    => "Student Information",
                   1738:                    'fd'    => "Field",
                   1739:                    'ov'    => "Old Value",
                   1740:                    'nv'    => "New Value",
                   1741:                    'fn'    => "First name",
                   1742:                    'mn'    => "Middle name",
                   1743:                    'ln'    => "Last name",
                   1744:                    'gen'   => "Generation",
                   1745:                    'sec'   => "Section",
                   1746:                    'ri'    => "Role Information",
                   1747:                    'st'    => "Start Time",
                   1748:                    'et'    => "End Time",
                   1749: 				       );
1.50      matthew  1750:     $r->print(<<END);
1.94      sakharuk 1751:     <h2>$lt{'mdu'} $slogin \@ $sdom </h2>
                   1752: <h3>$lt{'si'}</h3>
1.53      matthew  1753: <table rules="rows" border="1" cellpadding="3" >
                   1754: <tr>
1.94      sakharuk 1755:     <th> $lt{'fd'} </th>
                   1756:     <th> $lt{'ov'} </th>
                   1757:     <th> $lt{'nv'} </th>
1.53      matthew  1758: </tr>
                   1759: <tr>
1.94      sakharuk 1760:     <td> <b>$lt{'fn'}</b> </td>
1.53      matthew  1761:     <td> $old{'firstname'} </td>
                   1762:     <td> $firstname </td>
                   1763: </tr><tr>
1.94      sakharuk 1764:     <td> <b>$lt{'mn'}</b> </td>
1.53      matthew  1765:     <td> $old{'middlename'} </td>
                   1766:     <td> $middlename </td>
                   1767: </tr><tr>
1.94      sakharuk 1768:     <td> <b>$lt{'ln'}</b> </td>
1.53      matthew  1769:     <td> $old{'lastname'} </td>
                   1770:     <td> $lastname </td>
                   1771: </tr><tr>
1.94      sakharuk 1772:     <td> <b>$lt{'gen'}</b> </td>
1.53      matthew  1773:     <td> $old{'generation'} </td>
                   1774:     <td> $generation </td>
                   1775: </tr><tr>
                   1776:     <td> <b>ID</b> </td>
                   1777:     <td> $old{'id'} </td>
                   1778:     <td> $sid </td>
1.59      matthew  1779: </tr><tr>
1.94      sakharuk 1780:     <td> <b>$lt{'sec'}</b> </td>
1.59      matthew  1781:     <td> $old{'section'} </td>
                   1782:     <td> $section</td>
1.53      matthew  1783: </tr>
1.50      matthew  1784: </table>
1.94      sakharuk 1785: <h3>$lt{'ri'}</h3>
1.50      matthew  1786: <table>
1.94      sakharuk 1787: <tr><td align="right"><b>$lt{'st'}:</b></td><td> $displayable_starttime </td></tr>
                   1788: <tr><td align="right"><b>$lt{'et'}:</b></td><td> $displayable_endtime   </td></tr>
1.50      matthew  1789: </table>
1.52      matthew  1790: <p>
1.50      matthew  1791: END
1.53      matthew  1792:     #
1.63      matthew  1793:     # Send request(s) to modify data (final undef is for 'desiredhost',
                   1794:     # which is a moot point because the student already has an account.
                   1795:     my $modify_section_results = &modifystudent($sdom,$slogin,
1.127     albertel 1796:                                                 $env{'request.course.id'},
1.63      matthew  1797:                                                 $section,undef);
                   1798:     if ($modify_section_results !~ /^ok/) {
1.94      sakharuk 1799:         $r->print(&mt('An error occured during the attempt to change the section for this student.')."<br />");
1.63      matthew  1800:     }
1.52      matthew  1801:     my $roleresults = &Apache::lonnet::modifystudent
1.53      matthew  1802:         ($sdom,$slogin,$sid,undef,undef,$firstname,$middlename,$lastname,
1.127     albertel 1803:          $generation,$section,$endtime,$starttime,$env{'form.forceid'});
1.53      matthew  1804:     if ($roleresults eq 'refused' ) {
1.94      sakharuk 1805:         $r->print(&mt('Your request to change the role information for this student was refused. You do not appear to have sufficient authority to change student information.'));
1.50      matthew  1806:     } elsif ($roleresults !~ /ok/) {
1.94      sakharuk 1807:         $r->print(&mt('An error occurred during the attempt to change the role information for this student.')."  <br />".
                   1808:                   &mt('The error reported was')." ".
1.50      matthew  1809:                   $roleresults);
1.53      matthew  1810:         &Apache::lonnet::logthis("londropadd:failed attempt to modify student".
                   1811:                                  " data for ".$slogin." \@ ".$sdom." by ".
1.127     albertel 1812:                                  $env{'user.name'}." \@ ".$env{'user.domain'}.
1.53      matthew  1813:                                  ":".$roleresults);
1.50      matthew  1814:     } else { # everything is okay!
1.94      sakharuk 1815:         $r->print(&mt('Student information updated successfully.')." <br />".
                   1816:                   &mt('The student must log out and log in again to see these changes.'));
1.50      matthew  1817:     }
1.94      sakharuk 1818:     my $Masd=&mt('Modify another students data');
1.50      matthew  1819:     $r->print(<<END);
1.52      matthew  1820: </p><p>
1.59      matthew  1821: <input type="hidden" name="action" value="modifystudent" />
                   1822: <input type="hidden" name="sortby" value="$sortby" />
                   1823: <input type="hidden" name="Status" value="$status" />
1.94      sakharuk 1824: <a href="javascript:document.studentform.submit();">$Masd</a>
1.50      matthew  1825: END
1.138     albertel 1826:     $r->print(&Apache::loncommon::end_page());
1.50      matthew  1827:     return;
                   1828: }
                   1829: 
                   1830: sub get_enrollment_data {
                   1831:     my ($sname,$sdomain) = @_;
1.127     albertel 1832:     my $courseid = $env{'request.course.id'};
1.50      matthew  1833:     $courseid =~ s:_:/:g;
                   1834:     my %roles = &Apache::lonnet::dump('roles',$sdomain,$sname);
                   1835:     my ($tmp) = keys(%roles);
                   1836:     # Bail out if we were unable to get the students roles
1.87      matthew  1837:     return ('error'.$tmp) if ($tmp =~ /^(con_lost|error|no_such_host)/i);
1.50      matthew  1838:     # Go through the roles looking for enrollment in this course
                   1839:     my ($end,$start) = (undef,undef);
                   1840:     my $section = '';
                   1841:     my $count = scalar(keys(%roles));
                   1842:     while (my ($course,$role) = each(%roles)) {
                   1843:         if ($course=~ /^\/$courseid\/*\s*(\w+)*_st$/ ) {
                   1844:             #
                   1845:             # Get active role
                   1846:             $section=$1;
                   1847:             (undef,$end,$start)=split(/\_/,$role);
                   1848:             my $now=time;
                   1849:             my $notactive=0;
                   1850:             if ($start) {
                   1851:                 if ($now<$start) { $notactive=1; }
                   1852:             }
                   1853:             if ($end) {
                   1854:                 if ($now>$end) { $notactive=1; }
                   1855:             } 
                   1856:             unless ($notactive) { return ($start,$end,$section); }
                   1857:         }
                   1858:     }
                   1859:     return ($start,$end,$section);
                   1860: }
                   1861: 
1.56      matthew  1862: #################################################
                   1863: #################################################
                   1864: 
                   1865: =pod
                   1866: 
                   1867: =item show_drop_list
                   1868: 
                   1869: Display a list of students to drop
                   1870: Inputs: 
                   1871: 
                   1872: =over 4
                   1873: 
                   1874: =item $r, Apache request
                   1875: 
                   1876: =item $classlist, hash pointer returned from loncoursedata::get_classlist();
                   1877: 
                   1878: =item $keylist, array pointer returned from loncoursedata::get_classlist() 
                   1879: which describes the order elements are stored in the %$classlist values.
                   1880: 
                   1881: =item $nosort, if true, sorting links are omitted.
                   1882: 
                   1883: =back
                   1884: 
                   1885: =cut
                   1886: 
                   1887: #################################################
                   1888: #################################################
1.11      www      1889: sub show_drop_list {
1.56      matthew  1890:     my ($r,$classlist,$keylist,$nosort)=@_;
1.127     albertel 1891:     my $cid=$env{'request.course.id'};
                   1892:     if (! exists($env{'form.sortby'})) {
1.59      matthew  1893:         &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
                   1894:                                                 ['sortby']);
                   1895:     }
1.127     albertel 1896:     my $sortby = $env{'form.sortby'};
1.110     matthew  1897:     if ($sortby !~ /^(username|domain|section|fullname|id|start|end)$/) {
1.54      matthew  1898:         $sortby = 'username';
                   1899:     }
1.56      matthew  1900:     #
1.54      matthew  1901:     my $action = "drop";
                   1902:     $r->print(<<END);
                   1903: <input type="hidden" name="sortby" value="$sortby" />
                   1904: <input type="hidden" name="action" value="$action" />
1.50      matthew  1905: <input type="hidden" name="state"  value="done" />
1.32      matthew  1906: <script>
1.51      matthew  1907: function checkAll(field) {
1.32      matthew  1908:     for (i = 0; i < field.length; i++)
                   1909:         field[i].checked = true ;
                   1910: }
                   1911: 
1.51      matthew  1912: function uncheckAll(field) {
1.32      matthew  1913:     for (i = 0; i < field.length; i++)
                   1914:         field[i].checked = false ;
                   1915: }
                   1916: </script>
                   1917: <p>
1.26      matthew  1918: <input type="hidden" name="phase" value="four">
1.56      matthew  1919: END
                   1920: 
1.110     matthew  1921: my %lt=&Apache::lonlocal::texthash('usrn'   => "username",
                   1922:                                    'dom'    => "domain",
                   1923:                                    'sn'     => "student name",
                   1924:                                    'sec'    => "section",
                   1925:                                    'start'  => "start date",
                   1926:                                    'end'    => "end date",
                   1927:                                    );
1.56      matthew  1928:     if ($nosort) {
1.139     albertel 1929: 	$r->print(&Apache::loncommon::start_data_table());
1.56      matthew  1930:         $r->print(<<END);
                   1931: <tr>
                   1932:     <th>&nbsp;</th>
1.94      sakharuk 1933:     <th>$lt{'usrn'}</th>
                   1934:     <th>$lt{'dom'}</th>
1.56      matthew  1935:     <th>ID</th>
1.94      sakharuk 1936:     <th>$lt{'sn'}</th>
                   1937:     <th>$lt{'sec'}</th>
1.110     matthew  1938:     <th>$lt{'start'}</th>
                   1939:     <th>$lt{'end'}</th>
1.56      matthew  1940: </tr>
                   1941: END
                   1942: 
                   1943:     } else  {
1.139     albertel 1944: 	$r->print(&Apache::loncommon::start_data_table());
1.56      matthew  1945:         $r->print(<<END);
1.54      matthew  1946: <tr><th>&nbsp;</th>
                   1947:     <th>
1.94      sakharuk 1948:        <a href="/adm/dropadd?action=$action&sortby=username">$lt{'usrn'}</a>
1.54      matthew  1949:     </th><th>
1.94      sakharuk 1950:        <a href="/adm/dropadd?action=$action&sortby=domain">$lt{'dom'}</a>
1.54      matthew  1951:     </th><th>
                   1952:        <a href="/adm/dropadd?action=$action&sortby=id">ID</a>
                   1953:     </th><th>
1.94      sakharuk 1954:        <a href="/adm/dropadd?action=$action&sortby=fullname">$lt{'sn'}</a>
1.54      matthew  1955:     </th><th>
1.94      sakharuk 1956:        <a href="/adm/dropadd?action=$action&sortby=section">$lt{'sec'}</a>
1.110     matthew  1957:     </th><th>
                   1958:        <a href="/adm/dropadd?action=$action&sortby=start">$lt{'start'}</a>
                   1959:     </th><th>
                   1960:        <a href="/adm/dropadd?action=$action&sortby=end">$lt{'end'}</a>
1.54      matthew  1961:     </th>
                   1962: </tr>
1.26      matthew  1963: END
1.56      matthew  1964:     }
                   1965:     #
                   1966:     # Sort the students
                   1967:     my %index;
                   1968:     my $i;
                   1969:     foreach (@$keylist) {
                   1970:         $index{$_} = $i++;
                   1971:     }
                   1972:     my $index  = $index{$sortby};
                   1973:     my $second = $index{'username'};
                   1974:     my $third  = $index{'domain'};
1.54      matthew  1975:     my @Sorted_Students = sort {
1.56      matthew  1976:         lc($classlist->{$a}->[$index])  cmp lc($classlist->{$b}->[$index])
                   1977:             ||
                   1978:         lc($classlist->{$a}->[$second]) cmp lc($classlist->{$b}->[$second])
                   1979:             ||
                   1980:         lc($classlist->{$a}->[$third]) cmp lc($classlist->{$b}->[$third])
                   1981:         } (keys(%$classlist));
1.54      matthew  1982:     foreach my $student (@Sorted_Students) {
1.52      matthew  1983:         my $error;
1.110     matthew  1984:         my $sdata = $classlist->{$student};
                   1985:         my $username = $sdata->[$index{'username'}];
                   1986:         my $domain   = $sdata->[$index{'domain'}];
                   1987:         my $section  = $sdata->[$index{'section'}];
                   1988:         my $name     = $sdata->[$index{'fullname'}];
                   1989:         my $id       = $sdata->[$index{'id'}];
                   1990:         my $start    = $sdata->[$index{'start'}];
                   1991:         my $end      = $sdata->[$index{'end'}];
                   1992:         if (! defined($start) || $start == 0) {
                   1993:             $start = &mt('none');
                   1994:         } else {
                   1995:             $start = &Apache::lonlocal::locallocaltime($start);
                   1996:         }
                   1997:         if (! defined($end) || $end == 0) {
                   1998:             $end = &mt('none');
                   1999:         } else {
                   2000:             $end = &Apache::lonlocal::locallocaltime($end);
                   2001:         }
                   2002:         my $status   = $sdata->[$index{'status'}];
1.51      matthew  2003:         next if ($status ne 'Active');
                   2004:         #
1.139     albertel 2005:         $r->print(&Apache::loncommon::start_data_table_row());
1.51      matthew  2006:         $r->print(<<"END");
                   2007:     <td><input type="checkbox" name="droplist" value="$student"></td>
                   2008:     <td>$username</td>
                   2009:     <td>$domain</td>
                   2010:     <td>$id</td>
                   2011:     <td>$name</td>
                   2012:     <td>$section</td>
1.110     matthew  2013:     <td>$start</td>
                   2014:     <td>$end</td>
1.26      matthew  2015: END
1.139     albertel 2016:         $r->print(&Apache::loncommon::end_data_table_row());
1.25      matthew  2017:     }
1.139     albertel 2018:     $r->print(&Apache::loncommon::end_data_table().'<br />');
1.111     matthew  2019:     %lt=&Apache::lonlocal::texthash(
1.94      sakharuk 2020: 	               'dp'   => "Drop Students",
                   2021:                        'ca'   => "check all",
                   2022:                        'ua'   => "uncheck all",
                   2023: 				       );
1.32      matthew  2024:     $r->print(<<"END");
                   2025: </p><p>
1.94      sakharuk 2026: <input type="button" value="$lt{'ca'}" onclick="javascript:checkAll(document.studentform.droplist)"> &nbsp;
                   2027: <input type="button" value="$lt{'ua'}" onclick="javascript:uncheckAll(document.studentform.droplist)"> 
                   2028: <p><input type=submit value="$lt{'dp'}"></p>
1.32      matthew  2029: END
1.51      matthew  2030:     return;
1.10      www      2031: }
                   2032: 
1.48      matthew  2033: #
                   2034: # Print out the initial form to get the courselist file
                   2035: #
                   2036: sub print_first_courselist_upload_form {
                   2037:     my $r=shift;
1.88      matthew  2038:     my $str;
                   2039:     $str  = '<input type="hidden" name="phase" value="two">';
                   2040:     $str .= '<input type="hidden" name="action" value="upload" />';
                   2041:     $str .= '<input type="hidden"   name="state"  value="got_file" />';
                   2042:     $str .= "<h3>".&mt('Upload a class list')."</h3>\n";
                   2043:     $str .= &Apache::loncommon::upfile_select_html();
                   2044:     $str .= "<p>\n";
                   2045:     $str .= '<input type="submit" name="fileupload" value="'.
                   2046:         &mt('Upload class list').'">'."\n";
1.131     albertel 2047:     $str .= '<label><input type="checkbox" name="noFirstLine" /> '.
                   2048:         &mt('Ignore First Line')."</label></p>\n";
1.88      matthew  2049:     $str .= &Apache::loncommon::help_open_topic("Course_Create_Class_List",
1.92      sakharuk 2050:                          &mt("How do I create a class list from a spreadsheet")).
1.88      matthew  2051:                              "<br />\n";
                   2052:     $str .= &Apache::loncommon::help_open_topic("Course_Convert_To_CSV",
1.92      sakharuk 2053:                            &mt("How do I create a CSV file from a spreadsheet")).
1.88      matthew  2054:                                "<br />\n";
1.138     albertel 2055:     $str .= &Apache::loncommon::end_page();
1.88      matthew  2056:     $r->print($str);
1.48      matthew  2057:     return;
                   2058: }
                   2059: 
1.10      www      2060: # ================================================= Drop/Add from uploaded file
                   2061: sub upfile_drop_add {
                   2062:     my $r=shift;
1.24      albertel 2063:     &Apache::loncommon::load_tmp_file($r);
                   2064:     my @studentdata=&Apache::loncommon::upfile_record_sep();
1.127     albertel 2065:     if($env{'form.noFirstLine'}){shift(@studentdata);}
                   2066:     my @keyfields = split(/\,/,$env{'form.keyfields'});
                   2067:     my $cid = $env{'request.course.id'};
1.25      matthew  2068:     my %fields=();
1.127     albertel 2069:     for (my $i=0; $i<=$env{'form.nfields'}; $i++) {
                   2070:         if ($env{'form.upfile_associate'} eq 'reverse') {
                   2071:             if ($env{'form.f'.$i} ne 'none') {
                   2072:                 $fields{$keyfields[$i]}=$env{'form.f'.$i};
1.25      matthew  2073:             }
                   2074:         } else {
1.127     albertel 2075:             $fields{$env{'form.f'.$i}}=$keyfields[$i];
1.25      matthew  2076:         }
                   2077:     }
1.99      matthew  2078:     #
                   2079:     # Store the field choices away
                   2080:     foreach my $field (qw/username names 
                   2081:                        fname mname lname gen id sec ipwd email/) {
1.127     albertel 2082:         $env{'form.'.$field.'_choice'}=$fields{$field};
1.99      matthew  2083:     }
                   2084:     &Apache::loncommon::store_course_settings('enrollment_upload',
                   2085:                                               { 'username_choice' => 'scalar',
                   2086:                                                 'names_choice' => 'scalar',
                   2087:                                                 'fname_choice' => 'scalar',
                   2088:                                                 'mname_choice' => 'scalar',
                   2089:                                                 'lname_choice' => 'scalar',
                   2090:                                                 'gen_choice' => 'scalar',
                   2091:                                                 'id_choice' => 'scalar',
                   2092:                                                 'sec_choice' => 'scalar',
                   2093:                                                 'ipwd_choice' => 'scalar',
                   2094:                                                 'email_choice' => 'scalar' });
                   2095: 
1.26      matthew  2096:     #
1.68      matthew  2097:     my ($startdate,$enddate) = &get_dates_from_form();
1.127     albertel 2098:     if ($env{'form.makedatesdefault'}) {
1.68      matthew  2099:         $r->print(&make_dates_default($startdate,$enddate));
                   2100:     }
1.31      matthew  2101:     # Determine domain and desired host (home server)
1.127     albertel 2102:     my $domain=$env{'form.lcdomain'};
                   2103:     my $desiredhost = $env{'form.lcserver'};
1.31      matthew  2104:     if (lc($desiredhost) eq 'default') {
                   2105:         $desiredhost = undef;
                   2106:     } else {
1.45      matthew  2107:         my %home_servers = &Apache::loncommon::get_library_servers($domain);
1.31      matthew  2108:         if (! exists($home_servers{$desiredhost})) {
1.88      matthew  2109:             $r->print('<font color="#ff0000">'.&mt('Error').'</font>'.
                   2110:                       &mt('Invalid home server specified'));
1.138     albertel 2111:             $r->print(&Apache::loncommon::end_page());
1.31      matthew  2112:             return;
                   2113:         }
                   2114:     }
1.26      matthew  2115:     # Determine authentication mechanism
                   2116:     my $amode  = '';
                   2117:     my $genpwd = '';
1.127     albertel 2118:     if ($env{'form.login'} eq 'krb') {
1.47      albertel 2119:         $amode='krb';
1.127     albertel 2120: 	$amode.=$env{'form.krbver'};
                   2121:         $genpwd=$env{'form.krbarg'};
                   2122:     } elsif ($env{'form.login'} eq 'int') {
1.25      matthew  2123:         $amode='internal';
1.127     albertel 2124:         if ((defined($env{'form.intarg'})) && ($env{'form.intarg'})) {
                   2125:             $genpwd=$env{'form.intarg'};
1.25      matthew  2126:         }
1.127     albertel 2127:     } elsif ($env{'form.login'} eq 'loc') {
1.25      matthew  2128:         $amode='localauth';
1.127     albertel 2129:         if ((defined($env{'form.locarg'})) && ($env{'form.locarg'})) {
                   2130:             $genpwd=$env{'form.locarg'};
1.79      matthew  2131:         }
                   2132:     }
                   2133:     if ($amode =~ /^krb/) {
                   2134:         if (! defined($genpwd) || $genpwd eq '') {
                   2135:             $r->print('<font color="red" size="+1">'.
1.88      matthew  2136:                       &mt('Unable to enroll students').'</font>  '.
                   2137:                       &mt('No Kerberos domain was specified.').'</p>');
1.79      matthew  2138:             $amode = ''; # This causes the loop below to be skipped
1.25      matthew  2139:         }
                   2140:     }
                   2141:     unless (($domain=~/\W/) || ($amode eq '')) {
1.26      matthew  2142:         #######################################
                   2143:         ##         Enroll Students           ##
                   2144:         #######################################
1.88      matthew  2145:         $r->print('<h3>'.&mt('Enrolling Students')."</h3>\n<p>\n");
1.25      matthew  2146:         my $count=0;
                   2147:         my $flushc=0;
                   2148:         my %student=();
1.26      matthew  2149:         # Get new classlist
1.25      matthew  2150:         foreach (@studentdata) {
                   2151:             my %entries=&Apache::loncommon::record_sep($_);
1.26      matthew  2152:             # Determine student name
1.25      matthew  2153:             unless (($entries{$fields{'username'}} eq '') ||
                   2154:                     (!defined($entries{$fields{'username'}}))) {
1.26      matthew  2155:                 my ($fname, $mname, $lname,$gen) = ('','','','');
1.25      matthew  2156:                 if (defined($fields{'names'})) {
1.26      matthew  2157:                     ($lname,$fname,$mname)=($entries{$fields{'names'}}=~
                   2158:                                             /([^\,]+)\,\s*(\w+)\s*(.*)$/);
1.25      matthew  2159:                 } else {
                   2160:                     if (defined($fields{'fname'})) {
                   2161:                         $fname=$entries{$fields{'fname'}};
                   2162:                     }
                   2163:                     if (defined($fields{'mname'})) {
                   2164:                         $mname=$entries{$fields{'mname'}};
                   2165:                     }
                   2166:                     if (defined($fields{'lname'})) {
                   2167:                         $lname=$entries{$fields{'lname'}};
                   2168:                     }
                   2169:                     if (defined($fields{'gen'})) {
                   2170:                         $gen=$entries{$fields{'gen'}};
                   2171:                     }
                   2172:                 }
                   2173:                 if ($entries{$fields{'username'}}=~/\W/) {
1.88      matthew  2174:                     $r->print('<br />'.
                   2175:       &mt('<b>[_1]</b>: Unacceptable username for user [_2] [_3] [_4] [_5]',
                   2176:           $entries{$fields{'username'}},$fname,$mname,$lname,$gen).
                   2177:                               '</b>');
1.25      matthew  2178:                 } else {
1.26      matthew  2179:                     # determine section number
1.25      matthew  2180:                     my $sec='';
                   2181:                     my $username=$entries{$fields{'username'}};
                   2182:                     if (defined($fields{'sec'})) {
                   2183:                         if (defined($entries{$fields{'sec'}})) {
                   2184:                             $sec=$entries{$fields{'sec'}};
                   2185:                         }
                   2186:                     }
1.80      matthew  2187:                     # remove non alphanumeric values from section
                   2188:                     $sec =~ s/\W//g;
1.26      matthew  2189:                     # determine student id number
1.25      matthew  2190:                     my $id='';
                   2191:                     if (defined($fields{'id'})) {
                   2192:                         if (defined($entries{$fields{'id'}})) {
                   2193:                             $id=$entries{$fields{'id'}};
                   2194:                         }
                   2195:                         $id=~tr/A-Z/a-z/;
                   2196:                     }
1.73      www      2197:                     # determine email address
                   2198:                     my $email='';
                   2199:                     if (defined($fields{'email'})) {
                   2200:                         if (defined($entries{$fields{'email'}})) {
                   2201:                             $email=$entries{$fields{'email'}};
                   2202:                             unless ($email=~/^[^\@]+\@[^\@]+$/) { $email=''; }
                   2203:                         }
                   2204:                     }
1.26      matthew  2205:                     # determine student password
1.25      matthew  2206:                     my $password='';
                   2207:                     if ($genpwd) { 
                   2208:                         $password=$genpwd; 
                   2209:                     } else {
                   2210:                         if (defined($fields{'ipwd'})) {
                   2211:                             if ($entries{$fields{'ipwd'}}) {
                   2212:                                 $password=$entries{$fields{'ipwd'}};
                   2213:                             }
                   2214:                         }
                   2215:                     }
1.56      matthew  2216:                     # Clean up whitespace
                   2217:                     foreach (\$domain,\$username,\$id,\$fname,\$mname,
                   2218:                              \$lname,\$gen,\$sec) {
                   2219:                         $$_ =~ s/(\s+$|^\s+)//g;
                   2220:                     }
1.127     albertel 2221:                     if ($password || $env{'form.login'} eq 'loc') {
1.33      matthew  2222:                         &modifystudent($domain,$username,$cid,$sec,
                   2223:                                        $desiredhost);
1.25      matthew  2224:                         my $reply=&Apache::lonnet::modifystudent
                   2225:                             ($domain,$username,$id,$amode,$password,
                   2226:                              $fname,$mname,$lname,$gen,$sec,$enddate,
1.127     albertel 2227:                              $startdate,$env{'form.forceid'},$desiredhost,
1.73      www      2228:                              $email);
1.26      matthew  2229:                         if ($reply ne 'ok') {
1.72      matthew  2230:                             $reply =~ s/^error://;
1.88      matthew  2231:                             $r->print('<br />'.
                   2232:                 &mt('<b>[_1]</b>:  Unable to enroll: [_2]',$username,$reply));
1.10      www      2233:          		} else {
1.7       www      2234:                             $count++; $flushc++;
                   2235:                             $student{$username}=1;
1.6       www      2236:                             $r->print('. ');
1.7       www      2237:                             if ($flushc>15) {
                   2238: 				$r->rflush;
                   2239:                                 $flushc=0;
                   2240:                             }
1.6       www      2241:                         }
1.25      matthew  2242:                     } else {
1.88      matthew  2243:                         $r->print('<br />'.
                   2244:       &mt('<b>[_1]</b>: Unable to enroll.  No password specified.',$username)
                   2245:                                   );
1.25      matthew  2246:                     }
                   2247:                 }
1.26      matthew  2248:             }
                   2249:         } # end of foreach (@studentdata)
1.88      matthew  2250:         $r->print("</p>\n<p>\n".&mt('Processed [_1] student(s).',$count).
                   2251:                   "</p>\n");
                   2252:         $r->print("<p>\n".
                   2253:                   &mt('If active, the new role will be available when the '.
                   2254:                   'students next log in to LON-CAPA.')."</p>\n");
1.26      matthew  2255:         #####################################
                   2256:         #           Drop students           #
                   2257:         #####################################
1.127     albertel 2258:         if ($env{'form.fullup'} eq 'yes') {
1.88      matthew  2259:             $r->print('<h3>'.&mt('Dropping Students')."</h3>\n");
1.26      matthew  2260:             #  Get current classlist
1.56      matthew  2261:             my ($classlist,$keylist)=&Apache::loncoursedata::get_classlist();
                   2262:             if (! defined($classlist)) {
1.88      matthew  2263:                 $r->print(&mt('There are no students currently enrolled.').
                   2264:                           "\n");
1.56      matthew  2265:             } else {
                   2266:                 # Remove the students we just added from the list of students.
1.25      matthew  2267:                 foreach (@studentdata) {
                   2268:                     my %entries=&Apache::loncommon::record_sep($_);
                   2269:                     unless (($entries{$fields{'username'}} eq '') ||
                   2270:                             (!defined($entries{$fields{'username'}}))) {
1.56      matthew  2271:                         delete($classlist->{$entries{$fields{'username'}}.
1.26      matthew  2272:                                                 ':'.$domain});
1.25      matthew  2273:                     }
                   2274:                 }
1.56      matthew  2275:                 # Print out list of dropped students.
                   2276:                 &show_drop_list($r,$classlist,$keylist,'nosort');
1.25      matthew  2277:             }
                   2278:         }
1.26      matthew  2279:     } # end of unless
1.10      www      2280: }
                   2281: 
1.11      www      2282: # ================================================================== Phase four
                   2283: sub drop_student_list {
                   2284:     my $r=shift;
                   2285:     my $count=0;
1.128     albertel 2286:     my @droplist = &Apache::loncommon::get_env_multiple('form.droplist');
1.35      matthew  2287:     foreach (@droplist) {
1.26      matthew  2288:         my ($uname,$udom)=split(/\:/,$_);
1.56      matthew  2289:         # drop student
1.127     albertel 2290:         my $result = &modifystudent($udom,$uname,$env{'request.course.id'});
1.37      matthew  2291:         if ($result eq 'ok' || $result eq 'ok:') {
1.88      matthew  2292:             $r->print(&mt('Dropped [_1]',$uname.'@'.$udom).'<br>');
1.59      matthew  2293:             $count++;
1.35      matthew  2294:         } else {
1.88      matthew  2295:             $r->print(
                   2296:           &mt('Error dropping [_1]:[_2]',$uname.'@'.$udom,$result).
1.35      matthew  2297:                       '<br />');
                   2298:         }
1.20      harris41 2299:     }
1.88      matthew  2300:     $r->print('<p><b>'.&mt('Dropped [_1] student(s).',$count).'</b></p>');
                   2301:     $r->print('<p>'.&mt('Re-enrollment will re-activate data.')) if ($count);
1.11      www      2302: }
                   2303: 
1.50      matthew  2304: ###################################################################
                   2305: ###################################################################
                   2306: 
                   2307: =pod
                   2308: 
                   2309: =item &handler
                   2310: 
                   2311: The typical handler you see in all these modules.  Takes $r, the
                   2312: http request, as an argument.  
                   2313: 
                   2314: The response to the request is governed by two form variables
                   2315: 
                   2316:  form.action      form.state     response
                   2317:  ---------------------------------------------------
                   2318:  undefined        undefined      print main menu
                   2319:  upload           undefined      print courselist upload menu
                   2320:  upload           got_file       deal with uploaded file,
                   2321:                                  print the upload managing menu
                   2322:  upload           enrolling      enroll students based on upload
                   2323:  drop             undefined      print the classlist ready to drop
                   2324:  drop             done           drop the selected students
1.74      matthew  2325:  enrollstudent    undefined      print student username domain form
                   2326:  enrollstudent    gotusername    print single student enroll menu
1.50      matthew  2327:  enrollstudent    enrolling      enroll student
                   2328:  classlist        undefined      print html classlist
                   2329:  classlist        csv            print csv classlist
                   2330:  modifystudent    undefined      print classlist to select student to modify
                   2331:  modifystudent    selected       print modify student menu
                   2332:  modifystudent    done           make modifications to student record
                   2333: 
                   2334: =cut
                   2335: 
                   2336: ###################################################################
                   2337: ###################################################################
1.10      www      2338: sub handler {
1.26      matthew  2339:     my $r=shift;
                   2340:     if ($r->header_only) {
1.86      www      2341:         &Apache::loncommon::content_type($r,'text/html');
1.26      matthew  2342:         $r->send_http_header;
                   2343:         return OK;
                   2344:     }
1.48      matthew  2345:     &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1.50      matthew  2346:                                             ['action','state']);
1.102     matthew  2347: 
                   2348:     &Apache::lonhtmlcommon::clear_breadcrumbs();
                   2349:     &Apache::lonhtmlcommon::add_breadcrumb
                   2350:         ({href=>"/adm/dropadd",
                   2351:           text=>"Enrollment Manager",
                   2352:           faq=>9,bug=>'Instructor Interface',});
1.26      matthew  2353:     #  Needs to be in a course
1.127     albertel 2354:     if (! ($env{'request.course.fn'})) {
1.121     matthew  2355:         # Not in a course
1.127     albertel 2356:         $env{'user.error.msg'}=
1.132     raeburn  2357:             "/adm/dropadd:cst:0:0:Cannot manage or view course groups, ".
                   2358:                                   "or drop or add students";
1.50      matthew  2359:         return HTTP_NOT_ACCEPTABLE; 
                   2360:     }
                   2361:     #
1.121     matthew  2362:     my $view_permission = 
1.127     albertel 2363:         &Apache::lonnet::allowed('vcl',$env{'request.course.id'});
1.121     matthew  2364:     my $enrl_permission = 
1.127     albertel 2365:         &Apache::lonnet::allowed('cst',$env{'request.course.id'});
1.132     raeburn  2366: 
                   2367:     my $grp_view_permission =
                   2368:         &Apache::lonnet::allowed('vcg',$env{'request.course.id'});
                   2369:     my $grp_manage_permission =
                   2370:         &Apache::lonnet::allowed('mdg',$env{'request.course.id'});
                   2371: 
                   2372: 
                   2373:     if (! $grp_view_permission && ! $grp_manage_permission && 
                   2374:                                   ! $view_permission && ! $enrl_permission) {
1.127     albertel 2375:         $env{'user.error.msg'}=
1.132     raeburn  2376:      "/adm/coursegroups:cst:0:0:Cannot manage or view course groups, ".
                   2377:                                  "or drop or add students";
                   2378:         return HTTP_NOT_ACCEPTABLE;
1.121     matthew  2379:     }
1.132     raeburn  2380: 
1.121     matthew  2381:     #
1.50      matthew  2382:     # Only output the header information if they did not request csv format
                   2383:     #
1.103     matthew  2384:     # Start page
                   2385:     &Apache::loncommon::content_type($r,'text/html');
                   2386:     $r->send_http_header;
                   2387:     $r->print(&header());
1.50      matthew  2388:     #
                   2389:     # Main switch on form.action and form.state, as appropriate
1.127     albertel 2390:     if (! exists($env{'form.action'})) {
1.141   ! albertel 2391:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Enrollment Manager'));
1.132     raeburn  2392:         &print_main_menu($r,$enrl_permission,$view_permission,$grp_manage_permission,
                   2393:                          $grp_view_permission);
1.127     albertel 2394:     } elsif ($env{'form.action'} eq 'upload' && $enrl_permission) {
1.102     matthew  2395:         &Apache::lonhtmlcommon::add_breadcrumb
                   2396:             ({href=>'/adm/dropadd?action=upload&state=',
1.106     matthew  2397:               text=>"Upload Classlist"});
1.141   ! albertel 2398:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Upload Classlist',
        !          2399: 						      'Course_Create_Class_List'));
1.127     albertel 2400:         if (! exists($env{'form.state'})) {
1.50      matthew  2401:             &print_first_courselist_upload_form($r);            
1.127     albertel 2402:         } elsif ($env{'form.state'} eq 'got_file') {
1.50      matthew  2403:             &print_upload_manager_form($r);
1.127     albertel 2404:         } elsif ($env{'form.state'} eq 'enrolling') {
                   2405:             if ($env{'form.datatoken'}) {
1.26      matthew  2406:                 &upfile_drop_add($r);
1.50      matthew  2407:             } else {
                   2408:                 # Hmmm, this is an error
1.26      matthew  2409:             }
1.50      matthew  2410:         } else {
                   2411:             &print_first_courselist_upload_form($r);            
1.26      matthew  2412:         }
1.127     albertel 2413:     } elsif ($env{'form.action'} eq 'drop' && $enrl_permission) {
1.102     matthew  2414:         &Apache::lonhtmlcommon::add_breadcrumb
                   2415:             ({href=>'/adm/dropadd?action=drop',
1.106     matthew  2416:               text=>"Drop Students"});
1.141   ! albertel 2417:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Drop Students',
        !          2418: 						      'Course_Drop_Student'));
1.127     albertel 2419:         if (! exists($env{'form.state'})) {
1.51      matthew  2420:             &print_drop_menu($r);
1.127     albertel 2421:         } elsif ($env{'form.state'} eq 'done') {
1.26      matthew  2422:             &drop_student_list($r);
1.50      matthew  2423:         } else {
1.55      matthew  2424:             &print_drop_menu($r);
1.26      matthew  2425:         }
1.127     albertel 2426:     } elsif ($env{'form.action'} eq 'enrollstudent' && $enrl_permission) {
1.102     matthew  2427:         &Apache::lonhtmlcommon::add_breadcrumb
                   2428:             ({href=>'/adm/dropadd?action=enrollstudent',
1.106     matthew  2429:               text=>"Enroll Student"});
1.141   ! albertel 2430:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Enroll Student',
        !          2431: 						      'Course_Add_Student'));
1.127     albertel 2432:         if (! exists($env{'form.state'})) {
1.74      matthew  2433:             &get_student_username_domain_form($r);
1.127     albertel 2434:         } elsif ($env{'form.state'} eq 'gotusername') {
1.50      matthew  2435:             &print_enroll_single_student_form($r);
1.127     albertel 2436:         } elsif ($env{'form.state'} eq 'enrolling') {
1.26      matthew  2437:             &enroll_single_student($r);
1.50      matthew  2438:         } else {
1.74      matthew  2439:             &get_student_username_domain_form($r);
1.26      matthew  2440:         }
1.127     albertel 2441:     } elsif ($env{'form.action'} eq 'classlist' && $view_permission) {
1.102     matthew  2442:         &Apache::lonhtmlcommon::add_breadcrumb
                   2443:             ({href=>'/adm/dropadd?action=classlist',
1.106     matthew  2444:               text=>"View Classlist"});
1.141   ! albertel 2445:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('View Classlist',
        !          2446: 						      'Course_View_Class_List'));
1.127     albertel 2447:         if (! exists($env{'form.state'})) {
1.103     matthew  2448:             &print_html_classlist($r,undef);
1.127     albertel 2449:         } elsif ($env{'form.state'} eq 'csv') {
1.103     matthew  2450:             &print_html_classlist($r,'csv');
1.127     albertel 2451:         } elsif ($env{'form.state'} eq 'excel') {
1.103     matthew  2452:             &print_html_classlist($r,'excel');
1.50      matthew  2453:         } else {
1.103     matthew  2454:             &print_html_classlist($r,undef);
1.50      matthew  2455:         }
1.127     albertel 2456:     } elsif ($env{'form.action'} eq 'modifystudent' && $enrl_permission) {
1.102     matthew  2457:         &Apache::lonhtmlcommon::add_breadcrumb
                   2458:             ({href=>'/adm/dropadd?action=modifystudent',
1.106     matthew  2459:               text=>"Modify Student Data"});
1.141   ! albertel 2460:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Modify Student Data',
        !          2461: 						      'Course_Modify_Student_Data'));
1.127     albertel 2462:         if (! exists($env{'form.state'})) {
1.50      matthew  2463:             &print_html_classlist($r);
1.127     albertel 2464:         } elsif ($env{'form.state'} eq 'selected') {
1.50      matthew  2465:             &print_modify_student_form($r);
1.127     albertel 2466:         } elsif ($env{'form.state'} eq 'done') {
1.50      matthew  2467:             &modify_single_student($r);
                   2468:         } else {
                   2469:             &print_html_classlist($r);
                   2470:         }        
                   2471:     } else {
                   2472:         # We should not end up here, but I guess it is possible
                   2473:         &Apache::lonnet::logthis("Undetermined state in londropadd.pm.  ".
1.127     albertel 2474:                                  "form.action = ".$env{'form.action'}.
1.50      matthew  2475:                                  "Someone should fix this.");
1.141   ! albertel 2476:         $r->print(&Apache::lonhtmlcommon::breadcrumbs('Enrollment Manager'));
1.121     matthew  2477:         &print_main_menu($r,$enrl_permission,$view_permission);
1.50      matthew  2478:     }
                   2479:     #
                   2480:     # Finish up
1.138     albertel 2481:     $r->print('</form>'.&Apache::loncommon::end_page());
1.26      matthew  2482:     return OK;
1.1       www      2483: }
                   2484: 
1.50      matthew  2485: ###################################################################
                   2486: ###################################################################
                   2487: 
1.1       www      2488: 1;
                   2489: __END__
1.50      matthew  2490: 
1.1       www      2491: 

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