Annotation of loncom/interface/londropadd.pm, revision 1.138
1.1 www 1: # The LearningOnline Network with CAPA
2: # Handler to drop and add students in courses
3: #
1.138 ! albertel 4: # $Id: londropadd.pm,v 1.137 2006/02/18 01:27:07 raeburn 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> ');
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 = ' ';
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 = ' ';
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.127 albertel 908: &Apache::lonnet::coursedescription($env{'request.course.id'});
1.68 matthew 909: } else {
1.94 sakharuk 910: $result .= &mt('Unable to set default dates for course').":".$put_result.
1.68 matthew 911: '<br />';
912: }
913: return $result;
914: }
915:
1.74 matthew 916: ##
917: ## Single student enrollment routines (some of them)
918: ##
919: sub get_student_username_domain_form {
920: my $r = shift;
921: my $domform = &Apache::loncommon::select_dom_form
1.127 albertel 922: ($env{'course.'.$env{'request.course.id'}.'.domain'},'cudomain',0);
1.94 sakharuk 923: my %lt=&Apache::lonlocal::texthash(
924: 'eos' => "Enroll One Student",
925: 'usr' => "Username",
926: 'dom' => "Domain",
927: 'been' => "Begin Enrollment",
928: );
1.74 matthew 929: $r->print(<<END);
930: <input type="hidden" name="action" value="enrollstudent" />
931: <input type="hidden" name="state" value="gotusername" />
1.94 sakharuk 932: <h3>$lt{'eos'}</h3>
1.74 matthew 933: <table>
1.94 sakharuk 934: <tr><th>$lt{'usr'}:</th>
1.74 matthew 935: <td><input type="text" name="cuname" size="15" /></td></tr>
1.94 sakharuk 936: <tr><th>$lt{'dom'}:</th>
1.74 matthew 937: <td>$domform</td></tr>
938: <tr><th> </th>
939: <td>
1.94 sakharuk 940: <input type="submit" name="Begin Enrollment" value="$lt{'been'}" />
1.74 matthew 941: </td></tr>
942: </table>
1.120 albertel 943: <script type="text/javascript">
944: // the if prevents the script error if the browser can not handle this
945: if ( document.studentform.cuname ) { document.studentform.cuname.focus(); }
946: </script>
1.74 matthew 947: END
948: return;
949: }
950:
1.50 matthew 951: sub print_enroll_single_student_form {
1.10 www 952: my $r=shift;
1.94 sakharuk 953: $r->print("<h3>".&mt('Enroll One Student')."</h3>");
1.74 matthew 954: #
1.127 albertel 955: my $username = $env{'form.cuname'};
956: my $domain = $env{'form.cudomain'};
1.123 albertel 957: $username=~s/\W//gs;
958: $domain=~s/\W//gs;
1.74 matthew 959: my $home = &Apache::lonnet::homeserver($username,$domain);
960: # $new_user flags whether we are creating a new user or using an old one
961: my $new_user = 1;
962: if ($home ne 'no_host') {
963: $new_user = 0;
964: }
965: #
966: my $user_data_html = '';
967: my $javascript_validations = '';
968: if ($new_user) {
1.127 albertel 969: my $defdom=$env{'course.'.$env{'request.course.id'}.'.domain'};
1.74 matthew 970: # Set up authentication forms
971: my ($krbdef,$krbdefdom) =
1.75 matthew 972: &Apache::loncommon::get_kerberos_defaults($domain);
1.89 matthew 973: $javascript_validations=&javascript_validations('auth',$krbdefdom);
1.74 matthew 974: my %param = ( formname => 'document.studentform',
975: kerb_def_dom => $krbdefdom,
976: kerb_def_auth => $krbdef
977: );
978: my $krbform = &Apache::loncommon::authform_kerberos(%param);
979: my $intform = &Apache::loncommon::authform_internal(%param);
980: my $locform = &Apache::loncommon::authform_local(%param);
981: #
982: # Set up domain selection form
983: my $homeserver_form = '';
984: my %servers = &Apache::loncommon::get_library_servers($domain);
985: $homeserver_form = '<select name="lcserver" size="1">'."\n".
986: '<option value="default" selected>default</option>'."\n";
987: while (my ($servername,$serverdescription) = each (%servers)) {
988: $homeserver_form .= '<option value="'.$servername.'">'.
989: $serverdescription."</option>\n";
990: }
991: $homeserver_form .= "</select>\n";
992: #
993: #
1.94 sakharuk 994: my %lt=&Apache::lonlocal::texthash(
995: 'udf' => "User Data for",
996: 'fn' => "First Name",
997: 'mn' => "Middle Name",
998: 'ln' => "Last Name",
999: 'gen' => "Generation",
1000: 'hs' => "Home Server",
1001: 'pswd' => "Password",
1002: 'psam' => "Please select an authentication mechanism",
1.124 www 1003: 'mail' => "Email Address"
1.94 sakharuk 1004: );
1.130 www 1005: my $authhelp=&Apache::loncommon::help_open_topic('Auth_Options');
1.74 matthew 1006: $user_data_html = <<END;
1.94 sakharuk 1007: <h3>$lt{'udf'} $username\@$domain</h3>
1.74 matthew 1008: <table>
1.94 sakharuk 1009: <tr><th>$lt{'fn'}:</th>
1.74 matthew 1010: <td><input type="text" name="cfirst" size="15"></td></tr>
1.94 sakharuk 1011: <tr><th>$lt{'mn'}:</th>
1.74 matthew 1012: <td><input type="text" name="cmiddle" size="15"></td></tr>
1.94 sakharuk 1013: <tr><th>$lt{'ln'}:</th>
1.74 matthew 1014: <td><input type="text" name="clast" size="15"></td></tr>
1.94 sakharuk 1015: <tr><th>$lt{'gen'}:</th>
1.74 matthew 1016: <td><input type="text" name="cgen" size="5"> </td></tr>
1.94 sakharuk 1017: <tr><th>$lt{'hs'}:</th>
1.74 matthew 1018: <td>$homeserver_form</td></tr>
1.124 www 1019: <tr><th>$lt{'mail'}:</th>
1020: <td><input type="text" name="emailaddress" size="20" /></td></tr>
1.74 matthew 1021: </table>
1.94 sakharuk 1022: <h3>$lt{'pswd'}</h3>
1.130 www 1023: $lt{'psam'}$authhelp
1.74 matthew 1024: <table>
1025: <p>
1026: $krbform
1.75 matthew 1027: <br />
1.74 matthew 1028: $intform
1.75 matthew 1029: <br />
1.74 matthew 1030: $locform
1031: </p>
1032: END
1033: } else {
1034: # User already exists. Do not worry about authentication
1035: my %uenv = &Apache::lonnet::dump('environment',$domain,$username);
1.89 matthew 1036: $javascript_validations = &javascript_validations('noauth');
1.94 sakharuk 1037: my %lt=&Apache::lonlocal::texthash(
1038: 'udf' => "User Data for",
1039: 'fn' => "First Name",
1040: 'mn' => "Middle Name",
1041: 'ln' => "Last Name",
1042: 'gen' => "Generation",
1.124 www 1043: 'mail' => "Email Address",
1.94 sakharuk 1044: );
1.74 matthew 1045: $user_data_html = <<END;
1.94 sakharuk 1046: <h3>$lt{'udf'} $username\@$domain</h3>
1.74 matthew 1047: <input type="hidden" name="lcserver" value="default" />
1048: <table>
1.94 sakharuk 1049: <tr><th>$lt{'fn'}:</th>
1.74 matthew 1050: <td>
1051: <input type="text" name="cfirst" value="$uenv{'firstname'}" size="15" />
1052: </td></tr>
1.94 sakharuk 1053: <tr><th>$lt{'mn'}:</th>
1.74 matthew 1054: <td>
1055: <input type="text" name="cmiddle" value="$uenv{'middlename'}" size="15" />
1056: </td></tr>
1.94 sakharuk 1057: <tr><th>$lt{'ln'}:</th>
1.74 matthew 1058: <td>
1059: <input type="text" name="clast"value="$uenv{'lastname'}" size="15" />
1060: </td></tr>
1.94 sakharuk 1061: <tr><th>$lt{'gen'}:</th>
1.74 matthew 1062: <td>
1063: <input type="text" name="cgen" value="$uenv{'generation'}" size="5" />
1064: </td></tr>
1.124 www 1065: <tr><th>$lt{'mail'}:</th>
1066: <td>
1067: <input type="text" name="emailaddress" value="$uenv{'permanentemail'}" size="20" />
1068: </td></tr>
1.74 matthew 1069: </table>
1070: END
1071: }
1.68 matthew 1072: my $date_table = &date_setting_table();
1.74 matthew 1073: # Print it all out
1.94 sakharuk 1074: my %lt=&Apache::lonlocal::texthash(
1075: 'cd' => "Course Data",
1076: 'gs' => "Group/Section",
1077: 'idsn' => "ID/Student Number",
1078: 'disn' => "Disable ID/Student Number Safeguard and Force Change of Conflicting IDs (only do if you know what you are doing)",
1079: 'eas' => "Enroll as student",
1080: );
1.50 matthew 1081: $r->print(<<END);
1.74 matthew 1082: <input type="hidden" name="action" value="enrollstudent" />
1083: <input type="hidden" name="state" value="done" />
1084: <input type="hidden" name="cuname" value="$username" />
1085: <input type="hidden" name="lcdomain" value="$domain" />
1.28 matthew 1086: <script type="text/javascript" language="Javascript">
1.12 www 1087: function verify(vf) {
1088: var founduname=0;
1089: var foundpwd=0;
1090: var foundname=0;
1091: var foundid=0;
1092: var foundsec=0;
1093: var tw;
1.26 matthew 1094: if ((typeof(vf.cuname.value) !="undefined") && (vf.cuname.value!='') &&
1.31 matthew 1095: (typeof(vf.lcdomain.value)!="undefined") && (vf.lcdomain.value!='')) {
1.12 www 1096: founduname=1;
1097: }
1.14 harris41 1098: if ((typeof(vf.cfirst.value)!="undefined") && (vf.cfirst.value!='') &&
1.26 matthew 1099: (typeof(vf.clast.value) !="undefined") && (vf.clast.value!='')) {
1.12 www 1100: foundname=1;
1101: }
1.14 harris41 1102: if ((typeof(vf.csec.value)!="undefined") && (vf.csec.value!='')) {
1.12 www 1103: foundsec=1;
1104: }
1.14 harris41 1105: if ((typeof(vf.cstid.value)!="undefined") && (vf.cstid.value!='')) {
1.12 www 1106: foundid=1;
1107: }
1108: if (founduname==0) {
1109: alert('You need to specify at least the username and domain fields');
1110: return;
1111: }
1.24 albertel 1112: verify_message(vf,founduname,foundpwd,foundname,foundid,foundsec);
1.12 www 1113: }
1114:
1.24 albertel 1115: $javascript_validations
1.12 www 1116:
1.24 albertel 1117: function clearpwd(vf) {
1118: //nothing else needs clearing
1.15 albertel 1119: }
1120:
1.12 www 1121: </script>
1.11 www 1122:
1.74 matthew 1123: $user_data_html
1.50 matthew 1124:
1.94 sakharuk 1125: <h3>$lt{'cd'}</h3>
1.50 matthew 1126:
1.94 sakharuk 1127: <p>$lt{'gs'}: <input type="text" name="csec" size="5" />
1.50 matthew 1128: <p>
1.68 matthew 1129: $date_table
1.50 matthew 1130: </p>
1.94 sakharuk 1131: <h3>$lt{'idsn'}</h3>
1.50 matthew 1132: <p>
1.94 sakharuk 1133: $lt{'idsn'}: <input type="text" name="cstid" size="10">
1.26 matthew 1134: </p><p>
1.131 albertel 1135: <label>
1.26 matthew 1136: <input type="checkbox" name="forceid" value="yes">
1.94 sakharuk 1137: $lt{'disn'}
1.131 albertel 1138: </label>
1.50 matthew 1139: </p><p>
1.94 sakharuk 1140: <input type="button" onClick="verify(this.form)" value="$lt{'eas'}">
1.26 matthew 1141: </p>
1.50 matthew 1142: END
1143: return;
1.10 www 1144: }
1145:
1146: # ========================================================= Menu Phase Two Drop
1.51 matthew 1147: sub print_drop_menu {
1.10 www 1148: my $r=shift;
1.92 sakharuk 1149: $r->print("<h3>".&mt('Drop Students')."</h3>");
1.127 albertel 1150: my $cid=$env{'request.course.id'};
1.56 matthew 1151: my ($classlist,$keylist) = &Apache::loncoursedata::get_classlist();
1152: if (! defined($classlist)) {
1.94 sakharuk 1153: $r->print(&mt('There are no students currently enrolled.')."\n");
1.51 matthew 1154: return;
1.25 matthew 1155: }
1.51 matthew 1156: # Print out the available choices
1.56 matthew 1157: &show_drop_list($r,$classlist,$keylist);
1.51 matthew 1158: return;
1.11 www 1159: }
1160:
1.40 matthew 1161: # ============================================== view classlist
1.50 matthew 1162: sub print_html_classlist {
1.103 matthew 1163: my ($r,$mode) = @_;
1.127 albertel 1164: if (! exists($env{'form.sortby'})) {
1165: $env{'form.sortby'} = 'username';
1.57 matthew 1166: }
1.127 albertel 1167: if ($env{'form.Status'} !~ /^(Any|Expired|Active)$/) {
1168: $env{'form.Status'} = 'Active';
1.57 matthew 1169: }
1170: my $status_select = &Apache::lonhtmlcommon::StatusOptions
1.127 albertel 1171: ($env{'form.Status'});
1172: my $cid=$env{'request.course.id'};
1173: my $cdom=$env{'course.'.$cid.'.domain'};
1174: my $cnum=$env{'course.'.$cid.'.num'};
1.103 matthew 1175: #
1176: # List course personnel
1.100 www 1177: my %coursepersonnel=&Apache::lonnet::get_course_adv_roles($cdom.'/'.$cnum);
1.110 matthew 1178: #
1.127 albertel 1179: if (! defined($env{'form.output'}) ||
1180: $env{'form.output'} !~ /^(csv|excel|html)$/ ) {
1181: $env{'form.output'} = 'html';
1.110 matthew 1182: }
1183: #
1.103 matthew 1184: $r->print('<br /><table border="2">');
1.110 matthew 1185: foreach my $role (sort keys %coursepersonnel) {
1186: next if ($role =~ /^\s*$/);
1187: $r->print('<tr><td>'.$role.'</td><td>');
1188: foreach my $user (split(',',$coursepersonnel{$role})) {
1189: my ($puname,$pudom)=split(':',$user);
1.100 www 1190: $r->print(' '.&Apache::loncommon::aboutmewrapper(
1.110 matthew 1191: &Apache::loncommon::plainname($puname,
1192: $pudom),
1193: $puname,$pudom));
1.100 www 1194: }
1195: $r->print('</td></tr>');
1196: }
1197: $r->print('</table>');
1.103 matthew 1198: #
1199: # Interface output
1200: $r->print('<input type="hidden" name="action" value="'.
1.127 albertel 1201: $env{'form.action'}.'" />');
1.103 matthew 1202: $r->print("<p>\n");
1.127 albertel 1203: if ($env{'form.action'} ne 'modifystudent') {
1.103 matthew 1204: my %lt=&Apache::lonlocal::texthash('csv' => "CSV",
1205: 'excel' => "Excel",
1206: 'html' => 'HTML');
1.110 matthew 1207: my $output_selector = '<select size="1" name="output" >';
1.103 matthew 1208: foreach my $outputformat ('html','csv','excel') {
1209: my $option = '<option value="'.$outputformat.'" ';
1.127 albertel 1210: if ($outputformat eq $env{'form.output'}) {
1.104 matthew 1211: $option .= 'selected ';
1.103 matthew 1212: }
1213: $option .='>'.$lt{$outputformat}.'</option>';
1214: $output_selector .= "\n".$option;
1215: }
1216: $output_selector .= '</select>';
1217: $r->print(&mt('Output Format: [_1]',$output_selector).(' 'x3));
1.59 matthew 1218: }
1.103 matthew 1219: $r->print(&mt('Student Status: [_1]',$status_select)."\n");
1.105 matthew 1220: $r->print('<input type="submit" value="'.&mt('Update Display').'" />'.
1221: "\n</p>\n");
1.103 matthew 1222: #
1223: # Print the classlist
1224: $r->print('<h2>'.&mt('Current Class List').'</h2>');
1.56 matthew 1225: my ($classlist,$keylist)=&Apache::loncoursedata::get_classlist();
1226: if (! defined($classlist)) {
1.94 sakharuk 1227: $r->print(&mt('There are no students currently enrolled.')."\n");
1.40 matthew 1228: } else {
1229: # Print out the available choices
1.127 albertel 1230: if ($env{'form.action'} eq 'modifystudent') {
1.110 matthew 1231: &show_class_list($r,'view','modify',
1.127 albertel 1232: $env{'form.Status'},$classlist,$keylist);
1.110 matthew 1233: } else {
1.127 albertel 1234: &show_class_list($r,$env{'form.output'},'aboutme',
1235: $env{'form.Status'},$classlist,$keylist);
1.50 matthew 1236: }
1.41 matthew 1237: }
1238: }
1239:
1.40 matthew 1240: # =================================================== Show student list to drop
1241: sub show_class_list {
1.110 matthew 1242: my ($r,$mode,$linkto,$statusmode,$classlist,$keylist)=@_;
1.127 albertel 1243: my $cid=$env{'request.course.id'};
1.60 matthew 1244: #
1245: # Variables for excel output
1.104 matthew 1246: my ($excel_workbook, $excel_sheet, $excel_filename,$row,$format);
1.60 matthew 1247: #
1.103 matthew 1248: # Variables for csv output
1249: my ($CSVfile,$CSVfilename);
1250: #
1.127 albertel 1251: my $sortby = $env{'form.sortby'};
1.114 raeburn 1252: if ($sortby !~ /^(username|domain|section|fullname|id|start|end|type)$/) {
1.53 matthew 1253: $sortby = 'username';
1254: }
1.134 raeburn 1255: if (! exists($env{'form.displayphotos'})) {
1256: $env{'form.displayphotos'} = 'off';
1257: }
1258: my $displayphotos = $env{'form.displayphotos'};
1259:
1.42 matthew 1260: # Print out header
1.114 raeburn 1261: unless ($mode eq 'autoenroll') {
1262: $r->print(<<END);
1.127 albertel 1263: <input type="hidden" name="state" value="$env{'form.state'}" />
1.114 raeburn 1264: END
1265: }
1.103 matthew 1266: $r->print(<<END);
1267: <input type="hidden" name="sortby" value="$sortby" />
1.134 raeburn 1268: <input type="hidden" name="displayphotos" value="$displayphotos" />
1.103 matthew 1269: END
1.114 raeburn 1270: if ($mode eq 'html' || $mode eq 'view' || $mode eq 'autoenroll') {
1.50 matthew 1271: if ($linkto eq 'aboutme') {
1.94 sakharuk 1272: $r->print(&mt('Select a user name to view the users personal page.'));
1.50 matthew 1273: } elsif ($linkto eq 'modify') {
1.94 sakharuk 1274: $r->print(&mt('Select a user name to modify the students information'));
1.50 matthew 1275: }
1.94 sakharuk 1276: my %lt=&Apache::lonlocal::texthash(
1.110 matthew 1277: 'usrn' => "username",
1278: 'dom' => "domain",
1279: 'sn' => "student name",
1280: 'sec' => "section",
1281: 'start' => "start date",
1282: 'end' => "end date",
1.134 raeburn 1283: 'type' => "enroll type/action",
1284: 'photo' => "photo",
1.94 sakharuk 1285: );
1.114 raeburn 1286: unless ($mode eq 'autoenroll') {
1287: $r->print(<<END);
1.59 matthew 1288: <input type="hidden" name="sname" value="" />
1289: <input type="hidden" name="sdom" value="" />
1.114 raeburn 1290: END
1291: }
1.136 raeburn 1292: if ($env{'course.'.$env{'request.course.id'}.'.internal.showphoto'}) {
1.134 raeburn 1293: $r->print('
1294: <script type="text/javascript">
1295: function photowindow(photolink) {
1296: var title = "Photo_Viewer";
1297: var options = "scrollbars=1,resizable=1,menubar=0";
1298: options += ",width=240,height=240";
1299: stdeditbrowser = open(photolink,title,options,"1");
1300: stdeditbrowser.focus();
1301: }
1302: </script>
1303: ');
1304: }
1.115 raeburn 1305: $r->print("
1.40 matthew 1306: <p>
1307: <table border=2>
1.115 raeburn 1308: <tr>
1309: ");
1310: if ($mode eq 'autoenroll') {
1311: $r->print("
1312: <th><a href=\"javascript:document.studentform.sortby.value='type';document.studentform.submit();\">$lt{'type'}</a></th>
1313: ");
1314: } else {
1315: $r->print("
1316: <th>Count</th>
1317: ");
1318: }
1319: $r->print(<<END);
1320: <th>
1.94 sakharuk 1321: <a href="javascript:document.studentform.sortby.value='username';document.studentform.submit();">$lt{'usrn'}</a>
1.53 matthew 1322: </th><th>
1.94 sakharuk 1323: <a href="javascript:document.studentform.sortby.value='domain';document.studentform.submit();">$lt{'dom'}</a>
1.53 matthew 1324: </th><th>
1.57 matthew 1325: <a href="javascript:document.studentform.sortby.value='id';document.studentform.submit();">ID</a>
1.53 matthew 1326: </th><th>
1.94 sakharuk 1327: <a href="javascript:document.studentform.sortby.value='fullname';document.studentform.submit();">$lt{'sn'}</a>
1.53 matthew 1328: </th><th>
1.94 sakharuk 1329: <a href="javascript:document.studentform.sortby.value='section';document.studentform.submit();">$lt{'sec'}</a>
1.110 matthew 1330: </th><th>
1331: <a href="javascript:document.studentform.sortby.value='start';document.studentform.submit();">$lt{'start'}</a>
1332: </th><th>
1333: <a href="javascript:document.studentform.sortby.value='end';document.studentform.submit();">$lt{'end'}</a>
1.53 matthew 1334: </th>
1.40 matthew 1335: END
1.136 raeburn 1336: if ($env{'course.'.$env{'request.course.id'}.'.internal.showphoto'}) {
1.135 albertel 1337: my %photo_options = &Apache::lonlocal::texthash(
1.134 raeburn 1338: 'on' => 'Show',
1339: 'off' => 'Hide',
1340: );
1341: my $photochg = 'on';
1342: if ($displayphotos eq 'on') {
1343: $photochg = 'off';
1344: }
1345: $r->print(' <th>'."\n".' '.
1346: '<a href="javascript:document.studentform.displayphotos.value='.
1347: "'".$photochg."'".';document.studentform.submit();">'.
1348: $photo_options{$photochg}.'</a> '.$lt{'photo'}."\n".
1349: ' </th>'."\n");
1350: }
1351: $r->print(" </tr>\n");
1.41 matthew 1352: } elsif ($mode eq 'csv') {
1.103 matthew 1353: #
1354: # Open a file
1355: $CSVfilename = '/prtspool/'.
1.127 albertel 1356: $env{'user.name'}.'_'.$env{'user.domain'}.'_'.
1.103 matthew 1357: time.'_'.rand(1000000000).'.csv';
1358: unless ($CSVfile = Apache::File->new('>/home/httpd'.$CSVfilename)) {
1359: $r->log_error("Couldn't open $CSVfilename for output $!");
1360: $r->print("Problems occured in writing the csv file. ".
1361: "This error has been logged. ".
1362: "Please alert your LON-CAPA administrator.");
1363: $CSVfile = undef;
1364: }
1365: #
1366: # Write headers and data to file
1.58 matthew 1367: if($statusmode eq 'Expired') {
1.103 matthew 1368: print $CSVfile '"'.&mt('Students with expired roles').'"'."\n";
1.58 matthew 1369: }
1370: if ($statusmode eq 'Any') {
1.103 matthew 1371: print $CSVfile '"'.join('","',map {
1372: &Apache::loncommon::csv_translate(&mt($_))
1373: } ("username","domain","ID","student name",
1.110 matthew 1374: "section","start date","end date","status")).'"'."\n";
1.58 matthew 1375: } else {
1.103 matthew 1376: print $CSVfile '"'.join('","',map {
1377: &Apache::loncommon::csv_translate(&mt($_))
1378: } ("username","domain","ID","student name",
1.110 matthew 1379: "section","start date","end date")).'"'."\n";
1.58 matthew 1380: }
1.60 matthew 1381: } elsif ($mode eq 'excel') {
1382: # Create the excel spreadsheet
1.126 matthew 1383: ($excel_workbook,$excel_filename,$format) =
1384: &Apache::loncommon::create_workbook($r);
1385: return if (! defined($excel_workbook));
1.60 matthew 1386: $excel_sheet = $excel_workbook->addworksheet('classlist');
1387: #
1.76 albertel 1388: my $description = 'Class List for '.
1.127 albertel 1389: $env{'course.'.$env{'request.course.id'}.'.description'};
1.104 matthew 1390: $excel_sheet->write($row++,0,$description,$format->{'h1'});
1.60 matthew 1391: #
1392: $excel_sheet->write($row++,0,["username","domain","ID",
1.110 matthew 1393: "student name","section",
1394: "start date","end date","status"],
1395: $format->{'bold'});
1.41 matthew 1396: }
1.56 matthew 1397: #
1398: # Sort the students
1399: my %index;
1400: my $i;
1401: foreach (@$keylist) {
1402: $index{$_} = $i++;
1403: }
1404: my $index = $index{$sortby};
1405: my $second = $index{'username'};
1406: my $third = $index{'domain'};
1.53 matthew 1407: my @Sorted_Students = sort {
1.56 matthew 1408: lc($classlist->{$a}->[$index]) cmp lc($classlist->{$b}->[$index])
1409: ||
1410: lc($classlist->{$a}->[$second]) cmp lc($classlist->{$b}->[$second])
1411: ||
1412: lc($classlist->{$a}->[$third]) cmp lc($classlist->{$b}->[$third])
1413: } (keys(%$classlist));
1.108 matthew 1414: my $studentcount = 0;
1.115 raeburn 1415: my $autocount = 0;
1416: my $manualcount = 0;
1417: my $unlockcount = 0;
1418: my $lockcount = 0;
1.53 matthew 1419: foreach my $student (@Sorted_Students) {
1.110 matthew 1420: my $sdata = $classlist->{$student};
1421: my $username = $sdata->[$index{'username'}];
1422: my $domain = $sdata->[$index{'domain'}];
1423: my $section = $sdata->[$index{'section'}];
1424: my $name = $sdata->[$index{'fullname'}];
1425: my $id = $sdata->[$index{'id'}];
1426: my $status = $sdata->[$index{'status'}];
1427: my $start = $sdata->[$index{'start'}];
1428: my $end = $sdata->[$index{'end'}];
1.115 raeburn 1429: my $type = $sdata->[$index{'type'}];
1.57 matthew 1430: next if (($statusmode ne 'Any') && ($status ne $statusmode));
1.114 raeburn 1431: if ($mode eq 'view' || $mode eq 'html' || $mode eq 'autoenroll') {
1432: if (! defined($start) || $start == 0) {
1433: $start = &mt('none');
1434: } else {
1435: $start = &Apache::lonlocal::locallocaltime($start);
1436: }
1437: if (! defined($end) || $end == 0) {
1438: $end = &mt('none');
1439: } else {
1440: $end = &Apache::lonlocal::locallocaltime($end);
1441: }
1.115 raeburn 1442: $r->print("<tr>\n ");
1443: if ($mode eq 'autoenroll') {
1444: my $lockedtype = $sdata->[$index{'lockedtype'}];
1445: $studentcount++;
1446: my $cellentry;
1447: if ($type eq 'auto') {
1.131 albertel 1448: $cellentry = '<b>'.&mt('auto').'</b> <label><input type="checkbox" name="chgauto" value="'.$username.':'.$domain.'" /> Change</label>';
1.115 raeburn 1449: $autocount ++;
1450: } else {
1.131 albertel 1451: $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.'" /> Change</label></nobr></td></tr><tr><td><nobr>';
1.115 raeburn 1452: $manualcount ++;
1453: if ($lockedtype) {
1.131 albertel 1454: $cellentry .= '<label><input type="checkbox" name="unlockchg" value="'.$username.':'.$domain.'" /> '.&mt('Unlock').'</label>';
1.115 raeburn 1455: $unlockcount ++;
1456: } else {
1.131 albertel 1457: $cellentry .= '<label><input type="checkbox" name="lockchg" value="'.$username.':'.$domain.'" /> '.&mt('Lock').'</label>';
1.115 raeburn 1458: $lockcount ++;
1459: }
1.118 raeburn 1460: $cellentry .= '</nobr></td></tr></table>';
1.115 raeburn 1461: }
1462: $r->print("<td>$cellentry<td>\n ");
1463: } else {
1464: $r->print("<td>".(++$studentcount)."</td><td>\n ");
1465: }
1.51 matthew 1466: if ($linkto eq 'nothing') {
1467: $r->print($username);
1468: } elsif ($linkto eq 'aboutme') {
1469: $r->print(&Apache::loncommon::aboutmewrapper($username,
1470: $username,
1471: $domain));
1472: } elsif ($linkto eq 'modify') {
1.59 matthew 1473: $r->print('<a href="'.
1474: "javascript:document.studentform.sname.value='".
1475: $username.
1476: "';document.studentform.sdom.value='".$domain.
1477: "';document.studentform.state.value='selected".
1478: "';document.studentform.submit();".'">'.
1.53 matthew 1479: $username."</a>\n");
1.50 matthew 1480: }
1.51 matthew 1481: $r->print(<<"END");
1.50 matthew 1482: </td>
1.51 matthew 1483: <td>$domain</td>
1484: <td>$id</td>
1485: <td>$name</td>
1486: <td>$section</td>
1.110 matthew 1487: <td>$start</td>
1488: <td>$end</td>
1.114 raeburn 1489: END
1.134 raeburn 1490: if ($env{'course.'.$env{'request.course.id'}.
1.136 raeburn 1491: '.internal.showphoto'}) {
1.134 raeburn 1492: if ($displayphotos eq 'on') {
1.135 albertel 1493: my $imgurl =
1494: &Apache::lonnet::retrievestudentphoto($domain,
1495: $username,'gif',
1496: 'thumbnail');
1.134 raeburn 1497:
1498: $r->print(' <td align="right"><a href="javascript:photowindow('."'".&Apache::lonnet::studentphoto($domain,$username,'jpg')."'".')"><img src="'.$imgurl.'" border="1"></a></td>');
1499: } else {
1500: $r->print(' <td> </td> ');
1501: }
1502: }
1503: $r->print(' </tr> ');
1.51 matthew 1504: } elsif ($mode eq 'csv') {
1.103 matthew 1505: next if (! defined($CSVfile));
1.51 matthew 1506: # no need to bother with $linkto
1.114 raeburn 1507: if (! defined($start) || $start == 0) {
1508: $start = &mt('none');
1509: } else {
1510: $start = &Apache::lonlocal::locallocaltime($start);
1511: }
1512: if (! defined($end) || $end == 0) {
1513: $end = &mt('none');
1514: } else {
1515: $end = &Apache::lonlocal::locallocaltime($end);
1516: }
1.51 matthew 1517: my @line = ();
1.110 matthew 1518: foreach ($username,$domain,$id,$name,$section,$start,$end) {
1.51 matthew 1519: push @line,&Apache::loncommon::csv_translate($_);
1.58 matthew 1520: }
1521: if ($statusmode eq 'Any') {
1522: push @line,&Apache::loncommon::csv_translate($status);
1.41 matthew 1523: }
1.103 matthew 1524: print $CSVfile '"'.join('","',@line).'"'."\n";
1.60 matthew 1525: } elsif ($mode eq 'excel') {
1.110 matthew 1526: $excel_sheet->write($row,0,[$username,$domain,$id,
1527: $name,$section]);
1528: my $col = 5;
1529: foreach my $time ($start,$end) {
1.129 matthew 1530: if (defined($time) && $time != 0) {
1531: $excel_sheet->write($row,$col++,
1.110 matthew 1532: &Apache::lonstathelpers::calc_serial($time),
1533: $format->{'date'});
1.129 matthew 1534: } else {
1535: $excel_sheet->write($row,$col++,'none');
1536: }
1.110 matthew 1537: }
1538: $excel_sheet->write($row,$col++,$status);
1539: $row++;
1.40 matthew 1540: }
1541: }
1.114 raeburn 1542: if ($mode eq 'view' || $mode eq 'html' || $mode eq 'autoenroll') {
1.60 matthew 1543: $r->print('</table><br>');
1544: } elsif ($mode eq 'excel') {
1545: $excel_workbook->close();
1546: $r->print('<p><a href="'.$excel_filename.'">'.
1.94 sakharuk 1547: &mt('Your Excel spreadsheet').'</a> '.&mt('is ready for download').'.</p>'."\n");
1.103 matthew 1548: } elsif ($mode eq 'csv') {
1549: close($CSVfile);
1550: $r->print('<a href="'.$CSVfilename.'">'.
1551: &mt('Your CSV file').'</a> is ready for download.'.
1552: "\n");
1553: $r->rflush();
1.60 matthew 1554: }
1.114 raeburn 1555: if ($mode eq 'autoenroll') {
1.115 raeburn 1556: return ($studentcount,$autocount,$manualcount,$lockcount,$unlockcount);
1.114 raeburn 1557: }
1.115 raeburn 1558: return;
1.40 matthew 1559: }
1560:
1.50 matthew 1561:
1562: #
1563: # print out form for modification of a single students data
1564: #
1565: sub print_modify_student_form {
1566: my $r = shift();
1567: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1.59 matthew 1568: ['sdom','sname']);
1.127 albertel 1569: my $sname = $env{'form.sname'};
1570: my $sdom = $env{'form.sdom'};
1571: my $sortby = $env{'form.sortby'};
1.50 matthew 1572: # determine the students name information
1573: my %info=&Apache::lonnet::get('environment',
1574: ['firstname','middlename',
1.52 matthew 1575: 'lastname','generation','id'],
1.50 matthew 1576: $sdom, $sname);
1577: my ($tmp) = keys(%info);
1578: if ($tmp =~ /^(con_lost|error|no_such_host)/i) {
1.94 sakharuk 1579: $r->print('<font color="#ff0000" size="+2">'.&mt('Error').'</font>'.
1.50 matthew 1580: '<p>'.
1.94 sakharuk 1581: &mt('Unable to retrieve environment data for').' '.$sname.
1582: &mt('in domain').' '.$sdom.'</p><p>'.
1.138 ! albertel 1583: &mt('Please contact your LON-CAPA administrator regarding this situation.').'</p>'.&Apache::loncommon::end_page());
1.50 matthew 1584: return;
1585: }
1586: # determine the students starting and ending times and section
1587: my ($starttime,$endtime,$section) = &get_enrollment_data($sname,$sdom);
1.87 matthew 1588: if ($starttime =~ /^error/) {
1.94 sakharuk 1589: $r->print('<h2>'&mt('Error').'</h2>');
1.87 matthew 1590: $r->print('<p>'.$starttime.'</p>');
1591: return;
1592: }
1.101 matthew 1593: #
1.50 matthew 1594: # Deal with date forms
1.101 matthew 1595: my $current_date_description = '';
1596: my $textdate = '';
1597:
1598: if (! defined($starttime) || $starttime == 0) {
1599: $current_date_description = &mt('Current Starting Date: not set').
1600: '<br />';
1601: } else {
1602: $current_date_description =
1603: &mt('Current Starting Date: [_1]',
1604: &Apache::lonlocal::locallocaltime($starttime)).'<br />';
1605: }
1606: if (! defined($endtime) || $endtime == 0) {
1607: $current_date_description.= &mt('Current Ending Date: not set').
1608: '<br />';
1609: } else {
1610: $current_date_description.=
1611: &mt('Current Ending Date: [_1]',
1612: &Apache::lonlocal::locallocaltime($endtime)).'<br />';
1613:
1614: }
1.68 matthew 1615: my $date_table = &date_setting_table($starttime,$endtime);
1.59 matthew 1616: #
1.127 albertel 1617: if (! exists($env{'form.Status'}) ||
1618: $env{'form.Status'} !~ /^(Any|Expired|Active)$/) {
1619: $env{'form.Status'} = 'crap';
1.59 matthew 1620: }
1.94 sakharuk 1621: # Make sure student is enrolled in course
1622: my %lt=&Apache::lonlocal::texthash(
1623: 'mef' => "Modify Enrollment for",
1624: 'odcc' => "Only domain coordinators can change a users password.",
1625: 'sn' => "Student Name",
1626: 'fn' => "First",
1627: 'mn' => "Middle",
1628: 'ln' => "Last",
1629: 'gen' => "Generation",
1630: 'sid' => "Student ID",
1631: 'disn' => "Disable ID/Student Number Safeguard and Force Change of Conflicting IDs (only do if you know what you are doing)",
1632: 'sec' => "Section",
1633: 'sm' => "Submit Modifications",
1634: );
1.50 matthew 1635: $r->print(<<END);
1.52 matthew 1636: <p>
1637: <font size="+1">
1.94 sakharuk 1638: $lt{'odcc'}
1.52 matthew 1639: </font>
1640: </p>
1.50 matthew 1641: <input type="hidden" name="slogin" value="$sname" />
1642: <input type="hidden" name="sdomain" value="$sdom" />
1643: <input type="hidden" name="action" value="modifystudent" />
1.53 matthew 1644: <input type="hidden" name="state" value="done" />
1645: <input type="hidden" name="sortby" value="$sortby" />
1.127 albertel 1646: <input type="hidden" name="Status" value="$env{'form.Status'}" />
1.94 sakharuk 1647: <h2>$lt{'mef'} $info{'firstname'} $info{'middlename'}
1.50 matthew 1648: $info{'lastname'} $info{'generation'}, $sname\@$sdom</h2>
1649: <p>
1.94 sakharuk 1650: <b>$lt{'sn'}</b>
1.50 matthew 1651: <table>
1.94 sakharuk 1652: <tr><th>$lt{'fn'}</th><th>$lt{'mn'}</th><th>$lt{'ln'}</th><th>$lt{'gen'}</th></tr>
1.50 matthew 1653: <tr><td>
1654: <input type="text" name="firstname" value="$info{'firstname'}" /></td><td>
1655: <input type="text" name="middlename" value="$info{'middlename'}" /></td><td>
1656: <input type="text" name="lastname" value="$info{'lastname'}" /></td><td>
1657: <input type="text" name="generation" value="$info{'generation'}" /></td></tr>
1658: </table>
1659: </p><p>
1.94 sakharuk 1660: <b>$lt{'sid'}</b>: <input type="text" name="id" value="$info{'id'}" size="12"/>
1.52 matthew 1661: </p><p>
1.131 albertel 1662: <label>
1.53 matthew 1663: <input type="checkbox" name="forceid" >
1.94 sakharuk 1664: $lt{'disn'}
1.131 albertel 1665: </label>
1.53 matthew 1666: </p><p>
1.101 matthew 1667: <b>$lt{'sec'}</b>: <input type="text" name="section" value="$section" size="14"/>
1.50 matthew 1668: </p>
1.101 matthew 1669: <p>$current_date_description</p>
1.68 matthew 1670: <p>$date_table</p>
1.94 sakharuk 1671: <input type="submit" value="$lt{'sm'}" />
1.50 matthew 1672: END
1.138 ! albertel 1673: $r->print(&Apache::loncommon::end_page());
1.50 matthew 1674: return;
1675: }
1676:
1677: #
1678: # modify a single students section
1679: #
1680: sub modify_single_student {
1.138 ! albertel 1681: my ($r) = @_;
1.68 matthew 1682: #
1.80 matthew 1683: # Remove non alphanumeric values from the section
1.127 albertel 1684: $env{'form.section'} =~ s/\W//g;
1.77 matthew 1685: #
1.68 matthew 1686: # Do the date defaults first
1687: my ($starttime,$endtime) = &get_dates_from_form();
1.127 albertel 1688: if ($env{'form.makedatesdefault'}) {
1.68 matthew 1689: $r->print(&make_dates_default($starttime,$endtime));
1690: }
1.59 matthew 1691: # Get the 'sortby' and 'Status' variables so the user goes back to their
1692: # previous screen
1.127 albertel 1693: my $sortby = $env{'form.sortby'};
1694: my $status = $env{'form.Status'};
1.53 matthew 1695: #
1696: # We always need this information
1.127 albertel 1697: my $slogin = $env{'form.slogin'};
1698: my $sdom = $env{'form.sdomain'};
1.53 matthew 1699: #
1700: # Get the old data
1701: my %old=&Apache::lonnet::get('environment',
1702: ['firstname','middlename',
1703: 'lastname','generation','id'],
1704: $sdom, $slogin);
1.59 matthew 1705: $old{'section'} = &Apache::lonnet::getsection($sdom,$slogin,
1.127 albertel 1706: $env{'request.course.id'});
1.53 matthew 1707: my ($tmp) = keys(%old);
1708: if ($tmp =~ /^(con_lost|error|no_such_host)/i) {
1.94 sakharuk 1709: $r->print(&mt('There was an error determining the environment values for')." $slogin \@ $sdom.");
1.53 matthew 1710: return;
1711: }
1712: undef $tmp;
1713: #
1714: # Get the new data
1.127 albertel 1715: my $firstname = $env{'form.firstname'};
1716: my $middlename = $env{'form.middlename'};
1717: my $lastname = $env{'form.lastname'};
1718: my $generation = $env{'form.generation'};
1719: my $section = $env{'form.section'};
1720: my $courseid = $env{'request.course.id'};
1721: my $sid = $env{'form.id'};
1.50 matthew 1722: my $displayable_starttime = localtime($starttime);
1723: my $displayable_endtime = localtime($endtime);
1.53 matthew 1724: #
1725: # check for forceid override
1.63 matthew 1726: if ((defined($old{'id'})) && ($old{'id'} ne '') &&
1.127 albertel 1727: ($sid ne $old{'id'}) && (! exists($env{'form.forceid'}))) {
1.94 sakharuk 1728: $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 1729: $sid = $old{'id'};
1730: }
1731: #
1.50 matthew 1732: # talk to the user about what we are going to do
1.94 sakharuk 1733: my %lt=&Apache::lonlocal::texthash(
1734: 'mdu' => "Modifying data for user",
1735: 'si' => "Student Information",
1736: 'fd' => "Field",
1737: 'ov' => "Old Value",
1738: 'nv' => "New Value",
1739: 'fn' => "First name",
1740: 'mn' => "Middle name",
1741: 'ln' => "Last name",
1742: 'gen' => "Generation",
1743: 'sec' => "Section",
1744: 'ri' => "Role Information",
1745: 'st' => "Start Time",
1746: 'et' => "End Time",
1747: );
1.50 matthew 1748: $r->print(<<END);
1.94 sakharuk 1749: <h2>$lt{'mdu'} $slogin \@ $sdom </h2>
1750: <h3>$lt{'si'}</h3>
1.53 matthew 1751: <table rules="rows" border="1" cellpadding="3" >
1752: <tr>
1.94 sakharuk 1753: <th> $lt{'fd'} </th>
1754: <th> $lt{'ov'} </th>
1755: <th> $lt{'nv'} </th>
1.53 matthew 1756: </tr>
1757: <tr>
1.94 sakharuk 1758: <td> <b>$lt{'fn'}</b> </td>
1.53 matthew 1759: <td> $old{'firstname'} </td>
1760: <td> $firstname </td>
1761: </tr><tr>
1.94 sakharuk 1762: <td> <b>$lt{'mn'}</b> </td>
1.53 matthew 1763: <td> $old{'middlename'} </td>
1764: <td> $middlename </td>
1765: </tr><tr>
1.94 sakharuk 1766: <td> <b>$lt{'ln'}</b> </td>
1.53 matthew 1767: <td> $old{'lastname'} </td>
1768: <td> $lastname </td>
1769: </tr><tr>
1.94 sakharuk 1770: <td> <b>$lt{'gen'}</b> </td>
1.53 matthew 1771: <td> $old{'generation'} </td>
1772: <td> $generation </td>
1773: </tr><tr>
1774: <td> <b>ID</b> </td>
1775: <td> $old{'id'} </td>
1776: <td> $sid </td>
1.59 matthew 1777: </tr><tr>
1.94 sakharuk 1778: <td> <b>$lt{'sec'}</b> </td>
1.59 matthew 1779: <td> $old{'section'} </td>
1780: <td> $section</td>
1.53 matthew 1781: </tr>
1.50 matthew 1782: </table>
1.94 sakharuk 1783: <h3>$lt{'ri'}</h3>
1.50 matthew 1784: <table>
1.94 sakharuk 1785: <tr><td align="right"><b>$lt{'st'}:</b></td><td> $displayable_starttime </td></tr>
1786: <tr><td align="right"><b>$lt{'et'}:</b></td><td> $displayable_endtime </td></tr>
1.50 matthew 1787: </table>
1.52 matthew 1788: <p>
1.50 matthew 1789: END
1.53 matthew 1790: #
1.63 matthew 1791: # Send request(s) to modify data (final undef is for 'desiredhost',
1792: # which is a moot point because the student already has an account.
1793: my $modify_section_results = &modifystudent($sdom,$slogin,
1.127 albertel 1794: $env{'request.course.id'},
1.63 matthew 1795: $section,undef);
1796: if ($modify_section_results !~ /^ok/) {
1.94 sakharuk 1797: $r->print(&mt('An error occured during the attempt to change the section for this student.')."<br />");
1.63 matthew 1798: }
1.52 matthew 1799: my $roleresults = &Apache::lonnet::modifystudent
1.53 matthew 1800: ($sdom,$slogin,$sid,undef,undef,$firstname,$middlename,$lastname,
1.127 albertel 1801: $generation,$section,$endtime,$starttime,$env{'form.forceid'});
1.53 matthew 1802: if ($roleresults eq 'refused' ) {
1.94 sakharuk 1803: $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 1804: } elsif ($roleresults !~ /ok/) {
1.94 sakharuk 1805: $r->print(&mt('An error occurred during the attempt to change the role information for this student.')." <br />".
1806: &mt('The error reported was')." ".
1.50 matthew 1807: $roleresults);
1.53 matthew 1808: &Apache::lonnet::logthis("londropadd:failed attempt to modify student".
1809: " data for ".$slogin." \@ ".$sdom." by ".
1.127 albertel 1810: $env{'user.name'}." \@ ".$env{'user.domain'}.
1.53 matthew 1811: ":".$roleresults);
1.50 matthew 1812: } else { # everything is okay!
1.94 sakharuk 1813: $r->print(&mt('Student information updated successfully.')." <br />".
1814: &mt('The student must log out and log in again to see these changes.'));
1.50 matthew 1815: }
1.94 sakharuk 1816: my $Masd=&mt('Modify another students data');
1.50 matthew 1817: $r->print(<<END);
1.52 matthew 1818: </p><p>
1.59 matthew 1819: <input type="hidden" name="action" value="modifystudent" />
1820: <input type="hidden" name="sortby" value="$sortby" />
1821: <input type="hidden" name="Status" value="$status" />
1.94 sakharuk 1822: <a href="javascript:document.studentform.submit();">$Masd</a>
1.50 matthew 1823: END
1.138 ! albertel 1824: $r->print(&Apache::loncommon::end_page());
1.50 matthew 1825: return;
1826: }
1827:
1828: sub get_enrollment_data {
1829: my ($sname,$sdomain) = @_;
1.127 albertel 1830: my $courseid = $env{'request.course.id'};
1.50 matthew 1831: $courseid =~ s:_:/:g;
1832: my %roles = &Apache::lonnet::dump('roles',$sdomain,$sname);
1833: my ($tmp) = keys(%roles);
1834: # Bail out if we were unable to get the students roles
1.87 matthew 1835: return ('error'.$tmp) if ($tmp =~ /^(con_lost|error|no_such_host)/i);
1.50 matthew 1836: # Go through the roles looking for enrollment in this course
1837: my ($end,$start) = (undef,undef);
1838: my $section = '';
1839: my $count = scalar(keys(%roles));
1840: while (my ($course,$role) = each(%roles)) {
1841: if ($course=~ /^\/$courseid\/*\s*(\w+)*_st$/ ) {
1842: #
1843: # Get active role
1844: $section=$1;
1845: (undef,$end,$start)=split(/\_/,$role);
1846: my $now=time;
1847: my $notactive=0;
1848: if ($start) {
1849: if ($now<$start) { $notactive=1; }
1850: }
1851: if ($end) {
1852: if ($now>$end) { $notactive=1; }
1853: }
1854: unless ($notactive) { return ($start,$end,$section); }
1855: }
1856: }
1857: return ($start,$end,$section);
1858: }
1859:
1.56 matthew 1860: #################################################
1861: #################################################
1862:
1863: =pod
1864:
1865: =item show_drop_list
1866:
1867: Display a list of students to drop
1868: Inputs:
1869:
1870: =over 4
1871:
1872: =item $r, Apache request
1873:
1874: =item $classlist, hash pointer returned from loncoursedata::get_classlist();
1875:
1876: =item $keylist, array pointer returned from loncoursedata::get_classlist()
1877: which describes the order elements are stored in the %$classlist values.
1878:
1879: =item $nosort, if true, sorting links are omitted.
1880:
1881: =back
1882:
1883: =cut
1884:
1885: #################################################
1886: #################################################
1.11 www 1887: sub show_drop_list {
1.56 matthew 1888: my ($r,$classlist,$keylist,$nosort)=@_;
1.127 albertel 1889: my $cid=$env{'request.course.id'};
1890: if (! exists($env{'form.sortby'})) {
1.59 matthew 1891: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1892: ['sortby']);
1893: }
1.127 albertel 1894: my $sortby = $env{'form.sortby'};
1.110 matthew 1895: if ($sortby !~ /^(username|domain|section|fullname|id|start|end)$/) {
1.54 matthew 1896: $sortby = 'username';
1897: }
1.56 matthew 1898: #
1.54 matthew 1899: my $action = "drop";
1900: $r->print(<<END);
1901: <input type="hidden" name="sortby" value="$sortby" />
1902: <input type="hidden" name="action" value="$action" />
1.50 matthew 1903: <input type="hidden" name="state" value="done" />
1.32 matthew 1904: <script>
1.51 matthew 1905: function checkAll(field) {
1.32 matthew 1906: for (i = 0; i < field.length; i++)
1907: field[i].checked = true ;
1908: }
1909:
1.51 matthew 1910: function uncheckAll(field) {
1.32 matthew 1911: for (i = 0; i < field.length; i++)
1912: field[i].checked = false ;
1913: }
1914: </script>
1915: <p>
1.26 matthew 1916: <input type="hidden" name="phase" value="four">
1.56 matthew 1917: END
1918:
1.110 matthew 1919: my %lt=&Apache::lonlocal::texthash('usrn' => "username",
1920: 'dom' => "domain",
1921: 'sn' => "student name",
1922: 'sec' => "section",
1923: 'start' => "start date",
1924: 'end' => "end date",
1925: );
1.56 matthew 1926: if ($nosort) {
1927: $r->print(<<END);
1928: <table border=2>
1929: <tr>
1930: <th> </th>
1.94 sakharuk 1931: <th>$lt{'usrn'}</th>
1932: <th>$lt{'dom'}</th>
1.56 matthew 1933: <th>ID</th>
1.94 sakharuk 1934: <th>$lt{'sn'}</th>
1935: <th>$lt{'sec'}</th>
1.110 matthew 1936: <th>$lt{'start'}</th>
1937: <th>$lt{'end'}</th>
1.56 matthew 1938: </tr>
1939: END
1940:
1941: } else {
1942: $r->print(<<END);
1.26 matthew 1943: <table border=2>
1.54 matthew 1944: <tr><th> </th>
1945: <th>
1.94 sakharuk 1946: <a href="/adm/dropadd?action=$action&sortby=username">$lt{'usrn'}</a>
1.54 matthew 1947: </th><th>
1.94 sakharuk 1948: <a href="/adm/dropadd?action=$action&sortby=domain">$lt{'dom'}</a>
1.54 matthew 1949: </th><th>
1950: <a href="/adm/dropadd?action=$action&sortby=id">ID</a>
1951: </th><th>
1.94 sakharuk 1952: <a href="/adm/dropadd?action=$action&sortby=fullname">$lt{'sn'}</a>
1.54 matthew 1953: </th><th>
1.94 sakharuk 1954: <a href="/adm/dropadd?action=$action&sortby=section">$lt{'sec'}</a>
1.110 matthew 1955: </th><th>
1956: <a href="/adm/dropadd?action=$action&sortby=start">$lt{'start'}</a>
1957: </th><th>
1958: <a href="/adm/dropadd?action=$action&sortby=end">$lt{'end'}</a>
1.54 matthew 1959: </th>
1960: </tr>
1.26 matthew 1961: END
1.56 matthew 1962: }
1963: #
1964: # Sort the students
1965: my %index;
1966: my $i;
1967: foreach (@$keylist) {
1968: $index{$_} = $i++;
1969: }
1970: my $index = $index{$sortby};
1971: my $second = $index{'username'};
1972: my $third = $index{'domain'};
1.54 matthew 1973: my @Sorted_Students = sort {
1.56 matthew 1974: lc($classlist->{$a}->[$index]) cmp lc($classlist->{$b}->[$index])
1975: ||
1976: lc($classlist->{$a}->[$second]) cmp lc($classlist->{$b}->[$second])
1977: ||
1978: lc($classlist->{$a}->[$third]) cmp lc($classlist->{$b}->[$third])
1979: } (keys(%$classlist));
1.54 matthew 1980: foreach my $student (@Sorted_Students) {
1.52 matthew 1981: my $error;
1.110 matthew 1982: my $sdata = $classlist->{$student};
1983: my $username = $sdata->[$index{'username'}];
1984: my $domain = $sdata->[$index{'domain'}];
1985: my $section = $sdata->[$index{'section'}];
1986: my $name = $sdata->[$index{'fullname'}];
1987: my $id = $sdata->[$index{'id'}];
1988: my $start = $sdata->[$index{'start'}];
1989: my $end = $sdata->[$index{'end'}];
1990: if (! defined($start) || $start == 0) {
1991: $start = &mt('none');
1992: } else {
1993: $start = &Apache::lonlocal::locallocaltime($start);
1994: }
1995: if (! defined($end) || $end == 0) {
1996: $end = &mt('none');
1997: } else {
1998: $end = &Apache::lonlocal::locallocaltime($end);
1999: }
2000: my $status = $sdata->[$index{'status'}];
1.51 matthew 2001: next if ($status ne 'Active');
2002: #
2003: $r->print(<<"END");
1.26 matthew 2004: <tr>
1.51 matthew 2005: <td><input type="checkbox" name="droplist" value="$student"></td>
2006: <td>$username</td>
2007: <td>$domain</td>
2008: <td>$id</td>
2009: <td>$name</td>
2010: <td>$section</td>
1.110 matthew 2011: <td>$start</td>
2012: <td>$end</td>
1.26 matthew 2013: </tr>
2014: END
1.25 matthew 2015: }
2016: $r->print('</table><br>');
1.111 matthew 2017: %lt=&Apache::lonlocal::texthash(
1.94 sakharuk 2018: 'dp' => "Drop Students",
2019: 'ca' => "check all",
2020: 'ua' => "uncheck all",
2021: );
1.32 matthew 2022: $r->print(<<"END");
2023: </p><p>
1.94 sakharuk 2024: <input type="button" value="$lt{'ca'}" onclick="javascript:checkAll(document.studentform.droplist)">
2025: <input type="button" value="$lt{'ua'}" onclick="javascript:uncheckAll(document.studentform.droplist)">
2026: <p><input type=submit value="$lt{'dp'}"></p>
1.32 matthew 2027: END
1.51 matthew 2028: return;
1.10 www 2029: }
2030:
1.48 matthew 2031: #
2032: # Print out the initial form to get the courselist file
2033: #
2034: sub print_first_courselist_upload_form {
2035: my $r=shift;
1.88 matthew 2036: my $str;
2037: $str = '<input type="hidden" name="phase" value="two">';
2038: $str .= '<input type="hidden" name="action" value="upload" />';
2039: $str .= '<input type="hidden" name="state" value="got_file" />';
2040: $str .= "<h3>".&mt('Upload a class list')."</h3>\n";
2041: $str .= &Apache::loncommon::upfile_select_html();
2042: $str .= "<p>\n";
2043: $str .= '<input type="submit" name="fileupload" value="'.
2044: &mt('Upload class list').'">'."\n";
1.131 albertel 2045: $str .= '<label><input type="checkbox" name="noFirstLine" /> '.
2046: &mt('Ignore First Line')."</label></p>\n";
1.88 matthew 2047: $str .= &Apache::loncommon::help_open_topic("Course_Create_Class_List",
1.92 sakharuk 2048: &mt("How do I create a class list from a spreadsheet")).
1.88 matthew 2049: "<br />\n";
2050: $str .= &Apache::loncommon::help_open_topic("Course_Convert_To_CSV",
1.92 sakharuk 2051: &mt("How do I create a CSV file from a spreadsheet")).
1.88 matthew 2052: "<br />\n";
1.138 ! albertel 2053: $str .= &Apache::loncommon::end_page();
1.88 matthew 2054: $r->print($str);
1.48 matthew 2055: return;
2056: }
2057:
1.10 www 2058: # ================================================= Drop/Add from uploaded file
2059: sub upfile_drop_add {
2060: my $r=shift;
1.24 albertel 2061: &Apache::loncommon::load_tmp_file($r);
2062: my @studentdata=&Apache::loncommon::upfile_record_sep();
1.127 albertel 2063: if($env{'form.noFirstLine'}){shift(@studentdata);}
2064: my @keyfields = split(/\,/,$env{'form.keyfields'});
2065: my $cid = $env{'request.course.id'};
1.25 matthew 2066: my %fields=();
1.127 albertel 2067: for (my $i=0; $i<=$env{'form.nfields'}; $i++) {
2068: if ($env{'form.upfile_associate'} eq 'reverse') {
2069: if ($env{'form.f'.$i} ne 'none') {
2070: $fields{$keyfields[$i]}=$env{'form.f'.$i};
1.25 matthew 2071: }
2072: } else {
1.127 albertel 2073: $fields{$env{'form.f'.$i}}=$keyfields[$i];
1.25 matthew 2074: }
2075: }
1.99 matthew 2076: #
2077: # Store the field choices away
2078: foreach my $field (qw/username names
2079: fname mname lname gen id sec ipwd email/) {
1.127 albertel 2080: $env{'form.'.$field.'_choice'}=$fields{$field};
1.99 matthew 2081: }
2082: &Apache::loncommon::store_course_settings('enrollment_upload',
2083: { 'username_choice' => 'scalar',
2084: 'names_choice' => 'scalar',
2085: 'fname_choice' => 'scalar',
2086: 'mname_choice' => 'scalar',
2087: 'lname_choice' => 'scalar',
2088: 'gen_choice' => 'scalar',
2089: 'id_choice' => 'scalar',
2090: 'sec_choice' => 'scalar',
2091: 'ipwd_choice' => 'scalar',
2092: 'email_choice' => 'scalar' });
2093:
1.26 matthew 2094: #
1.68 matthew 2095: my ($startdate,$enddate) = &get_dates_from_form();
1.127 albertel 2096: if ($env{'form.makedatesdefault'}) {
1.68 matthew 2097: $r->print(&make_dates_default($startdate,$enddate));
2098: }
1.31 matthew 2099: # Determine domain and desired host (home server)
1.127 albertel 2100: my $domain=$env{'form.lcdomain'};
2101: my $desiredhost = $env{'form.lcserver'};
1.31 matthew 2102: if (lc($desiredhost) eq 'default') {
2103: $desiredhost = undef;
2104: } else {
1.45 matthew 2105: my %home_servers = &Apache::loncommon::get_library_servers($domain);
1.31 matthew 2106: if (! exists($home_servers{$desiredhost})) {
1.88 matthew 2107: $r->print('<font color="#ff0000">'.&mt('Error').'</font>'.
2108: &mt('Invalid home server specified'));
1.138 ! albertel 2109: $r->print(&Apache::loncommon::end_page());
1.31 matthew 2110: return;
2111: }
2112: }
1.26 matthew 2113: # Determine authentication mechanism
2114: my $amode = '';
2115: my $genpwd = '';
1.127 albertel 2116: if ($env{'form.login'} eq 'krb') {
1.47 albertel 2117: $amode='krb';
1.127 albertel 2118: $amode.=$env{'form.krbver'};
2119: $genpwd=$env{'form.krbarg'};
2120: } elsif ($env{'form.login'} eq 'int') {
1.25 matthew 2121: $amode='internal';
1.127 albertel 2122: if ((defined($env{'form.intarg'})) && ($env{'form.intarg'})) {
2123: $genpwd=$env{'form.intarg'};
1.25 matthew 2124: }
1.127 albertel 2125: } elsif ($env{'form.login'} eq 'loc') {
1.25 matthew 2126: $amode='localauth';
1.127 albertel 2127: if ((defined($env{'form.locarg'})) && ($env{'form.locarg'})) {
2128: $genpwd=$env{'form.locarg'};
1.79 matthew 2129: }
2130: }
2131: if ($amode =~ /^krb/) {
2132: if (! defined($genpwd) || $genpwd eq '') {
2133: $r->print('<font color="red" size="+1">'.
1.88 matthew 2134: &mt('Unable to enroll students').'</font> '.
2135: &mt('No Kerberos domain was specified.').'</p>');
1.79 matthew 2136: $amode = ''; # This causes the loop below to be skipped
1.25 matthew 2137: }
2138: }
2139: unless (($domain=~/\W/) || ($amode eq '')) {
1.26 matthew 2140: #######################################
2141: ## Enroll Students ##
2142: #######################################
1.88 matthew 2143: $r->print('<h3>'.&mt('Enrolling Students')."</h3>\n<p>\n");
1.25 matthew 2144: my $count=0;
2145: my $flushc=0;
2146: my %student=();
1.26 matthew 2147: # Get new classlist
1.25 matthew 2148: foreach (@studentdata) {
2149: my %entries=&Apache::loncommon::record_sep($_);
1.26 matthew 2150: # Determine student name
1.25 matthew 2151: unless (($entries{$fields{'username'}} eq '') ||
2152: (!defined($entries{$fields{'username'}}))) {
1.26 matthew 2153: my ($fname, $mname, $lname,$gen) = ('','','','');
1.25 matthew 2154: if (defined($fields{'names'})) {
1.26 matthew 2155: ($lname,$fname,$mname)=($entries{$fields{'names'}}=~
2156: /([^\,]+)\,\s*(\w+)\s*(.*)$/);
1.25 matthew 2157: } else {
2158: if (defined($fields{'fname'})) {
2159: $fname=$entries{$fields{'fname'}};
2160: }
2161: if (defined($fields{'mname'})) {
2162: $mname=$entries{$fields{'mname'}};
2163: }
2164: if (defined($fields{'lname'})) {
2165: $lname=$entries{$fields{'lname'}};
2166: }
2167: if (defined($fields{'gen'})) {
2168: $gen=$entries{$fields{'gen'}};
2169: }
2170: }
2171: if ($entries{$fields{'username'}}=~/\W/) {
1.88 matthew 2172: $r->print('<br />'.
2173: &mt('<b>[_1]</b>: Unacceptable username for user [_2] [_3] [_4] [_5]',
2174: $entries{$fields{'username'}},$fname,$mname,$lname,$gen).
2175: '</b>');
1.25 matthew 2176: } else {
1.26 matthew 2177: # determine section number
1.25 matthew 2178: my $sec='';
2179: my $username=$entries{$fields{'username'}};
2180: if (defined($fields{'sec'})) {
2181: if (defined($entries{$fields{'sec'}})) {
2182: $sec=$entries{$fields{'sec'}};
2183: }
2184: }
1.80 matthew 2185: # remove non alphanumeric values from section
2186: $sec =~ s/\W//g;
1.26 matthew 2187: # determine student id number
1.25 matthew 2188: my $id='';
2189: if (defined($fields{'id'})) {
2190: if (defined($entries{$fields{'id'}})) {
2191: $id=$entries{$fields{'id'}};
2192: }
2193: $id=~tr/A-Z/a-z/;
2194: }
1.73 www 2195: # determine email address
2196: my $email='';
2197: if (defined($fields{'email'})) {
2198: if (defined($entries{$fields{'email'}})) {
2199: $email=$entries{$fields{'email'}};
2200: unless ($email=~/^[^\@]+\@[^\@]+$/) { $email=''; }
2201: }
2202: }
1.26 matthew 2203: # determine student password
1.25 matthew 2204: my $password='';
2205: if ($genpwd) {
2206: $password=$genpwd;
2207: } else {
2208: if (defined($fields{'ipwd'})) {
2209: if ($entries{$fields{'ipwd'}}) {
2210: $password=$entries{$fields{'ipwd'}};
2211: }
2212: }
2213: }
1.56 matthew 2214: # Clean up whitespace
2215: foreach (\$domain,\$username,\$id,\$fname,\$mname,
2216: \$lname,\$gen,\$sec) {
2217: $$_ =~ s/(\s+$|^\s+)//g;
2218: }
1.127 albertel 2219: if ($password || $env{'form.login'} eq 'loc') {
1.33 matthew 2220: &modifystudent($domain,$username,$cid,$sec,
2221: $desiredhost);
1.25 matthew 2222: my $reply=&Apache::lonnet::modifystudent
2223: ($domain,$username,$id,$amode,$password,
2224: $fname,$mname,$lname,$gen,$sec,$enddate,
1.127 albertel 2225: $startdate,$env{'form.forceid'},$desiredhost,
1.73 www 2226: $email);
1.26 matthew 2227: if ($reply ne 'ok') {
1.72 matthew 2228: $reply =~ s/^error://;
1.88 matthew 2229: $r->print('<br />'.
2230: &mt('<b>[_1]</b>: Unable to enroll: [_2]',$username,$reply));
1.10 www 2231: } else {
1.7 www 2232: $count++; $flushc++;
2233: $student{$username}=1;
1.6 www 2234: $r->print('. ');
1.7 www 2235: if ($flushc>15) {
2236: $r->rflush;
2237: $flushc=0;
2238: }
1.6 www 2239: }
1.25 matthew 2240: } else {
1.88 matthew 2241: $r->print('<br />'.
2242: &mt('<b>[_1]</b>: Unable to enroll. No password specified.',$username)
2243: );
1.25 matthew 2244: }
2245: }
1.26 matthew 2246: }
2247: } # end of foreach (@studentdata)
1.88 matthew 2248: $r->print("</p>\n<p>\n".&mt('Processed [_1] student(s).',$count).
2249: "</p>\n");
2250: $r->print("<p>\n".
2251: &mt('If active, the new role will be available when the '.
2252: 'students next log in to LON-CAPA.')."</p>\n");
1.26 matthew 2253: #####################################
2254: # Drop students #
2255: #####################################
1.127 albertel 2256: if ($env{'form.fullup'} eq 'yes') {
1.88 matthew 2257: $r->print('<h3>'.&mt('Dropping Students')."</h3>\n");
1.26 matthew 2258: # Get current classlist
1.56 matthew 2259: my ($classlist,$keylist)=&Apache::loncoursedata::get_classlist();
2260: if (! defined($classlist)) {
1.88 matthew 2261: $r->print(&mt('There are no students currently enrolled.').
2262: "\n");
1.56 matthew 2263: } else {
2264: # Remove the students we just added from the list of students.
1.25 matthew 2265: foreach (@studentdata) {
2266: my %entries=&Apache::loncommon::record_sep($_);
2267: unless (($entries{$fields{'username'}} eq '') ||
2268: (!defined($entries{$fields{'username'}}))) {
1.56 matthew 2269: delete($classlist->{$entries{$fields{'username'}}.
1.26 matthew 2270: ':'.$domain});
1.25 matthew 2271: }
2272: }
1.56 matthew 2273: # Print out list of dropped students.
2274: &show_drop_list($r,$classlist,$keylist,'nosort');
1.25 matthew 2275: }
2276: }
1.26 matthew 2277: } # end of unless
1.10 www 2278: }
2279:
1.11 www 2280: # ================================================================== Phase four
2281: sub drop_student_list {
2282: my $r=shift;
2283: my $count=0;
1.128 albertel 2284: my @droplist = &Apache::loncommon::get_env_multiple('form.droplist');
1.35 matthew 2285: foreach (@droplist) {
1.26 matthew 2286: my ($uname,$udom)=split(/\:/,$_);
1.56 matthew 2287: # drop student
1.127 albertel 2288: my $result = &modifystudent($udom,$uname,$env{'request.course.id'});
1.37 matthew 2289: if ($result eq 'ok' || $result eq 'ok:') {
1.88 matthew 2290: $r->print(&mt('Dropped [_1]',$uname.'@'.$udom).'<br>');
1.59 matthew 2291: $count++;
1.35 matthew 2292: } else {
1.88 matthew 2293: $r->print(
2294: &mt('Error dropping [_1]:[_2]',$uname.'@'.$udom,$result).
1.35 matthew 2295: '<br />');
2296: }
1.20 harris41 2297: }
1.88 matthew 2298: $r->print('<p><b>'.&mt('Dropped [_1] student(s).',$count).'</b></p>');
2299: $r->print('<p>'.&mt('Re-enrollment will re-activate data.')) if ($count);
1.11 www 2300: }
2301:
1.50 matthew 2302: ###################################################################
2303: ###################################################################
2304:
2305: =pod
2306:
2307: =item &handler
2308:
2309: The typical handler you see in all these modules. Takes $r, the
2310: http request, as an argument.
2311:
2312: The response to the request is governed by two form variables
2313:
2314: form.action form.state response
2315: ---------------------------------------------------
2316: undefined undefined print main menu
2317: upload undefined print courselist upload menu
2318: upload got_file deal with uploaded file,
2319: print the upload managing menu
2320: upload enrolling enroll students based on upload
2321: drop undefined print the classlist ready to drop
2322: drop done drop the selected students
1.74 matthew 2323: enrollstudent undefined print student username domain form
2324: enrollstudent gotusername print single student enroll menu
1.50 matthew 2325: enrollstudent enrolling enroll student
2326: classlist undefined print html classlist
2327: classlist csv print csv classlist
2328: modifystudent undefined print classlist to select student to modify
2329: modifystudent selected print modify student menu
2330: modifystudent done make modifications to student record
2331:
2332: =cut
2333:
2334: ###################################################################
2335: ###################################################################
1.10 www 2336: sub handler {
1.26 matthew 2337: my $r=shift;
2338: if ($r->header_only) {
1.86 www 2339: &Apache::loncommon::content_type($r,'text/html');
1.26 matthew 2340: $r->send_http_header;
2341: return OK;
2342: }
1.48 matthew 2343: &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1.50 matthew 2344: ['action','state']);
1.102 matthew 2345:
2346: &Apache::lonhtmlcommon::clear_breadcrumbs();
2347: &Apache::lonhtmlcommon::add_breadcrumb
2348: ({href=>"/adm/dropadd",
2349: text=>"Enrollment Manager",
2350: faq=>9,bug=>'Instructor Interface',});
1.26 matthew 2351: # Needs to be in a course
1.127 albertel 2352: if (! ($env{'request.course.fn'})) {
1.121 matthew 2353: # Not in a course
1.127 albertel 2354: $env{'user.error.msg'}=
1.132 raeburn 2355: "/adm/dropadd:cst:0:0:Cannot manage or view course groups, ".
2356: "or drop or add students";
1.50 matthew 2357: return HTTP_NOT_ACCEPTABLE;
2358: }
2359: #
1.121 matthew 2360: my $view_permission =
1.127 albertel 2361: &Apache::lonnet::allowed('vcl',$env{'request.course.id'});
1.121 matthew 2362: my $enrl_permission =
1.127 albertel 2363: &Apache::lonnet::allowed('cst',$env{'request.course.id'});
1.132 raeburn 2364:
2365: my $grp_view_permission =
2366: &Apache::lonnet::allowed('vcg',$env{'request.course.id'});
2367: my $grp_manage_permission =
2368: &Apache::lonnet::allowed('mdg',$env{'request.course.id'});
2369:
2370:
2371: if (! $grp_view_permission && ! $grp_manage_permission &&
2372: ! $view_permission && ! $enrl_permission) {
1.127 albertel 2373: $env{'user.error.msg'}=
1.132 raeburn 2374: "/adm/coursegroups:cst:0:0:Cannot manage or view course groups, ".
2375: "or drop or add students";
2376: return HTTP_NOT_ACCEPTABLE;
1.121 matthew 2377: }
1.132 raeburn 2378:
1.121 matthew 2379: #
1.50 matthew 2380: # Only output the header information if they did not request csv format
2381: #
1.103 matthew 2382: # Start page
2383: &Apache::loncommon::content_type($r,'text/html');
2384: $r->send_http_header;
2385: $r->print(&header());
1.50 matthew 2386: #
2387: # Main switch on form.action and form.state, as appropriate
1.127 albertel 2388: if (! exists($env{'form.action'})) {
1.102 matthew 2389: $r->print(&Apache::lonhtmlcommon::breadcrumbs
2390: (undef,'Enrollment Manager'));
1.132 raeburn 2391: &print_main_menu($r,$enrl_permission,$view_permission,$grp_manage_permission,
2392: $grp_view_permission);
1.127 albertel 2393: } elsif ($env{'form.action'} eq 'upload' && $enrl_permission) {
1.102 matthew 2394: &Apache::lonhtmlcommon::add_breadcrumb
2395: ({href=>'/adm/dropadd?action=upload&state=',
1.106 matthew 2396: text=>"Upload Classlist"});
1.102 matthew 2397: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.117 albertel 2398: (undef,'Upload Classlist','Course_Create_Class_List'));
1.127 albertel 2399: if (! exists($env{'form.state'})) {
1.50 matthew 2400: &print_first_courselist_upload_form($r);
1.127 albertel 2401: } elsif ($env{'form.state'} eq 'got_file') {
1.50 matthew 2402: &print_upload_manager_form($r);
1.127 albertel 2403: } elsif ($env{'form.state'} eq 'enrolling') {
2404: if ($env{'form.datatoken'}) {
1.26 matthew 2405: &upfile_drop_add($r);
1.50 matthew 2406: } else {
2407: # Hmmm, this is an error
1.26 matthew 2408: }
1.50 matthew 2409: } else {
2410: &print_first_courselist_upload_form($r);
1.26 matthew 2411: }
1.127 albertel 2412: } elsif ($env{'form.action'} eq 'drop' && $enrl_permission) {
1.102 matthew 2413: &Apache::lonhtmlcommon::add_breadcrumb
2414: ({href=>'/adm/dropadd?action=drop',
1.106 matthew 2415: text=>"Drop Students"});
1.102 matthew 2416: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.117 albertel 2417: (undef,'Drop Students','Course_Drop_Student'));
1.127 albertel 2418: if (! exists($env{'form.state'})) {
1.51 matthew 2419: &print_drop_menu($r);
1.127 albertel 2420: } elsif ($env{'form.state'} eq 'done') {
1.26 matthew 2421: &drop_student_list($r);
1.50 matthew 2422: } else {
1.55 matthew 2423: &print_drop_menu($r);
1.26 matthew 2424: }
1.127 albertel 2425: } elsif ($env{'form.action'} eq 'enrollstudent' && $enrl_permission) {
1.102 matthew 2426: &Apache::lonhtmlcommon::add_breadcrumb
2427: ({href=>'/adm/dropadd?action=enrollstudent',
1.106 matthew 2428: text=>"Enroll Student"});
1.102 matthew 2429: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.117 albertel 2430: (undef,'Enroll Student','Course_Add_Student'));
1.127 albertel 2431: if (! exists($env{'form.state'})) {
1.74 matthew 2432: &get_student_username_domain_form($r);
1.127 albertel 2433: } elsif ($env{'form.state'} eq 'gotusername') {
1.50 matthew 2434: &print_enroll_single_student_form($r);
1.127 albertel 2435: } elsif ($env{'form.state'} eq 'enrolling') {
1.26 matthew 2436: &enroll_single_student($r);
1.50 matthew 2437: } else {
1.74 matthew 2438: &get_student_username_domain_form($r);
1.26 matthew 2439: }
1.127 albertel 2440: } elsif ($env{'form.action'} eq 'classlist' && $view_permission) {
1.102 matthew 2441: &Apache::lonhtmlcommon::add_breadcrumb
2442: ({href=>'/adm/dropadd?action=classlist',
1.106 matthew 2443: text=>"View Classlist"});
1.102 matthew 2444: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.117 albertel 2445: (undef,'View Classlist','Course_View_Class_List'));
1.127 albertel 2446: if (! exists($env{'form.state'})) {
1.103 matthew 2447: &print_html_classlist($r,undef);
1.127 albertel 2448: } elsif ($env{'form.state'} eq 'csv') {
1.103 matthew 2449: &print_html_classlist($r,'csv');
1.127 albertel 2450: } elsif ($env{'form.state'} eq 'excel') {
1.103 matthew 2451: &print_html_classlist($r,'excel');
1.50 matthew 2452: } else {
1.103 matthew 2453: &print_html_classlist($r,undef);
1.50 matthew 2454: }
1.127 albertel 2455: } elsif ($env{'form.action'} eq 'modifystudent' && $enrl_permission) {
1.102 matthew 2456: &Apache::lonhtmlcommon::add_breadcrumb
2457: ({href=>'/adm/dropadd?action=modifystudent',
1.106 matthew 2458: text=>"Modify Student Data"});
1.102 matthew 2459: $r->print(&Apache::lonhtmlcommon::breadcrumbs
1.117 albertel 2460: (undef,'Modify Student Data','Course_Modify_Student_Data'));
1.127 albertel 2461: if (! exists($env{'form.state'})) {
1.50 matthew 2462: &print_html_classlist($r);
1.127 albertel 2463: } elsif ($env{'form.state'} eq 'selected') {
1.50 matthew 2464: &print_modify_student_form($r);
1.127 albertel 2465: } elsif ($env{'form.state'} eq 'done') {
1.50 matthew 2466: &modify_single_student($r);
2467: } else {
2468: &print_html_classlist($r);
2469: }
2470: } else {
2471: # We should not end up here, but I guess it is possible
2472: &Apache::lonnet::logthis("Undetermined state in londropadd.pm. ".
1.127 albertel 2473: "form.action = ".$env{'form.action'}.
1.50 matthew 2474: "Someone should fix this.");
1.102 matthew 2475: $r->print(&Apache::lonhtmlcommon::breadcrumbs
2476: (undef,'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>