--- loncom/interface/lonmodifycourse.pm 2009/07/31 03:01:31 1.46 +++ loncom/interface/lonmodifycourse.pm 2019/07/25 20:23:52 1.95 @@ -1,7 +1,7 @@ # The LearningOnline Network with CAPA # handler for DC-only modifiable course settings # -# $Id: lonmodifycourse.pm,v 1.46 2009/07/31 03:01:31 raeburn Exp $ +# $Id: lonmodifycourse.pm,v 1.95 2019/07/25 20:23:52 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -34,24 +34,65 @@ use Apache::loncommon; use Apache::lonhtmlcommon; use Apache::lonlocal; use Apache::lonuserutils; +use Apache::loncreateuser; use Apache::lonpickcourse; -use LONCAPA::Enrollment; use lib '/home/httpd/lib/perl'; -use LONCAPA; +use LONCAPA qw(:DEFAULT :match); + +my $registered_cleanup; +my $modified_dom; sub get_dc_settable { - return ('courseowner','coursecode','authtype','autharg'); + my ($type,$cdom) = @_; + if ($type eq 'Community') { + return ('courseowner','selfenrollmgrdc','selfenrollmgrcc'); + } else { + my @items = ('courseowner','coursecode','authtype','autharg','selfenrollmgrdc', + 'selfenrollmgrcc','mysqltables'); + if (&showcredits($cdom)) { + push(@items,'defaultcredits'); + } + my %passwdconf = &Apache::lonnet::get_passwdconf($cdom); + if (($passwdconf{'crsownerchg'}) && ($type ne 'Placement')) { + push(@items,'nopasswdchg'); + } + return @items; + } +} + +sub autoenroll_keys { + my $internals = ['coursecode','courseowner','authtype','autharg','defaultcredits', + 'autoadds','autodrops','autostart','autoend','sectionnums', + 'crosslistings','co-owners','autodropfailsafe']; + my $accessdates = ['default_enrollment_start_date','default_enrollment_end_date']; + return ($internals,$accessdates); } sub catalog_settable { - my ($confhash) = @_; + my ($confhash,$type) = @_; my @settable; if (ref($confhash) eq 'HASH') { - if ($confhash->{'togglecats'} ne 'crs') { - push(@settable,'togglecats'); - } - if ($confhash->{'categorize'} ne 'crs') { - push(@settable,'categorize'); + if ($type eq 'Community') { + if ($confhash->{'togglecatscomm'} ne 'comm') { + push(@settable,'togglecats'); + } + if ($confhash->{'categorizecomm'} ne 'comm') { + push(@settable,'categorize'); + } + } elsif ($type eq 'Placement') { + if ($confhash->{'togglecatsplace'} ne 'place') { + push(@settable,'togglecats'); + } + if ($confhash->{'categorizeplace'} ne 'place') { + push(@settable,'categorize'); + } + } else { + if ($confhash->{'togglecats'} ne 'crs') { + push(@settable,'togglecats'); + } + if ($confhash->{'categorize'} ne 'crs') { + push(@settable,'categorize'); + } } } else { push(@settable,('togglecats','categorize')); @@ -61,14 +102,18 @@ sub catalog_settable { sub get_enrollment_settings { my ($cdom,$cnum) = @_; - my %settings = &Apache::lonnet::dump('environment',$cdom,$cnum); + my ($internals,$accessdates) = &autoenroll_keys(); + my @items; + if ((ref($internals) eq 'ARRAY') && (ref($accessdates) eq 'ARRAY')) { + @items = map { 'internal.'.$_; } (@{$internals}); + push(@items,@{$accessdates}); + } + push(@items,'internal.nopasswdchg'); + my %settings = &Apache::lonnet::get('environment',\@items,$cdom,$cnum); my %enrollvar; $enrollvar{'autharg'} = ''; $enrollvar{'authtype'} = ''; - my %lt=&Apache::lonlocal::texthash( - 'noen' => "No end date", - ); - foreach my $item (keys %settings) { + foreach my $item (keys(%settings)) { if ($item =~ m/^internal\.(.+)$/) { my $type = $1; if ( ($type eq "autoadds") || ($type eq "autodrops") ) { @@ -79,17 +124,22 @@ sub get_enrollment_settings { } } elsif ( ($type eq "autostart") || ($type eq "autoend") ) { if ( ($type eq "autoend") && ($settings{$item} == 0) ) { - $enrollvar{$type} = $lt{'noen'}; + $enrollvar{$type} = &mt('No end date'); } else { - $enrollvar{$type} = localtime($settings{$item}); + $enrollvar{$type} = &Apache::lonlocal::locallocaltime($settings{$item}); } - } elsif ($type eq "sectionnums") { + } elsif (($type eq 'sectionnums') || ($type eq 'co-owners')) { $enrollvar{$type} = $settings{$item}; $enrollvar{$type} =~ s/,/, /g; } elsif ($type eq "authtype" || $type eq "autharg" || $type eq "coursecode" - || $type eq "crosslistings") { + || $type eq "crosslistings" || $type eq "selfenrollmgr" + || $type eq "autodropfailsafe" || $type eq 'nopasswdchg') { $enrollvar{$type} = $settings{$item}; + } elsif ($type eq 'defaultcredits') { + if (&showcredits($cdom)) { + $enrollvar{$type} = $settings{$item}; + } } elsif ($type eq 'courseowner') { if ($settings{$item} =~ /^[^:]+:[^:]+$/) { $enrollvar{$type} = $settings{$item}; @@ -102,11 +152,11 @@ sub get_enrollment_settings { } elsif ($item =~ m/^default_enrollment_(start|end)_date$/) { my $type = $1; if ( ($type eq 'end') && ($settings{$item} == 0) ) { - $enrollvar{$item} = $lt{'noen'}; + $enrollvar{$item} = &mt('No end date'); } elsif ( ($type eq 'start') && ($settings{$item} eq '') ) { $enrollvar{$item} = 'When enrolled'; } else { - $enrollvar{$item} = localtime($settings{$item}); + $enrollvar{$item} = &Apache::lonlocal::locallocaltime($settings{$item}); } } } @@ -115,224 +165,621 @@ sub get_enrollment_settings { sub print_course_search_page { my ($r,$dom,$domdesc) = @_; - &print_header($r); - my $filterlist = ['descriptfilter', - 'instcodefilter','ownerfilter', - 'coursefilter']; - my $filter = {}; - my ($numtitles,$type,$cctitle,$dctitle); - $type = $env{'form.type'}; + my $action = '/adm/modifycourse'; + my $type = $env{'form.type'}; if (!defined($env{'form.type'})) { $type = 'Course'; } - my $action = '/adm/modifycourse'; - $cctitle = &Apache::lonnet::plaintext('cc',$type); + &print_header($r,$type); + my ($filterlist,$filter) = &get_filters($dom); + my ($numtitles,$cctitle,$dctitle,@codetitles); + my $ccrole = 'cc'; + if ($type eq 'Community') { + $ccrole = 'co'; + } + $cctitle = &Apache::lonnet::plaintext($ccrole,$type); $dctitle = &Apache::lonnet::plaintext('dc'); - $r->print(&Apache::lonpickcourse::js_changer()); - if ($type eq 'Course') { - $r->print( - '

'.&mt('Search for a course in the [_1] domain',$domdesc).'

'. - &mt('Actions available after searching for a course:').''); - } else { - $r->print( - '

'.&mt('Search for a community in the [_1] domain',$domdesc).'

'. - &mt('Actions available after searching for a community:').''); - } - $r->print(&Apache::lonpickcourse::build_filters($filterlist,$type, - undef,undef,$filter,$action,\$numtitles,'modifycourse')); + $r->print(&Apache::loncommon::js_changer()); + if ($type eq 'Community') { + $r->print('

'.&mt('Search for a community in the [_1] domain',$domdesc).'

'); + } elsif ($type eq 'Placement') { + $r->print('

'.&mt('Search for a placement test in the [_1] domain',$domdesc).'

'); + } else { + $r->print('

'.&mt('Search for a course in the [_1] domain',$domdesc).'

'); + } + $r->print(&Apache::loncommon::build_filters($filterlist,$type,undef,undef,$filter,$action, + \$numtitles,'modifycourse',undef,undef,undef, + \@codetitles,$dom)); + + my ($actiontext,$roleoption,$settingsoption); + if ($type eq 'Community') { + $actiontext = &mt('Actions available after searching for a community:'); + } elsif ($type eq 'Placement') { + $actiontext = &mt('Actions available after searching for a placement test:') + } else { + $actiontext = &mt('Actions available after searching for a course:'); + } + if (&Apache::lonnet::allowed('ccc',$dom)) { + if ($type eq 'Community') { + $roleoption = &mt('Enter the community with the role of [_1]',$cctitle); + $settingsoption = &mt('View or modify community settings which only a [_1] may modify.',$dctitle); + } elsif ($type eq 'Placement') { + $roleoption = &mt('Enter the placement test with the role of [_1]',$cctitle); + $settingsoption = &mt('View or modify placement test settings which only a [_1] may modify.',$dctitle); + } else { + $roleoption = &mt('Enter the course with the role of [_1]',$cctitle); + $settingsoption = &mt('View or modify course settings which only a [_1] may modify.',$dctitle); + } + } elsif (&Apache::lonnet::allowed('rar',$dom)) { + my ($roles_by_num,$description,$accessref,$accessinfo) = &Apache::lonnet::get_all_adhocroles($dom); + if ((ref($roles_by_num) eq 'ARRAY') && (ref($description) eq 'HASH')) { + if (@{$roles_by_num} > 1) { + if ($type eq 'Community') { + $roleoption = &mt('Enter the community with one of the available ad hoc roles'); + } elsif ($type eq 'Placement') { + $roleoption = &mt('Enter the placement test with one of the available ad hoc roles.'); + } else { + $roleoption = &mt('Enter the course with one of the available ad hoc roles.'); + } + } else { + my $rolename = $description->{$roles_by_num->[0]}; + if ($type eq 'Community') { + $roleoption = &mt('Enter the community with the ad hoc role of: [_1]',$rolename); + } elsif ($type eq 'Placement') { + $roleoption = &mt('Enter the placement test with the ad hoc role of: [_1]',$rolename); + } else { + $roleoption = &mt('Enter the course with the ad hoc role of: [_1]',$rolename); + } + } + } + if ($type eq 'Community') { + $settingsoption = &mt('View community settings which only a [_1] may modify.',$dctitle); + } elsif ($type eq 'Placement') { + $settingsoption = &mt('View placement test settings which only a [_1] may modify.',$dctitle); + } else { + $settingsoption = &mt('View course settings which only a [_1] may modify.',$dctitle); + } + } + $r->print($actiontext.''); + return; } sub print_course_selection_page { - my ($r,$dom,$domdesc) = @_; - &print_header($r); - -# Criteria for course search - my $filterlist = ['descriptfilter', - 'instcodefilter','ownerfilter', - 'ownerdomfilter','coursefilter']; - my %filter; + my ($r,$dom,$domdesc,$permission) = @_; my $type = $env{'form.type'}; if (!defined($type)) { $type = 'Course'; } + &print_header($r,$type); + + if ($permission->{'adhocrole'} eq 'custom') { + my %lt = &Apache::lonlocal::texthash( + title => 'Ad hoc role selection', + preamble => 'Please choose an ad hoc role in the course.', + cancel => 'Click "OK" to enter the course, or "Cancel" to choose a different course.', + ); + my %jslt = &Apache::lonlocal::texthash ( + none => 'You are not eligible to use an ad hoc role for the selected course', + ok => 'OK', + exit => 'Cancel', + ); + &js_escape(\%jslt); + $r->print(<<"END"); + + + +END + } elsif ($permission->{'adhocrole'} eq 'coord') { + $r->print(<<"END"); + +END + } + +# Criteria for course search + my ($filterlist,$filter) = &get_filters(); my $action = '/adm/modifycourse'; my $dctitle = &Apache::lonnet::plaintext('dc'); - my $numtitles; - $r->print(&Apache::lonpickcourse::js_changer()); - $r->print(&mt('Revise your search criteria for this domain').' ('.$domdesc.').

'); - $r->print(&Apache::lonpickcourse::build_filters($filterlist,$type, - undef,undef,\%filter,$action,\$numtitles)); - $filter{'domainfilter'} = $dom; - my %courses = &Apache::lonpickcourse::search_courses($r,$type,0, - \%filter,$numtitles); + my ($numtitles,@codetitles); + $r->print(&Apache::loncommon::js_changer()); + $r->print(&mt('Revise your search criteria for this domain').' ('.$domdesc.').
'); + $r->print(&Apache::loncommon::build_filters($filterlist,$type,undef,undef,$filter,$action, + \$numtitles,'modifycourse',undef,undef,undef, + \@codetitles,$dom,$env{'form.form'})); + my %courses = &Apache::loncommon::search_courses($dom,$type,$filter,$numtitles, + undef,undef,undef,\@codetitles); &Apache::lonpickcourse::display_matched_courses($r,$type,0,$action,undef,undef,undef, - %courses); + $dom,undef,%courses); return; } +sub get_filters { + my ($dom) = @_; + my @filterlist = ('descriptfilter','instcodefilter','ownerfilter', + 'ownerdomfilter','coursefilter','sincefilter'); + # created filter + my $loncaparev = &Apache::lonnet::get_server_loncaparev($dom); + if ($loncaparev ne 'unknown_cmd') { + push(@filterlist,'createdfilter'); + } + my %filter; + foreach my $item (@filterlist) { + $filter{$item} = $env{'form.'.$item}; + } + return (\@filterlist,\%filter); +} + sub print_modification_menu { - my ($r,$cdesc,$domdesc,$dom) = @_; - &print_header($r,$cdesc); - my $type = 'Course'; - my $action = '/adm/modifycourse'; - my $cctitle = &Apache::lonnet::plaintext('cc',$type); - my $dctitle = &Apache::lonnet::plaintext('dc'); - my %lt=&Apache::lonlocal::texthash( - 'some' => "Certain settings which control auto-enrollment of students from your institution's student information system.", - 'crqo' => 'The total disk space allocated for storage of portfolio files in all groups in a course.', - ); + my ($r,$cdesc,$domdesc,$dom,$type,$cid,$coursehash,$permission) = @_; + &print_header($r,$type); + my ($ccrole,$categorytitle,$setquota_text,$setuploadquota_text,$cdom,$cnum); + if (ref($coursehash) eq 'HASH') { + $cdom = $coursehash->{'domain'}; + $cnum = $coursehash->{'num'}; + } else { + ($cdom,$cnum) = split(/_/,$cid); + } + if ($type eq 'Community') { + $ccrole = 'co'; + } else { + $ccrole = 'cc'; + } + my %linktext; + if ($permission->{'setparms'} eq 'edit') { + %linktext = ( + 'setquota' => 'View/Modify quotas for group portfolio files, and for uploaded content', + 'setanon' => 'View/Modify responders threshold for anonymous survey submissions display', + 'selfenroll' => 'View/Modify Self-Enrollment configuration', + 'setpostsubmit' => 'View/Modify submit button behavior, post-submission', + ); + } else { + %linktext = ( + 'setquota' => 'View quotas for group portfolio files, and for uploaded content', + 'setanon' => 'View responders threshold for anonymous survey submissions display', + 'selfenroll' => 'View Self-Enrollment configuration', + 'setpostsubmit' => 'View submit button behavior, post-submission', + ); + } + if ($type eq 'Community') { + if ($permission->{'setparms'} eq 'edit') { + $categorytitle = 'View/Modify Community Settings'; + $linktext{'setparms'} = 'View/Modify community owner'; + $linktext{'catsettings'} = 'View/Modify catalog settings for community'; + } else { + $categorytitle = 'View Community Settings'; + $linktext{'setparms'} = 'View community owner'; + $linktext{'catsettings'} = 'View catalog settings for community'; + } + $setquota_text = &mt('Total disk space allocated for storage of portfolio files in all groups in a community.'); + $setuploadquota_text = &mt('Disk space allocated for storage of content uploaded directly to a community via Content Editor.'); + } else { + if ($permission->{'setparms'} eq 'edit') { + $categorytitle = 'View/Modify Course Settings'; + $linktext{'catsettings'} = 'View/Modify catalog settings for course'; + if (($type ne 'Placement') && (&showcredits($dom))) { + $linktext{'setparms'} = 'View/Modify course owner, institutional code, default authentication, credits, self-enrollment and table lifetime'; + } else { + $linktext{'setparms'} = 'View/Modify course owner, institutional code, default authentication, self-enrollment and table lifetime'; + } + } else { + $categorytitle = 'View Course Settings'; + $linktext{'catsettings'} = 'View catalog settings for course'; + if (($type ne 'Placement') && (&showcredits($dom))) { + $linktext{'setparms'} = 'View course owner, institutional code, default authentication, credits, self-enrollment and table lifetime'; + } else { + $linktext{'setparms'} = 'View course owner, institutional code, default authentication, self-enrollment and table lifetime'; + } + } + $setquota_text = &mt('Total disk space allocated for storage of portfolio files in all groups in a course.'); + $setuploadquota_text = &mt('Disk space allocated for storage of content uploaded directly to a course via Content Editor.'); + } + my $anon_text = &mt('Responder threshold required to display anonymous survey submissions.'); + my $postsubmit_text = &mt('Override defaults for submit button behavior post-submission for this specific course.'); + my $mysqltables_text = &mt('Override default for lifetime of "temporary" MySQL tables containing student performance data.'); + $linktext{'viewparms'} = 'Display current settings for automated enrollment'; + + my %domconf = &Apache::lonnet::get_dom('configuration',['coursecategories'],$dom); + my @additional_params = &catalog_settable($domconf{'coursecategories'},$type); + + sub manage_selfenrollment { + my ($cdom,$cnum,$type,$coursehash,$permission) = @_; + if ($permission->{'selfenroll'}) { + my ($managed_by_cc,$managed_by_dc) = &Apache::lonuserutils::selfenrollment_administration($cdom,$cnum,$type,$coursehash); + if (ref($managed_by_dc) eq 'ARRAY') { + if (@{$managed_by_dc}) { + return 1; + } + } + } + return 0; + } + + sub phaseurl { + my $phase = shift; + return "javascript:changePage(document.menu,'$phase')" + } my @menu = - ( - { text => 'Modify quota for group portfolio', - phase => 'setquota', + ({ categorytitle => $categorytitle, + items => [ + { + linktext => $linktext{'setparms'}, + url => &phaseurl('setparms'), + permission => $permission->{'setparms'}, + #help => '', + icon => 'crsconf.png', + linktitle => '' }, - { text => 'Display current settings for automated enrollment', - phase => 'viewparms', + { + linktext => $linktext{'setquota'}, + url => &phaseurl('setquota'), + permission => $permission->{'setquota'}, + #help => '', + icon => 'groupportfolioquota.png', + linktitle => '' }, - { text => 'Modify institutional code, course owner and/or default authentication', - phase => 'setparms', - }, - ); - my %domconf = &Apache::lonnet::get_dom('configuration',['coursecategories'],$dom); - my @additional_params = &catalog_settable($domconf{'coursecategories'}); - if (@additional_params > 0) { - push (@menu, { text => 'Modify course catalog settings for course', - phase => 'catsettings', - }); - } - my $menu_html = '

'.&mt('View/Modify settings for: ').$cdesc.'

'."\n". - &mt('Although almost all course settings can be modified by a [_1], a number of settings exist which only a [_2] may change:',$cctitle,$dctitle).' -' + .'' + ."\n" + .&hidden_form_elements(); $r->print($menu_html); + $r->print(&Apache::lonhtmlcommon::generate_menu(@menu)); + $r->print(''); return; } -sub print_ccrole_selected { - my ($r,$cdesc,$domdesc) = @_; - &print_header($r); +sub print_adhocrole_selected { + my ($r,$type,$permission) = @_; + &print_header($r,$type); my ($cdom,$cnum) = split(/_/,$env{'form.pickedcourse'}); - $r->print('
- - + my ($newrole,$selectrole); + if ($permission->{'adhocrole'} eq 'coord') { + if ($type eq 'Community') { + $newrole = "co./$cdom/$cnum"; + } else { + $newrole = "cc./$cdom/$cnum"; + } + $selectrole = 1; + } elsif ($permission->{'adhocrole'} eq 'custom') { + my ($okroles,$description) = &Apache::lonnet::get_my_adhocroles($env{'form.pickedcourse'},1); + if (ref($okroles) eq 'ARRAY') { + my $possrole = $env{'form.adhocrole'}; + if (($possrole ne '') && (grep(/^\Q$possrole\E$/,@{$okroles}))) { + my $confname = &Apache::lonnet::get_domainconfiguser($cdom); + $newrole = "cr/$cdom/$confname/$possrole./$cdom/$cnum"; + $selectrole = 1; + } + } + } + if ($selectrole) { + $r->print(' + +
'); + } else { + $r->print('
'. + '
'); + } + return; } sub print_settings_display { - my ($r,$cdom,$cnum,$cdesc,$type) = @_; + my ($r,$cdom,$cnum,$cdesc,$type,$permission) = @_; my %enrollvar = &get_enrollment_settings($cdom,$cnum); - my %longtype = &course_settings_descrip(); + my %longtype = &course_settings_descrip($type); my %lt = &Apache::lonlocal::texthash( - 'cset' => "Course setting", - 'valu' => "Current value", - 'caes' => 'Current automated enrollment settings for ', - 'cour' => "Course settings that control automated enrollment in this LON-CAPA course are currently:", - 'cose' => "Course settings for LON-CAPA courses that control auto-enrollment based on classlist data available from your institution's student information system fall into two groups:", - 'dcon' => "Modifiable by DC only", - 'back' => "Back to options page", + 'valu' => 'Current value', + 'cour' => 'Current settings are:', + 'cose' => "Settings which control auto-enrollment using classlists from your institution's student information system fall into two groups:", + 'dcon' => 'Modifiable only by Domain Coordinator', + 'back' => 'Pick another action', ); - - my $cctitle = &Apache::lonnet::plaintext('cc',$type); + my $ccrole = 'cc'; + if ($type eq 'Community') { + $ccrole = 'co'; + } + my $cctitle = &Apache::lonnet::plaintext($ccrole,$type); my $dctitle = &Apache::lonnet::plaintext('dc'); - my @modifiable_params = &get_dc_settable(); - + my @modifiable_params = &get_dc_settable($type,$cdom); + my ($internals,$accessdates) = &autoenroll_keys(); + my @items; + if ((ref($internals) eq 'ARRAY') && (ref($accessdates) eq 'ARRAY')) { + @items = (@{$internals},@{$accessdates}); + } my $disp_table = &Apache::loncommon::start_data_table()."\n". &Apache::loncommon::start_data_table_header_row()."\n". - "$lt{'cset'}\n". + " \n". "$lt{'valu'}\n". "$lt{'dcon'}\n". &Apache::loncommon::end_data_table_header_row()."\n"; - foreach my $key (sort(keys(%enrollvar))) { + foreach my $item (@items) { $disp_table .= &Apache::loncommon::start_data_table_row()."\n". - "$longtype{$key}\n". - "$enrollvar{$key}\n"; - if (grep(/^\Q$key\E$/,@modifiable_params)) { - $disp_table .= ''.&mt('Yes').''."\n"; + "$longtype{$item}\n". + "$enrollvar{$item}\n"; + if (grep(/^\Q$item\E$/,@modifiable_params)) { + $disp_table .= ''.&mt('Yes').''."\n"; } else { - $disp_table .= ''.&mt('No').''."\n"; + $disp_table .= ''.&mt('No').''."\n"; } $disp_table .= &Apache::loncommon::end_data_table_row()."\n"; } $disp_table .= &Apache::loncommon::end_data_table()."\n"; - &print_header($r,$cdesc); - $r->print(' -

'.$lt{'caes'}.$cdesc.'

-
-

-'.$lt{'cose'}.' -

-

-'.$lt{'cour'}.' -

-'.$disp_table.' -

-'.$lt{'back'}.'     -'.&mt('Modify [_1]-only settings',$dctitle).''."\n". -&hidden_form_elements(). -'

'); + &print_header($r,$type); + my ($enroll_link_start,$enroll_link_end,$setparms_link_start,$setparms_link_end); + if (&Apache::lonnet::allowed('ccc',$cdom)) { + my $newrole = $ccrole.'./'.$cdom.'/'.$cnum; + my $escuri = &HTML::Entities::encode('/adm/roles?selectrole=1&'.$newrole. + '=1&destinationurl=/adm/populate','&<>"'); + $enroll_link_start = ''; + $enroll_link_end = ''; + } + if ($permission->{'setparms'}) { + $setparms_link_start = ''; + $setparms_link_end = ''; + } + $r->print('

'.&mt('Current automated enrollment settings for:'). + ' '.$cdesc.'

'. + '
'."\n". + '

'.$lt{'cose'}.'

'. + '

'.$lt{'cour'}.'

'.$disp_table.'

'. + ''.$lt{'back'}.''."\n". + &hidden_form_elements(). + '

' + ); } sub print_setquota { - my ($r,$cdom,$cnum,$cdesc,$type) = @_; - my $dctitle = &Apache::lonnet::plaintext('dc'); - my $cctitle = &Apache::lonnet::plaintext('cc',$type); - my $subdiv = &mt('Although a [_1] will assign the disk quota for each individual group, the size of the quota is constrained by the total disk space allocated by the [_2] for portfolio files in a course.',$cctitle,$dctitle); + my ($r,$cdom,$cnum,$cdesc,$type,$readonly) = @_; + my $lctype = lc($type); + my $headline = &mt("Set disk space quotas for $lctype: [_1]", + ''.$cdesc.''); my %lt = &Apache::lonlocal::texthash( - 'cquo' => 'Disk space for storage of group portfolio', - 'gpqu' => 'Course portfolio files disk space', - 'each' => 'Each course group can be assigned a quota for portfolio files uploaded to the group.', + 'gpqu' => 'Disk space for storage of group portfolio files', + 'upqu' => 'Disk space for storage of content directly uploaded to course via Content Editor', 'modi' => 'Save', - 'back' => "Back to options page", + 'back' => 'Pick another action', ); - my %settings = &Apache::lonnet::get('environment',['internal.coursequota'],$cdom,$cnum); + my %staticdefaults = ( + coursequota => 20, + uploadquota => 500, + ); + my %settings = &Apache::lonnet::get('environment',['internal.coursequota','internal.uploadquota','internal.coursecode'], + $cdom,$cnum); my $coursequota = $settings{'internal.coursequota'}; + my $uploadquota = $settings{'internal.uploadquota'}; if ($coursequota eq '') { - $coursequota = 20; + $coursequota = $staticdefaults{'coursequota'}; + } + if ($uploadquota eq '') { + my %domdefs = &Apache::lonnet::get_domain_defaults($cdom); + my $quotatype = &Apache::lonuserutils::get_extended_type($cdom,$cnum,$type,\%settings); + $uploadquota = $domdefs{$quotatype.'quota'}; + if ($uploadquota eq '') { + $uploadquota = $staticdefaults{'uploadquota'}; + } } - &print_header($r,$cdesc); + &print_header($r,$type); my $hidden_elements = &hidden_form_elements(); + my $porthelpitem = &Apache::loncommon::help_open_topic('Modify_Course_Quota'); + my $uploadhelpitem = &Apache::loncommon::help_open_topic('Modify_Course_Upload_Quota'); + my ($disabled,$submit); + if ($readonly) { + $disabled = ' disabled="disabled"'; + } else { + $submit = ''; + } $r->print(< -

$lt{'cquo'}.

+
+

$headline

+

+$porthelpitem $lt{'gpqu'}: MB + +
+ +$uploadhelpitem $lt{'upqu'}: MB + +

-$lt{'each'}
-$subdiv -

-$lt{'gpqu'}: Mb      - +$submit

$hidden_elements $lt{'back'} @@ -341,17 +788,175 @@ ENDDOCUMENT return; } +sub print_set_anonsurvey_threshold { + my ($r,$cdom,$cnum,$cdesc,$type,$readonly) = @_; + my %lt = &Apache::lonlocal::texthash( + 'resp' => 'Responder threshold for anonymous survey submissions display:', + 'sufa' => 'Anonymous survey submissions displayed when responders exceeds', + 'modi' => 'Save', + 'back' => 'Pick another action', + ); + my %settings = &Apache::lonnet::get('environment',['internal.anonsurvey_threshold'],$cdom,$cnum); + my $threshold = $settings{'internal.anonsurvey_threshold'}; + if ($threshold eq '') { + my %domconfig = + &Apache::lonnet::get_dom('configuration',['coursedefaults'],$cdom); + if (ref($domconfig{'coursedefaults'}) eq 'HASH') { + $threshold = $domconfig{'coursedefaults'}{'anonsurvey_threshold'}; + if ($threshold eq '') { + $threshold = 10; + } + } else { + $threshold = 10; + } + } + &print_header($r,$type); + my $hidden_elements = &hidden_form_elements(); + my ($disabled,$submit); + if ($readonly) { + $disabled = ' disabled="disabled"'; + } else { + $submit = ''; + } + my $helpitem = &Apache::loncommon::help_open_topic('Modify_Anonsurvey_Threshold'); + $r->print(< +

$lt{'resp'} $cdesc

+

+$helpitem $lt{'sufa'}:      +$submit +

+$hidden_elements +$lt{'back'} + +ENDDOCUMENT + return; +} + +sub print_postsubmit_config { + my ($r,$cdom,$cnum,$cdesc,$type,$readonly) = @_; + my %lt = &Apache::lonlocal::texthash ( + 'conf' => 'Configure submit button behavior after student makes a submission', + 'disa' => 'Disable submit button/keypress following student submission', + 'nums' => 'Number of seconds submit is disabled', + 'modi' => 'Save', + 'back' => 'Pick another action', + 'yes' => 'Yes', + 'no' => 'No', + ); + my %settings = &Apache::lonnet::get('environment',['internal.postsubmit','internal.postsubtimeout', + 'internal.coursecode','internal.textbook'],$cdom,$cnum); + my $postsubmit = $settings{'internal.postsubmit'}; + if ($postsubmit eq '') { + my %domconfig = + &Apache::lonnet::get_dom('configuration',['coursedefaults'],$cdom); + $postsubmit = 1; + if (ref($domconfig{'coursedefaults'}) eq 'HASH') { + if (ref($domconfig{'coursedefaults'}{'postsubmit'}) eq 'HASH') { + if ($domconfig{'coursedefaults'}{'postsubmit'}{'client'} eq 'off') { + $postsubmit = 0; + } + } + } + } + my ($checkedon,$checkedoff,$display); + if ($postsubmit) { + $checkedon = 'checked="checked"'; + $display = 'block'; + } else { + $checkedoff = 'checked="checked"'; + $display = 'none'; + } + my $postsubtimeout = $settings{'internal.postsubtimeout'}; + my $default = &domain_postsubtimeout($cdom,$type,\%settings); + my $zero = &mt('(Enter 0 to disable until next page reload, or leave blank to use the domain default: [_1])',$default); + if ($postsubtimeout eq '') { + $postsubtimeout = $default; + } + &print_header($r,$type); + my $hidden_elements = &hidden_form_elements(); + my ($disabled,$submit); + if ($readonly) { + $disabled = ' disabled="disabled"'; + } else { + $submit = ''; + } + my $helpitem = &Apache::loncommon::help_open_topic('Modify_Postsubmit_Config'); + $r->print(< +

$lt{'conf'} ($cdesc)

+

+$helpitem $lt{'disa'}: +   + +

+$lt{'nums'}
+$zero
+
+$submit +

+$hidden_elements +$lt{'back'} + +ENDDOCUMENT + return; +} + +sub domain_postsubtimeout { + my ($cdom,$type,$settings) = @_; + return unless (ref($settings) eq 'HASH'); + my $lctype = lc($type); + unless (($type eq 'Community') || ($type eq 'Placement')) { + $lctype = 'unofficial'; + if ($settings->{'internal.coursecode'}) { + $lctype = 'official'; + } elsif ($settings->{'internal.textbook'}) { + $lctype = 'textbook'; + } + } + my %domconfig = + &Apache::lonnet::get_dom('configuration',['coursedefaults'],$cdom); + my $postsubtimeout = 60; + if (ref($domconfig{'coursedefaults'}) eq 'HASH') { + if (ref($domconfig{'coursedefaults'}{'postsubmit'}) eq 'HASH') { + if (ref($domconfig{'coursedefaults'}{'postsubmit'}{'timeout'}) eq 'HASH') { + if ($domconfig{'coursedefaults'}{'postsubmit'}{'timeout'}{$lctype} ne '') { + $postsubtimeout = $domconfig{'coursedefaults'}{'postsubmit'}{'timeout'}{$lctype}; + } + } + } + } + return $postsubtimeout; +} + sub print_catsettings { - my ($r,$cdom,$cnum,$cdesc) = @_; - &print_header($r,$cdesc); + my ($r,$cdom,$cnum,$cdesc,$type,$readonly) = @_; + &print_header($r,$type); my %lt = &Apache::lonlocal::texthash( - 'back' => 'Back to options page', + 'back' => 'Pick another action', + 'catset' => 'Catalog Settings for Course', + 'visi' => 'Visibility in Course/Community Catalog', + 'exclude' => 'Exclude from course catalog:', + 'categ' => 'Categorize Course', + 'assi' => 'Assign one or more categories and/or subcategories to this course.' ); + if ($type eq 'Community') { + $lt{'catset'} = &mt('Catalog Settings for Community'); + $lt{'exclude'} = &mt('Exclude from course catalog'); + $lt{'categ'} = &mt('Categorize Community'); + $lt{'assi'} = &mt('Assign one or more subcategories to this community.'); + } $r->print('
'. - '

'.&mt('Catalog Settings for Course').'

'); + '

'.$lt{'catset'}.' '.$cdesc.'

'); my %domconf = &Apache::lonnet::get_dom('configuration',['coursecategories'],$cdom); - my @cat_params = &catalog_settable($domconf{'coursecategories'}); + my @cat_params = &catalog_settable($domconf{'coursecategories'},$type); if (@cat_params > 0) { + my $disabled; + if ($readonly) { + $disabled = ' disabled="disabled"'; + } my %currsettings = &Apache::lonnet::get('environment',['hidefromcat','categories'],$cdom,$cnum); if (grep(/^togglecats$/,@cat_params)) { @@ -361,32 +966,50 @@ sub print_catsettings { $excludeon = $excludeoff; $excludeoff = ''; } - $r->print('

'.&mt('Visibility in Course Catalog').'

'. - &mt("Unless excluded, a course is listed in this domain's publicly accessible course catalog, if at least one of the following applies").':
    '. - '
  • '.&mt('Auto-cataloging is enabled and the course is assigned an institutional code').'
  • '. - '
  • '.&mt('The course has been categorized into at least one of the course categories defined for the domain.').'
'. - &mt('Exclude from course catalog').'    
'); + $r->print('

'.$lt{'visi'}.'

'. + $lt{'exclude'}. + '    

'); + if ($type eq 'Community') { + $r->print(&mt("If a community has been categorized using at least one of the categories defined for communities in the domain, it will be listed in the domain's publicly accessible Course/Community Catalog, unless excluded.")); + } elsif ($type eq 'Placement') { + $r->print(&mt("If a placement test has been categorized using at least one of the categories defined for placement tests in the domain, it will be listed in the domain's publicly accessible Course/Community Catalog, unless excluded.")); + } else { + $r->print(&mt("Unless excluded, a course will be listed in the domain's publicly accessible Course/Community Catalog, if at least one of the following applies").':

    '. + '
  • '.&mt('Auto-cataloging is enabled and the course is assigned an institutional code.').'
  • '. + '
  • '.&mt('The course has been categorized using at least one of the course categories defined for the domain.').'
'); + } + $r->print('

'); } if (grep(/^categorize$/,@cat_params)) { - $r->print('

'.&mt('Categorize Course').'

'); + $r->print('

'.$lt{'categ'}.'

'); if (ref($domconf{'coursecategories'}) eq 'HASH') { my $cathash = $domconf{'coursecategories'}{'cats'}; if (ref($cathash) eq 'HASH') { - $r->print(&mt('Assign one or more categories to this course.').'

'. + $r->print($lt{'assi'}.'

'. &Apache::loncommon::assign_categories_table($cathash, - $currsettings{'categories'})); + $currsettings{'categories'},$type,$disabled)); } else { $r->print(&mt('No categories defined for this domain')); } } else { $r->print(&mt('No categories defined for this domain')); } - $r->print('

'.&mt('If auto-cataloging based on institutional code is enabled in the domain, a course will continue to be listed in the catalog of official courses, in addition to receiving a listing under any manually assigned categor(ies).').'

'); + unless (($type eq 'Community') || ($type eq 'Placement')) { + $r->print('

'.&mt('If auto-cataloging based on institutional code is enabled in the domain, a course will continue to be listed in the catalog of official courses, in addition to receiving a listing under any manually assigned categor(ies).').'

'); + } + } + unless ($readonly) { + $r->print('

'); } - $r->print(''); } else { - $r->print(''.&mt('Catalog settings in this domain are set in course context via "Set Course Environment".').'

'."\n". + $r->print(''); + if ($type eq 'Community') { + $r->print(&mt('Catalog settings in this domain are set in community context via "Community Configuration".')); + } else { + $r->print(&mt('Catalog settings in this domain are set in course context via "Course Configuration".')); + } + $r->print('

'."\n". ''. $lt{'back'}.''); } @@ -395,370 +1018,611 @@ sub print_catsettings { } sub print_course_modification_page { - my ($r,$cdom,$cnum,$cdesc,$domdesc) = @_; - my %longtype = &course_settings_descrip(); - my %enrollvar = &get_enrollment_settings($cdom,$cnum); - my $ownertable; + my ($r,$cdom,$cnum,$cdesc,$crstype,$readonly) = @_; my %lt=&Apache::lonlocal::texthash( 'actv' => "Active", 'inac' => "Inactive", - 'ccor' => "Course Coordinator", - 'noen' => "No end date", 'ownr' => "Owner", 'name' => "Name", 'unme' => "Username:Domain", 'stus' => "Status", - 'cquo' => "Disk space for storage of group portfolio", - 'gpqu' => "Course portfolio files disk space", - 'each' => "Each course group can be assigned a quota for portfolio files uploaded to the group.", - 'cose' => "Course settings for LON-CAPA courses that control automated student enrollment based on classlist data available from your institution's student information system fall into two groups: (a) settings that can be modified by a Course Coordinator using the ", - 'aenm' => "Automated Enrollment Manager", - 'andb' => " and (b) settings that may only be modified by a Domain Coordinator via this page.", - 'caes' => 'Current automated enrollment settings', - 'cour' => "Course settings that control automated enrollment in this LON-CAPA course -are currently:", - 'nocc' => "There is currently no course owner set for this course. In addition, no active course coordinators from this LON-CAPA domain were found, so you will not be able assign a course owner. If you wish to assign a course owner, it is recommended that you use the 'User Roles' screen to add a course coordinator with a LON-CAPA account in this domain to the course.", - 'ccus' => "A course coordinator can use the 'Automated Enrollment Manager' to change -all settings except course code, course owner, and default authentication method for students added to your course (who are also new to the LON-CAPA system at this domain).", - 'ccod' => "Course Code", - 'ccus' => "The course code is used during automated enrollment to map the internal LON-CAPA course ID for this course to the corresponding course section ID(s) used by the office responsible for providing official class lists for courses at your institution.", - 'cown' => "Course Owner", - 'cous' => "The course owner is used in the retrieval of class lists from your institution's student information system when access to class lists data incorporates validation of instructor status.", - 'tabl' => 'The table below contains a list of active course coordinators in this course, who are from this domain', - 'usrd' => 'Use the radio buttons to select a different course owner.', - 'deam' => "Default Authentication method", - 'deus' => "The default authentication method, and default authentication parameter (domain, initial password or argument) are used when automatic enrollment of students in a course requires addition of new user accounts in your domain, and the class list file contains empty entries for the <authtype> and <autharg> properties for the new student. If you choose 'internally authenticated', and leave the initial password field empty, the automated enrollment process will create a randomized password for each new student account that it adds to your LON-CAPA domain.", + 'nocc' => 'There is currently no owner set for this course.', 'gobt' => "Save", + 'sett' => 'Setting', + 'domd' => 'Domain default', + 'whom' => 'Who configures', ); - - my @coursepersonnel = &Apache::lonnet::getkeys('nohist_userroles',$cdom,$cnum); - my @local_ccs = (); - my %cc_status = (); - my %pname = (); - my $active_cc; - foreach my $person (@coursepersonnel) { - my ($role,$user) = split(/:/,$person,2); - $user =~ s/:$//; - if (($role eq 'cc') && ($user ne '')) { - if (!grep(/^\Q$user\E$/,@local_ccs)) { - my ($ccname,$ccdom) = split(/:/,$user); - $active_cc = - &Apache::loncommon::check_user_status($ccdom,$ccname,$cdom, - $cnum,'cc'); - if ($active_cc eq 'active') { - push(@local_ccs,$user); - $pname{$user} = &Apache::loncommon::plainname($ccname,$ccdom); - $cc_status{$user} = $lt{'actv'}; - } - } + my ($ownertable,$ccrole,$javascript_validations,$authenitems,$ccname,$disabled); + my %enrollvar = &get_enrollment_settings($cdom,$cnum); + my %settings = &Apache::lonnet::get('environment',['internal.coursecode','internal.textbook', + 'internal.selfenrollmgrdc','internal.selfenrollmgrcc', + 'internal.mysqltables'],$cdom,$cnum); + my $type = &Apache::lonuserutils::get_extended_type($cdom,$cnum,$crstype,\%settings); + my @specific_managebydc = split(/,/,$settings{'internal.selfenrollmgrdc'}); + my @specific_managebycc = split(/,/,$settings{'internal.selfenrollmgrcc'}); + my %domdefaults = &Apache::lonnet::get_domain_defaults($cdom); + my %passwdconf = &Apache::lonnet::get_passwdconf($cdom); + my @default_managebydc = split(/,/,$domdefaults{$type.'selfenrolladmdc'}); + if ($crstype eq 'Community') { + $ccrole = 'co'; + $lt{'nocc'} = &mt('There is currently no owner set for this community.'); + } else { + $ccrole ='cc'; + ($javascript_validations,$authenitems) = &gather_authenitems($cdom,\%enrollvar,$readonly); + } + $ccname = &Apache::lonnet::plaintext($ccrole,$crstype); + if ($readonly) { + $disabled = ' disabled="disabled"'; + } + my %roleshash = &Apache::lonnet::get_my_roles($cnum,$cdom,'','',[$ccrole]); + my (@local_ccs,%cc_status,%pname); + foreach my $item (keys(%roleshash)) { + my ($uname,$udom) = split(/:/,$item); + if (!grep(/^\Q$uname\E:\Q$udom\E$/,@local_ccs)) { + push(@local_ccs,$uname.':'.$udom); + $pname{$uname.':'.$udom} = &Apache::loncommon::plainname($uname,$udom); + $cc_status{$uname.':'.$udom} = $lt{'actv'}; } } - if ( (!grep(/^$enrollvar{'courseowner'}$/,@local_ccs)) && - ($enrollvar{'courseowner'} ne '') ) { - my ($owneruname,$ownerdom) = split(/:/,$enrollvar{'courseowner'}); + if (($enrollvar{'courseowner'} ne '') && + (!grep(/^$enrollvar{'courseowner'}$/,@local_ccs))) { push(@local_ccs,$enrollvar{'courseowner'}); + my ($owneruname,$ownerdom) = split(/:/,$enrollvar{'courseowner'}); $pname{$enrollvar{'courseowner'}} = &Apache::loncommon::plainname($owneruname,$ownerdom); - $active_cc = &Apache::loncommon::check_user_status($ownerdom,$owneruname, - $cdom,$cnum,'cc'); + my $active_cc = &Apache::loncommon::check_user_status($ownerdom,$owneruname, + $cdom,$cnum,$ccrole); if ($active_cc eq 'active') { $cc_status{$enrollvar{'courseowner'}} = $lt{'actv'}; } else { $cc_status{$enrollvar{'courseowner'}} = $lt{'inac'}; } } - my $numlocalcc = @local_ccs; + @local_ccs = sort(@local_ccs); + if (@local_ccs == 0) { + $ownertable = $lt{'nocc'}; + } else { + my $numlocalcc = scalar(@local_ccs); + $ownertable = ''. + &Apache::loncommon::start_data_table()."\n". + &Apache::loncommon::start_data_table_header_row()."\n". + ''.$lt{'ownr'}.''. + ''.$lt{'name'}.''. + ''.$lt{'unme'}.''. + ''.$lt{'stus'}.''. + &Apache::loncommon::end_data_table_header_row()."\n"; + foreach my $cc (@local_ccs) { + $ownertable .= &Apache::loncommon::start_data_table_row()."\n"; + if ($cc eq $enrollvar{'courseowner'}) { + $ownertable .= ''."\n"; + } else { + $ownertable .= ''."\n"; + } + $ownertable .= + ''.$pname{$cc}.''."\n". + ''.$cc.''."\n". + ''.$cc_status{$cc}.' '.$ccname.''."\n". + &Apache::loncommon::end_data_table_row()."\n"; + } + $ownertable .= &Apache::loncommon::end_data_table(); + } + &print_header($r,$crstype,$javascript_validations); + my $dctitle = &Apache::lonnet::plaintext('dc'); + my $mainheader = &modifiable_only_title($crstype); + my $hidden_elements = &hidden_form_elements(); + $r->print(''."\n". + '

'.$mainheader.' '.$cdesc.'

'. + &Apache::lonhtmlcommon::start_pick_box()); + if ($crstype eq 'Community') { + $r->print(&Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Community_Owner'). + ' '.&mt('Community Owner'))."\n". + $ownertable."\n".&Apache::lonhtmlcommon::row_closure()); + } else { + $r->print(&Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Course_Instcode'). + ' '.&mt('Course Code'))."\n". + ''. + &Apache::lonhtmlcommon::row_closure()); + if (($crstype eq 'Course') && (&showcredits($cdom))) { + $r->print(&Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Course_Credithours'). + ' '.&mt('Credits (students)'))."\n". + ''. + &Apache::lonhtmlcommon::row_closure()); + } + $r->print(&Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Course_Defaultauth'). + ' '.&mt('Default Authentication method'))."\n". + $authenitems."\n". + &Apache::lonhtmlcommon::row_closure(). + &Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Course_Owner'). + ' '.&mt('Course Owner'))."\n". + $ownertable."\n".&Apache::lonhtmlcommon::row_closure()); + if (($passwdconf{'crsownerchg'}) && ($type ne 'Placement')) { + my $checked; + if ($enrollvar{'nopasswdchg'}) { + $checked = ' checked="checked"'; + } + $r->print(&Apache::lonhtmlcommon::row_title( + &Apache::loncommon::help_open_topic('Modify_Course_Chgpasswd'). + ' '.&mt('Changing passwords (internal)'))."\n". + '

'.$hidden_elements); + unless ($readonly) { + $r->print('print('this.form.submit();"'); + } else { + $r->print('javascript:verify_message(this.form);"'); + } + $r->print(' value="'.$lt{'gobt'}.'" />'); + } + $r->print('

'); + return; +} + +sub print_selfenrollconfig { + my ($r,$type,$cdesc,$coursehash,$readonly) = @_; + return unless(ref($coursehash) eq 'HASH'); + my $cnum = $coursehash->{'num'}; + my $cdom = $coursehash->{'domain'}; + my %currsettings = &get_selfenroll_settings($coursehash); + &print_header($r,$type); + $r->print('

'.&mt('Self-enrollment with a student role in: [_1]', + ''.$cdesc.'').'

'."\n"); + &Apache::loncreateuser::print_selfenroll_menu($r,'domain',$env{'form.pickedcourse'}, + $cdom,$cnum,\%currsettings, + &hidden_form_elements(),$readonly); + return; +} + +sub modify_selfenrollconfig { + my ($r,$type,$cdesc,$coursehash) = @_; + return unless(ref($coursehash) eq 'HASH'); + my $cnum = $coursehash->{'num'}; + my $cdom = $coursehash->{'domain'}; + my %currsettings = &get_selfenroll_settings($coursehash); + &print_header($r,$type); + $r->print('

'.&mt('Self-enrollment with a student role in: [_1]', + ''.$cdesc.'').'

'."\n"); + $r->print('
'."\n". + &hidden_form_elements().'
'); + &Apache::loncreateuser::update_selfenroll_config($r,$env{'form.pickedcourse'}, + $cdom,$cnum,'domain',$type,\%currsettings); + $r->print('
'); + return; +} + +sub get_selfenroll_settings { + my ($coursehash) = @_; + my %currsettings; + if (ref($coursehash) eq 'HASH') { + %currsettings = ( + selfenroll_types => $coursehash->{'internal.selfenroll_types'}, + selfenroll_registered => $coursehash->{'internal.selfenroll_registered'}, + selfenroll_section => $coursehash->{'internal.selfenroll_section'}, + selfenroll_notifylist => $coursehash->{'internal.selfenroll_notifylist'}, + selfenroll_approval => $coursehash->{'internal.selfenroll_approval'}, + selfenroll_limit => $coursehash->{'internal.selfenroll_limit'}, + selfenroll_cap => $coursehash->{'internal.selfenroll_cap'}, + selfenroll_start_date => $coursehash->{'internal.selfenroll_start_date'}, + selfenroll_end_date => $coursehash->{'internal.selfenroll_end_date'}, + selfenroll_start_access => $coursehash->{'internal.selfenroll_start_access'}, + selfenroll_end_access => $coursehash->{'internal.selfenroll_end_access'}, + default_enrollment_start_date => $coursehash->{'default_enrollment_start_date'}, + default_enrollment_end_date => $coursehash->{'default_enrollment_end_date'}, + uniquecode => $coursehash->{'internal.uniquecode'}, + ); + } + return %currsettings; +} + +sub modifiable_only_title { + my ($type) = @_; + my $dctitle = &Apache::lonnet::plaintext('dc'); + if ($type eq 'Community') { + return &mt('Community settings modifiable only by [_1] for:',$dctitle); + } else { + return &mt('Course settings modifiable only by [_1] for:',$dctitle); + } +} - my $helplink=&Apache::loncommon::help_open_topic('Modify_Course',&mt("Help on Modifying Courses")); +sub gather_authenitems { + my ($cdom,$enrollvar,$readonly) = @_; my ($krbdef,$krbdefdom)=&Apache::loncommon::get_kerberos_defaults($cdom); my $curr_authtype = ''; my $curr_authfield = ''; - if ($enrollvar{'authtype'} =~ /^krb/) { - $curr_authtype = 'krb'; - } elsif ($enrollvar{'authtype'} eq 'internal' ) { - $curr_authtype = 'int'; - } elsif ($enrollvar{'authtype'} eq 'localauth' ) { - $curr_authtype = 'loc'; + if (ref($enrollvar) eq 'HASH') { + if ($enrollvar->{'authtype'} =~ /^krb/) { + $curr_authtype = 'krb'; + } elsif ($enrollvar->{'authtype'} eq 'internal' ) { + $curr_authtype = 'int'; + } elsif ($enrollvar->{'authtype'} eq 'localauth' ) { + $curr_authtype = 'loc'; + } elsif ($enrollvar->{'authtype'} eq 'lti' ) { + $curr_authtype = 'lti'; + } } unless ($curr_authtype eq '') { $curr_authfield = $curr_authtype.'arg'; } - my $javascript_validations=&Apache::lonuserutils::javascript_validations('modifycourse',$krbdefdom,$curr_authtype,$curr_authfield); + my $javascript_validations = + &Apache::lonuserutils::javascript_validations('modifycourse',$krbdefdom, + $curr_authtype,$curr_authfield); my %param = ( formname => 'document.'.$env{'form.phase'}, - kerb_def_dom => $krbdefdom, - kerb_def_auth => $krbdef, + kerb_def_dom => $krbdefdom, + kerb_def_auth => $krbdef, mode => 'modifycourse', curr_authtype => $curr_authtype, - curr_autharg => $enrollvar{'autharg'} - ); + curr_autharg => $enrollvar->{'autharg'}, + readonly => $readonly, + ); my (%authform,$authenitems); $authform{'krb'} = &Apache::loncommon::authform_kerberos(%param); $authform{'int'} = &Apache::loncommon::authform_internal(%param); $authform{'loc'} = &Apache::loncommon::authform_local(%param); - foreach my $item ('krb','int','loc') { + $authform{'lti'} = &Apache::loncommon::authform_lti(%param); + foreach my $item ('krb','int','loc','lti') { if ($authform{$item} ne '') { $authenitems .= $authform{$item}.'
'; } - } - if ($numlocalcc == 0) { - $ownertable = $lt{'nocc'}; - } - - if ($numlocalcc > 0) { - @local_ccs = sort @local_ccs; - $ownertable = qq( - - - - - - - - \n". - "\n". - "\n". - "\n". - &Apache::loncommon::end_data_table_header_row()."\n"; - foreach my $cc (@local_ccs) { - $ownertable .= &Apache::loncommon::start_data_table_row()."\n"; - if ($cc eq $enrollvar{'courseowner'}) { - $ownertable .= ''."\n"; - } else { - $ownertable .= ''."\n"; - } - $ownertable .= - ''."\n". - ''."\n". - ''."\n". - &Apache::loncommon::end_data_table_row()."\n"; - } - $ownertable .= &Apache::loncommon::end_data_table()." - - -
$lt{'tabl'} - $cdom ($domdesc). $lt{'usrd'} -
 
). - &Apache::loncommon::start_data_table()."\n". - &Apache::loncommon::start_data_table_header_row()."\n". - "$lt{'ownr'}$lt{'name'}$lt{'unme'}$lt{'stus'}'.$pname{$cc}.''.$cc.''.$cc_status{$cc}.' '.$lt{'ccor'}.'
"; } - &print_header($r,$cdesc,$javascript_validations); - my $type = $env{'form.type'}; - if ($type eq '') { - $type = 'Course'; - } - my $dctitle = &Apache::lonnet::plaintext('dc'); - my $cctitle = &Apache::lonnet::plaintext('cc',$type); - my $mainheader = &mt('Course settings modifiable by [_1] only.',$dctitle); - my $hidden_elements = &hidden_form_elements(); - $r->print(< -

$mainheader

-

- - - - - - - - - - - -
Use the appropriate text boxes and radio buttons below to change some or all of the four automated enrollment settings that may only be changed by a Domain Coordinator. -
- $lt{'ccod'}:   -

- $lt{'ccus'} -
- $lt{'cown'}:

- $ownertable -

- $lt{'cous'} -
- $lt{'deam'}:

- $authenitems -
- $lt{'deus'}. -
 
- - - - -
- $hidden_elements - -
- -
-
-ENDDOCUMENT - return; + return($javascript_validations,$authenitems); } sub modify_course { my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_; - my %longtype = &course_settings_descrip(); - my %settings = &Apache::lonnet::get('environment',['internal.courseowner','internal.coursecode','internal.authtype','internal.autharg','internal.sectionnums','internal.crosslistings','description'],$cdom,$cnum); - my %currattr = (); - my %newattr = (); - my %cenv = (); - my $response; - my $chgresponse; - my $nochgresponse; - my $warning; - my $reply; - my @changes = (); - my @nochanges = (); - my @sections = (); - my @xlists = (); - my %changed = ( code => 0, - owner => 0, - ); - unless ($settings{'internal.sectionnums'} eq '') { - if ($settings{'internal.sectionnums'} =~ m/,/) { - @sections = split/,/,$settings{'internal.sectionnums'}; - } else { - $sections[0] = $settings{'internal.sectionnums'}; - } - } - unless ($settings{'internal.crosslistings'} eq'') { - if ($settings{'internal.crosslistings'} =~ m/,/) { - @xlists = split/,/,$settings{'internal.crosslistings'}; - } else { - $xlists[0] = $settings{'internal.crosslistings'}; + my %longtype = &course_settings_descrip($type); + my @items = ('internal.courseowner','description','internal.co-owners', + 'internal.pendingco-owners','internal.selfenrollmgrdc', + 'internal.selfenrollmgrcc','internal.mysqltables'); + my ($selfenrollrows,$selfenrolltitles) = &Apache::lonuserutils::get_selfenroll_titles(); + unless (($type eq 'Community') || ($type eq 'Placement')) { + push(@items,('internal.coursecode','internal.authtype','internal.autharg', + 'internal.sectionnums','internal.crosslistings')); + if (&showcredits($cdom)) { + push(@items,'internal.defaultcredits'); + } + my %passwdconf = &Apache::lonnet::get_passwdconf($cdom); + if ($passwdconf{'crsownerchg'}) { + push(@items,'internal.nopasswdchg'); } } - - my @modifiable_params = &get_dc_settable(); + my %settings = &Apache::lonnet::get('environment',\@items,$cdom,$cnum); + my $description = $settings{'description'}; + my ($ccrole,$response,$chgresponse,$nochgresponse,$reply,%currattr,%newattr, + %cenv,%changed,@changes,@nochanges,@sections,@xlists,@warnings); + my @modifiable_params = &get_dc_settable($type,$cdom); foreach my $param (@modifiable_params) { - my $attr = 'internal.'.$param; - $currattr{$param} = $settings{$attr}; + $currattr{$param} = $settings{'internal.'.$param}; } - - my $description = $settings{'description'}; - - if ($env{'form.login'} eq 'krb') { - $newattr{'authtype'} = $env{'form.login'}; - $newattr{'authtype'} .= $env{'form.krbver'}; - $newattr{'autharg'} = $env{'form.krbarg'}; - } elsif ($env{'form.login'} eq 'int') { - $newattr{'authtype'} ='internal'; - if ((defined($env{'form.intarg'})) && ($env{'form.intarg'})) { - $newattr{'autharg'} = $env{'form.intarg'}; - } - } elsif ($env{'form.login'} eq 'loc') { - $newattr{'authtype'} = 'localauth'; - if ((defined($env{'form.locarg'})) && ($env{'form.locarg'})) { - $newattr{'autharg'} = $env{'form.locarg'}; - } - } - if ( $newattr{'authtype'}=~ /^krb/) { - if ($newattr{'autharg'} eq '') { - $warning = '

' - .&mt('As you did not include the default Kerberos domain' + if ($type eq 'Community') { + %changed = ( owner => 0 ); + $ccrole = 'co'; + } else { + %changed = ( code => 0, + owner => 0, + passwd => 0, + ); + $ccrole = 'cc'; + unless ($settings{'internal.sectionnums'} eq '') { + if ($settings{'internal.sectionnums'} =~ m/,/) { + @sections = split/,/,$settings{'internal.sectionnums'}; + } else { + $sections[0] = $settings{'internal.sectionnums'}; + } + } + unless ($settings{'internal.crosslistings'} eq '') { + if ($settings{'internal.crosslistings'} =~ m/,/) { + @xlists = split/,/,$settings{'internal.crosslistings'}; + } else { + $xlists[0] = $settings{'internal.crosslistings'}; + } + } + if ($env{'form.login'} eq 'krb') { + $newattr{'authtype'} = $env{'form.login'}; + $newattr{'authtype'} .= $env{'form.krbver'}; + $newattr{'autharg'} = $env{'form.krbarg'}; + } elsif ($env{'form.login'} eq 'int') { + $newattr{'authtype'} ='internal'; + if ((defined($env{'form.intarg'})) && ($env{'form.intarg'})) { + $newattr{'autharg'} = $env{'form.intarg'}; + } + } elsif ($env{'form.login'} eq 'loc') { + $newattr{'authtype'} = 'localauth'; + if ((defined($env{'form.locarg'})) && ($env{'form.locarg'})) { + $newattr{'autharg'} = $env{'form.locarg'}; + } + } elsif ($env{'form.login'} eq 'lti') { + $newattr{'authtype'} = 'lti'; + } + if ( $newattr{'authtype'}=~ /^krb/) { + if ($newattr{'autharg'} eq '') { + push(@warnings, + &mt('As you did not include the default Kerberos domain' .' to be used for authentication in this class, the' .' institutional data used by the automated' .' enrollment process must include the Kerberos' - .' domain for each new student.') - .'

'; + .' domain for each new student.')); + } + } + + if ( exists($env{'form.coursecode'}) ) { + $newattr{'coursecode'}=$env{'form.coursecode'}; + unless ( $newattr{'coursecode'} eq $currattr{'coursecode'} ) { + $changed{'code'} = 1; + } + } + if ( exists($env{'form.mysqltables'}) ) { + $newattr{'mysqltables'} = $env{'form.mysqltables'}; + $newattr{'mysqltables'} =~ s/\D+//g; + } + if ($type ne 'Placement') { + if (&showcredits($cdom) && exists($env{'form.defaultcredits'})) { + $newattr{'defaultcredits'}=$env{'form.defaultcredits'}; + $newattr{'defaultcredits'} =~ s/[^\d\.]//g; + } + if (grep(/^nopasswdchg$/,@modifiable_params)) { + if ($env{'form.nopasswdchg'}) { + $newattr{'nopasswdchg'} = 1; + unless ($currattr{'nopasswdchg'}) { + $changed{'passwd'} = 1; + } + } elsif ($currattr{'nopasswdchg'}) { + $changed{'passwd'} = 1; + } + } } } + my @newmgrdc = (); + my @newmgrcc = (); + my @currmgrdc = split(/,/,$currattr{'selfenrollmgrdc'}); + my @currmgrcc = split(/,/,$currattr{'selfenrollmgrcc'}); + + foreach my $item (@{$selfenrollrows}) { + if ($env{'form.selfenrollmgr_'.$item} eq '0') { + push(@newmgrdc,$item); + } elsif ($env{'form.selfenrollmgr_'.$item} eq '1') { + push(@newmgrcc,$item); + } + } + + $newattr{'selfenrollmgrdc'}=join(',',@newmgrdc); + $newattr{'selfenrollmgrcc'}=join(',',@newmgrcc); + + my $cctitle; + if ($type eq 'Community') { + $cctitle = &mt('Community personnel'); + } else { + $cctitle = &mt('Course personnel'); + } + my $dctitle = &Apache::lonnet::plaintext('dc'); + if ( exists($env{'form.courseowner'}) ) { $newattr{'courseowner'}=$env{'form.courseowner'}; unless ( $newattr{'courseowner'} eq $currattr{'courseowner'} ) { $changed{'owner'} = 1; } } - - if ( exists($env{'form.coursecode'}) ) { - $newattr{'coursecode'}=$env{'form.coursecode'}; - unless ( $newattr{'coursecode'} eq $currattr{'coursecode'} ) { - $changed{'code'} = 1; - } - } - if ($changed{'owner'} || $changed{'code'}) { + + if ($changed{'owner'} || $changed{'code'} || $changed{'passwd'}) { my %crsinfo = &Apache::lonnet::courseiddump($cdom,'.',1,'.','.',$cnum, undef,undef,'.'); if (ref($crsinfo{$env{'form.pickedcourse'}}) eq 'HASH') { - $crsinfo{$env{'form.pickedcourse'}}{'inst_code'} = $env{'form.coursecode'}; - $crsinfo{$env{'form.pickedcourse'}}{'owner'} = $env{'form.courseowner'}; + if ($changed{'code'}) { + $crsinfo{$env{'form.pickedcourse'}}{'inst_code'} = $env{'form.coursecode'}; + } + if ($changed{'owner'}) { + $crsinfo{$env{'form.pickedcourse'}}{'owner'} = $env{'form.courseowner'}; + } + if ($changed{'passwd'}) { + if ($env{'form.nopasswdchg'}) { + $crsinfo{$env{'form.pickedcourse'}}{'nopasswdchg'} = 1; + } else { + delete($crsinfo{'nopasswdchg'}); + } + } my $chome = &Apache::lonnet::homeserver($cnum,$cdom); my $putres = &Apache::lonnet::courseidput($cdom,\%crsinfo,$chome,'notime'); + if (($putres eq 'ok') && (($changed{'owner'} || $changed{'code'}))) { + &update_coowners($cdom,$cnum,$chome,\%settings,\%newattr); + if ($changed{'code'}) { + &Apache::lonnet::devalidate_cache_new('instcats',$cdom); + # Update cache of self-cataloging courses on institution's server(s). + if (&Apache::lonnet::shared_institution($cdom)) { + unless ($registered_cleanup) { + my $handlers = $r->get_handlers('PerlCleanupHandler'); + $r->set_handlers('PerlCleanupHandler' => [\&devalidate_remote_instcats,@{$handlers}]); + $registered_cleanup=1; + $modified_dom = $cdom; + } + } + } + } } } foreach my $param (@modifiable_params) { if ($currattr{$param} eq $newattr{$param}) { push(@nochanges,$param); } else { - my $attr = 'internal.'.$param; - $cenv{$attr} = $newattr{$param}; + $cenv{'internal.'.$param} = $newattr{$param}; push(@changes,$param); } } if (@changes > 0) { - $chgresponse = &mt("The following automated enrollment parameters have been changed:
    "); + $chgresponse = &mt('The following settings have been changed:').'
      '; } - if (@nochanges > 0) { - $nochgresponse = &mt("The following automated enrollment parameters remain unchanged:
        "); + if (@nochanges > 0) { + $nochgresponse = &mt('The following settings remain unchanged:').'
          '; } if (@changes > 0) { my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom,$cnum); if ($putreply !~ /^ok$/) { - $response = &mt("There was a problem processing your requested changes. The automated enrollment settings for this course have been left unchanged.
          ").&mt('Error: ').$putreply; + $response = '

          '. + &mt('There was a problem processing your requested changes.').'
          '; + if ($type eq 'Community') { + $response .= &mt('Settings for this community have been left unchanged.'); + } else { + $response .= &mt('Settings for this course have been left unchanged.'); + } + $response .= '
          '.&mt('Error: ').$putreply.'

          '; } else { + if ($env{'course.'.$cdom.'_'.$cnum.'.description'} ne '') { + my %newenv; + map { $newenv{'course.'.$cdom.'_'.$cnum.'.internal.'.$_} = $newattr{$_}; } @changes; + &Apache::lonnet::appenv(\%newenv); + } foreach my $attr (@modifiable_params) { - if (grep/^$attr$/,@changes) { - $chgresponse .= "
        • $longtype{$attr} ".&mt("now set to \"").$newattr{$attr}."\".
        • "; + if (grep/^\Q$attr\E$/,@changes) { + my $shown = $newattr{$attr}; + if ($attr eq 'selfenrollmgrdc') { + $shown = &selfenroll_config_status(\@newmgrdc,$selfenrolltitles); + } elsif ($attr eq 'selfenrollmgrcc') { + $shown = &selfenroll_config_status(\@newmgrcc,$selfenrolltitles); + } elsif (($attr eq 'defaultcredits') && ($shown eq '')) { + $shown = &mt('None'); + } elsif (($attr eq 'mysqltables') && ($shown eq '')) { + $shown = &mt('domain default'); + } elsif ($attr eq 'nopasswdchg') { + if ($shown) { + $shown = &mt('Yes'); + } else { + $shown = &mt('No'); + } + } + $chgresponse .= '
        • '.&mt('[_1] now set to: [_2]',$longtype{$attr},$shown).'
        • '; } else { - $nochgresponse .= "
        • $longtype{$attr} ".&mt("still set to \"").$currattr{$attr}."\".
        • "; + my $shown = $currattr{$attr}; + if ($attr eq 'selfenrollmgrdc') { + $shown = &selfenroll_config_status(\@currmgrdc,$selfenrolltitles); + } elsif ($attr eq 'selfenrollmgrcc') { + $shown = &selfenroll_config_status(\@currmgrcc,$selfenrolltitles); + } elsif (($attr eq 'defaultcredits') && ($shown eq '')) { + $shown = &mt('None'); + } elsif (($attr eq 'mysqltables') && ($shown eq '')) { + $shown = &mt('domain default'); + } elsif ($attr eq 'nopasswdchg') { + if ($shown) { + $shown = &mt('Yes'); + } else { + $shown = &mt('No'); + } + } + $nochgresponse .= '
        • '.&mt('[_1] still set to: [_2]',$longtype{$attr},$shown).'
        • '; } } - if ($changed{'code'} || $changed{'owner'}) { + if (($type ne 'Community') && ($type ne 'Placement') && ($changed{'code'} || $changed{'owner'})) { if ( $newattr{'courseowner'} eq '') { - $warning .= &mt("There is no owner associated with this LON-CAPA course. If automated enrollment in LON-CAPA courses at your institution requires validation of course owners, automated enrollment will fail for this course.
          "); + push(@warnings,&mt('There is no owner associated with this LON-CAPA course.'). + '
          '.&mt('If automated enrollment at your institution requires validation of course owners, automated enrollment will fail.')); } else { + my %crsenv = &Apache::lonnet::get('environment',['internal.co-owners'],$cdom,$cnum); + my $coowners = $crsenv{'internal.co-owners'}; if (@sections > 0) { if ($changed{'code'}) { foreach my $sec (@sections) { if ($sec =~ m/^(.+):/) { + my $instsec = $1; my $inst_course_id = $newattr{'coursecode'}.$1; my $course_check = &Apache::lonnet::auto_validate_courseID($cnum,$cdom,$inst_course_id); if ($course_check eq 'ok') { - my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$inst_course_id,$newattr{'courseowner'}); - unless ($outcome eq 'ok') { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for ").$newattr{'coursecode'}.&mt(" - section $1 for the following reason: $outcome.
          "); + my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$inst_course_id,$newattr{'courseowner'},$coowners); + unless ($outcome eq 'ok') { + + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for "[_2]" - section: [_3] for the following reason: "[_4]".',$description,$newattr{'coursecode'},$instsec,$outcome).'
          '); } } else { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for ").$newattr{'coursecode'}.&mt(" - section $1 for the following reason: $course_check.
          "); + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for "[_2]" - section: [_3] for the following reason: "[_4]".',$description,$newattr{'coursecode'},$instsec,$course_check)); } } else { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for ").$newattr{'coursecode'}.&mt(" - section $sec because this is not a valid section entry.
          "); + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for "[_2]" - section: [_3], because this is not a valid section entry.',$description,$newattr{'coursecode'},$sec)); } } } elsif ($changed{'owner'}) { foreach my $sec (@sections) { if ($sec =~ m/^(.+):/) { - my $inst_course_id = $newattr{'coursecode'}.$1; - my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$inst_course_id,$newattr{'courseowner'}); + my $instsec = $1; + my $inst_course_id = $newattr{'coursecode'}.$instsec; + my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$inst_course_id,$newattr{'courseowner'},$coowners); unless ($outcome eq 'ok') { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for ").$newattr{'coursecode'}.&mt(" - section $1 for the following reason: $outcome.
          "); + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for "[_2]" - section: [_3] for the following reason: "[_4]".',$description,$newattr{'coursecode'},$instsec,$outcome)); } } else { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for ").$newattr{'coursecode'}.&mt(" - section $sec because this is not a valid section entry.
          "); + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for "[_2]" - section: [_3], because this is not a valid section entry.',$description,$newattr{'coursecode'},$sec)); } } } } else { - $warning .= &mt("As no section numbers are currently listed for LON-CAPA course: ").$description.&mt(", automated enrollment will not occur for any sections of coursecode: ").$newattr{'coursecode'}."
          "; + push(@warnings,&mt('As no section numbers are currently listed for "[_1]", automated enrollment will not occur for any sections of institutional course code: "[_2]".',$description,$newattr{'coursecode'})); } if ( (@xlists > 0) && ($changed{'owner'}) ) { foreach my $xlist (@xlists) { if ($xlist =~ m/^(.+):/) { - my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$1,$newattr{'courseowner'}); + my $instxlist = $1; + my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$instxlist,$newattr{'courseowner'},$coowners); unless ($outcome eq 'ok') { - $warning .= &mt("If automatic enrollment is enabled for LON-CAPA course: ").$description.&mt(", automated enrollment may fail for crosslisted class: ").$1.&mt(" for the following reason: $outcome.
          "); + push(@warnings,&mt('If automatic enrollment is enabled for "[_1]", automated enrollment may fail for crosslisted class "[_2]" for the following reason: "[_3]".',$description,$instxlist,$outcome)); } } } @@ -768,7 +1632,17 @@ sub modify_course { } } else { foreach my $attr (@modifiable_params) { - $nochgresponse .= "
        • $longtype{$attr} ".&mt("still set to")." \"".$currattr{$attr}."\".
        • "; + my $shown = $currattr{$attr}; + if ($attr eq 'selfenrollmgrdc') { + $shown = &selfenroll_config_status(\@currmgrdc,$selfenrolltitles); + } elsif ($attr eq 'selfenrollmgrcc') { + $shown = &selfenroll_config_status(\@currmgrcc,$selfenrolltitles); + } elsif (($attr eq 'defaultcredits') && ($shown eq '')) { + $shown = &mt('None'); + } elsif (($attr eq 'mysqltables') && ($shown eq '')) { + $shown = &mt('domain default'); + } + $nochgresponse .= '
        • '.&mt('[_1] still set to: [_2]',$longtype{$attr},$shown).'
        • '; } } @@ -778,99 +1652,468 @@ sub modify_course { if (@nochanges > 0) { $nochgresponse .= "


        "; } - unless ($warning eq '') { - $warning = &mt("The following warning messages were generated as a result of applying the changes you specified to course settings that can affect the automated enrollment process:

        ").$warning; + my ($warning,$numwarnings); + my $numwarnings = scalar(@warnings); + if ($numwarnings) { + $warning = &mt('The following [quant,_1,warning was,warnings were] generated when applying your changes to automated enrollment:',$numwarnings).'

          '; + foreach my $warn (@warnings) { + $warning .= '
        • '.$warn.'
        • '; + } + $warning .= '

        '; } - if ($response eq '') { - $reply = $chgresponse.$nochgresponse.$warning; - } else { + if ($response) { $reply = $response; + } else { + $reply = $chgresponse.$nochgresponse.$warning; } - &print_header($r,$cdesc); - $reply = '

        '. - &mt('Result of Changes to Automated Enrollment Settings.'). - '

        '."\n". - '
        '.$reply.'
        '. + &print_header($r,$type); + my $mainheader = &modifiable_only_title($type); + $reply = '

        '.$mainheader.' '.$cdesc.'

        '."\n". + '

        '.$reply.'

        '."\n". '
        '. - &hidden_form_elements(). - ''.&mt('Back to options page').' -
        '; + &hidden_form_elements(); + my @actions = + (''. + &mt('Pick another action').''); + if ($numwarnings) { + my $newrole = $ccrole.'./'.$cdom.'/'.$cnum; + my $escuri = &HTML::Entities::encode('/adm/roles?selectrole=1&'.$newrole. + '=1&destinationurl=/adm/populate','&<>"'); + + push(@actions, ''. + &mt('Go to Automated Enrollment Manager for course').''); + } + $reply .= &Apache::lonhtmlcommon::actionbox(\@actions).''; $r->print($reply); return; } +sub selfenroll_config_status { + my ($items,$selfenrolltitles) = @_; + my $shown; + if ((ref($items) eq 'ARRAY') && (ref($selfenrolltitles) eq 'HASH')) { + if (@{$items} > 0) { + $shown = '
          '; + foreach my $item (@{$items}) { + $shown .= '
        • '.$selfenrolltitles->{$item}.'
        • '; + } + $shown .= '
        '; + } else { + $shown = &mt('None'); + } + } + return $shown; +} + +sub update_coowners { + my ($cdom,$cnum,$chome,$settings,$newattr) = @_; + return unless ((ref($settings) eq 'HASH') && (ref($newattr) eq 'HASH')); + my %designhash = &Apache::loncommon::get_domainconf($cdom); + my (%cchash,$autocoowners); + if ($designhash{$cdom.'.autoassign.co-owners'}) { + $autocoowners = 1; + %cchash = &Apache::lonnet::get_my_roles($cnum,$cdom,undef,undef,['cc']); + } + if ($settings->{'internal.courseowner'} ne $newattr->{'courseowner'}) { + my $oldowner_to_coowner; + my @types = ('co-owners'); + if (($newattr->{'coursecode'}) && ($autocoowners)) { + my $oldowner = $settings->{'internal.courseowner'}; + if ($cchash{$oldowner.':cc'}) { + my ($result,$desc) = &Apache::lonnet::auto_validate_instcode($cnum,$cdom,$newattr->{'coursecode'},$oldowner); + if ($result eq 'valid') { + if ($settings->{'internal.co-owner'}) { + my @current = split(',',$settings->{'internal.co-owners'}); + unless (grep(/^\Q$oldowner\E$/,@current)) { + $oldowner_to_coowner = 1; + } + } else { + $oldowner_to_coowner = 1; + } + } + } + } else { + push(@types,'pendingco-owners'); + } + foreach my $type (@types) { + if ($settings->{'internal.'.$type}) { + my @current = split(',',$settings->{'internal.'.$type}); + my $newowner = $newattr->{'courseowner'}; + my @newvalues = (); + if (($newowner ne '') && (grep(/^\Q$newowner\E$/,@current))) { + foreach my $person (@current) { + unless ($person eq $newowner) { + push(@newvalues,$person); + } + } + } else { + @newvalues = @current; + } + if ($oldowner_to_coowner) { + push(@newvalues,$settings->{'internal.courseowner'}); + @newvalues = sort(@newvalues); + } + my $newownstr = join(',',@newvalues); + if ($newownstr ne $settings->{'internal.'.$type}) { + if ($type eq 'co-owners') { + my $deleted = ''; + unless (@newvalues) { + $deleted = 1; + } + &Apache::lonnet::store_coowners($cdom,$cnum,$chome, + $deleted,@newvalues); + } else { + my $pendingcoowners; + my $cid = $cdom.'_'.$cnum; + if (@newvalues) { + $pendingcoowners = join(',',@newvalues); + my %pendinghash = ( + 'internal.pendingco-owners' => $pendingcoowners, + ); + my $putresult = &Apache::lonnet::put('environment',\%pendinghash,$cdom,$cnum); + if ($putresult eq 'ok') { + if ($env{'course.'.$cid.'.num'} eq $cnum) { + &Apache::lonnet::appenv({'course.'.$cid.'.internal.pendingco-owners' => $pendingcoowners}); + } + } + } else { + my $delresult = &Apache::lonnet::del('environment',['internal.pendingco-owners'],$cdom,$cnum); + if ($delresult eq 'ok') { + if ($env{'course.'.$cid.'.internal.pendingco-owners'}) { + &Apache::lonnet::delenv('course.'.$cid.'.internal.pendingco-owners'); + } + } + } + } + } elsif ($oldowner_to_coowner) { + &Apache::lonnet::store_coowners($cdom,$cnum,$chome,'', + $settings->{'internal.courseowner'}); + + } + } elsif ($oldowner_to_coowner) { + &Apache::lonnet::store_coowners($cdom,$cnum,$chome,'', + $settings->{'internal.courseowner'}); + } + } + } + if ($settings->{'internal.coursecode'} ne $newattr->{'coursecode'}) { + if ($newattr->{'coursecode'} ne '') { + my %designhash = &Apache::loncommon::get_domainconf($cdom); + if ($designhash{$cdom.'.autoassign.co-owners'}) { + my @newcoowners = (); + if ($settings->{'internal.co-owners'}) { + my @currcoown = split(',',$settings->{'internal.co-owners'}); + my ($updatecoowners,$delcoowners); + foreach my $person (@currcoown) { + my ($result,$desc) = &Apache::lonnet::auto_validate_instcode($cnum,$cdom,$newattr->{'coursecode'},$person); + if ($result eq 'valid') { + push(@newcoowners,$person); + } + } + foreach my $item (sort(keys(%cchash))) { + my ($uname,$udom,$urole) = split(':',$item); + next if ($uname.':'.$udom eq $newattr->{'courseowner'}); + unless (grep(/^\Q$uname\E:\Q$udom\E$/,@newcoowners)) { + my ($result,$desc) = &Apache::lonnet::auto_validate_instcode($cnum,$cdom,$newattr->{'coursecode'},$uname.':'.$udom); + if ($result eq 'valid') { + push(@newcoowners,$uname.':'.$udom); + } + } + } + if (@newcoowners) { + my $coowners = join(',',sort(@newcoowners)); + unless ($coowners eq $settings->{'internal.co-owners'}) { + $updatecoowners = 1; + } + } else { + $delcoowners = 1; + } + if ($updatecoowners || $delcoowners) { + &Apache::lonnet::store_coowners($cdom,$cnum,$chome, + $delcoowners,@newcoowners); + } + } else { + foreach my $item (sort(keys(%cchash))) { + my ($uname,$udom,$urole) = split(':',$item); + push(@newcoowners,$uname.':'.$udom); + } + if (@newcoowners) { + &Apache::lonnet::store_coowners($cdom,$cnum,$chome,'', + @newcoowners); + } + } + } + } + } + return; +} + sub modify_quota { - my ($r,$cdom,$cnum,$cdesc,$domdesc) = @_; - &print_header($r,$cdesc); + my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_; + &print_header($r,$type); + my $lctype = lc($type); + my $headline = &mt("Disk space quotas for $lctype: [_1]", + ''.$cdesc.''); + $r->print('
        '."\n". + '

        '.$headline.'

        '); + my %oldsettings = &Apache::lonnet::get('environment',['internal.coursequota','internal.uploadquota'],$cdom,$cnum); + my %staticdefaults = ( + coursequota => 20, + uploadquota => 500, + ); + my %default; + $default{'coursequota'} = $staticdefaults{'coursequota'}; + my %domdefs = &Apache::lonnet::get_domain_defaults($cdom); + $default{'uploadquota'} = $domdefs{'uploadquota'}; + if ($default{'uploadquota'} eq '') { + $default{'uploadquota'} = $staticdefaults{'uploadquota'}; + } + my (%cenv,%showresult); + foreach my $item ('coursequota','uploadquota') { + if ($env{'form.'.$item} ne '') { + my $newquota = $env{'form.'.$item}; + if ($newquota =~ /^\s*(\d+\.?\d*|\.\d+)\s*$/) { + $newquota = $1; + if ($oldsettings{'internal.'.$item} == $newquota) { + if ($item eq 'coursequota') { + $r->print(&mt('The disk space allocated for group portfolio files remains unchanged as [_1] MB.',$newquota).'
        '); + } else { + $r->print(&mt('The disk space allocated for files uploaded via the Content Editor remains unchanged as [_1] MB.',$newquota).'
        '); + } + } else { + $cenv{'internal.'.$item} = $newquota; + $showresult{$item} = 1; + } + } else { + if ($item eq 'coursequota') { + $r->print(&mt('The proposed group portfolio quota contained invalid characters, so the quota is unchanged.').'
        '); + } else { + $r->print(&mt('The proposed quota for content uploaded via the Content Editor contained invalid characters, so the quota is unchanged.').'
        '); - my %lt = &Apache::lonlocal::texthash( - 'back' => 'Back to options page', - ); - $r->print(' - -

        '.&mt('Disk space for storage of group portfolio for [_1]',$cdesc). - '

        '); - my %oldsettings = &Apache::lonnet::get('environment',['internal.coursequota'],$cdom,$cnum); - my $defaultquota = 20; - if ($env{'form.coursequota'} ne '') { - my $newquota = $env{'form.coursequota'}; - if ($newquota =~ /^\s*(\d+\.?\d*|\.\d+)\s*$/) { - $newquota = $1; - if ($oldsettings{'internal.coursequota'} eq $env{'form.coursequota'}) { - $r->print(&mt('The disk space allocated for group portfolio remains unchanged as ').$env{'form.coursequota'}.' Mb'); + } + } + } + } + if (keys(%cenv)) { + my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom, + $cnum); + foreach my $key (sort(keys(%showresult))) { + if (($oldsettings{'internal.'.$key} eq '') && + ($env{'form.'.$key} == $default{$key})) { + if ($key eq 'uploadquota') { + if ($type eq 'Community') { + $r->print(&mt('The disk space allocated for files uploaded to this community via the Content Editor is the default quota for this domain: [_1] MB.', + $default{$key}).'
        '); + } else { + $r->print(&mt('The disk space allocated for files uploaded to this course via the Content Editor is the default quota for this domain: [_1] MB.', + $default{$key}).'
        '); + } + } else { + if ($type eq 'Community') { + $r->print(&mt('The disk space allocated for group portfolio files in this community is the default quota for this domain: [_1] MB.', + $default{$key}).'
        '); + } else { + $r->print(&mt('The disk space allocated for group portfolio files in this course is the default quota for this domain: [_1] MB.', + $default{$key}).'
        '); + } + } + delete($showresult{$key}); + } + } + if ($putreply eq 'ok') { + my %updatedsettings = &Apache::lonnet::get('environment',['internal.coursequota','internal.uploadquota'],$cdom,$cnum); + if ($showresult{'coursequota'}) { + $r->print(&mt('The disk space allocated for group portfolio files is now: [_1] MB.', + ''.$updatedsettings{'internal.coursequota'}.'').'
        '); + my $usage = &Apache::longroup::sum_quotas($cdom.'_'.$cnum); + if ($usage >= $updatedsettings{'internal.coursequota'}) { + my $newoverquota; + if ($usage < $oldsettings{'internal.coursequota'}) { + $newoverquota = 'now'; + } + $r->print('

        '); + if ($type eq 'Community') { + $r->print(&mt("Disk usage $newoverquota exceeds the quota for this community.").' '. + &mt('Upload of new portfolio files and assignment of a non-zero MB quota to new groups in the community will not be possible until some files have been deleted, and total usage is below community quota.')); + } else { + $r->print(&mt("Disk usage $newoverquota exceeds the quota for this course.").' '. + &mt('Upload of new portfolio files and assignment of a non-zero MB quota to new groups in the course will not be possible until some files have been deleted, and total usage is below course quota.')); + } + $r->print('

        '); + } + } + if ($showresult{'uploadquota'}) { + $r->print(&mt('The disk space allocated for content uploaded directly via the Content Editor is now: [_1] MB.', + ''.$updatedsettings{'internal.uploadquota'}.'').'
        '); + } + } else { + $r->print(&mt('An error occurred storing the quota(s) for group portfolio files and/or uploaded content: '). + $putreply); + } + } + $r->print('

        '. + ''. + &mt('Pick another action').''); + $r->print(&hidden_form_elements().'

        '); + return; +} + +sub modify_anonsurvey_threshold { + my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_; + &print_header($r,$type); + $r->print('
        '."\n". + '

        '.&mt('Responder threshold required for display of anonymous survey submissions:'). + ' '.$cdesc.'


        '); + my %oldsettings = &Apache::lonnet::get('environment',['internal.anonsurvey_threshold'],$cdom,$cnum); + my %domconfig = + &Apache::lonnet::get_dom('configuration',['coursedefaults'],$cdom); + my $defaultthreshold; + if (ref($domconfig{'coursedefaults'}) eq 'HASH') { + $defaultthreshold = $domconfig{'coursedefaults'}{'anonsurvey_threshold'}; + if ($defaultthreshold eq '') { + $defaultthreshold = 10; + } + } else { + $defaultthreshold = 10; + } + if ($env{'form.threshold'} eq '') { + $r->print(&mt('The proposed responder threshold for display of anonymous survey submissions was blank, so the threshold is unchanged.')); + } else { + my $newthreshold = $env{'form.threshold'}; + if ($newthreshold =~ /^\s*(\d+)\s*$/) { + $newthreshold = $1; + if ($oldsettings{'internal.anonsurvey_threshold'} eq $env{'form.threshold'}) { + $r->print(&mt('Responder threshold for anonymous survey submissions display remains unchanged: [_1].',$env{'form.threshold'})); } else { my %cenv = ( - 'internal.coursequota' => $env{'form.coursequota'}, + 'internal.anonsurvey_threshold' => $env{'form.threshold'}, ); my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom, $cnum); - if (($oldsettings{'internal.coursequota'} eq '') && - ($env{'form.coursequota'} == $defaultquota)) { - $r->print(&mt('The disk space allocated for group portfolio in this course is the default quota for this domain:').' '.$defaultquota.' Mb'); + if ($putreply eq 'ok') { + if ($env{'course.'.$cdom.'_'.$cnum.'.description'} ne '') { + &Apache::lonnet::appenv( + {'course.'.$cdom.'_'.$cnum.'.internal.anonsurvey_threshold' => $env{'form.threshold'}}); + } + } + if (($oldsettings{'internal.anonsurvey_threshold'} eq '') && + ($env{'form.threshold'} == $defaultthreshold)) { + $r->print(&mt('The responder threshold for display of anonymous survey submissions is the default for this domain: [_1].',$defaultthreshold)); } else { if ($putreply eq 'ok') { - my %updatedsettings = &Apache::lonnet::get('environment',['internal.coursequota'],$cdom,$cnum); - $r->print(&mt('The disk space allocated for group portfolio is now:').' '.$updatedsettings{'internal.coursequota'}.' Mb.'); - my $usage = &Apache::longroup::sum_quotas($cdom.'_'.$cnum); - if ($usage >= $updatedsettings{'internal.coursequota'}) { - my $newoverquota; - if ($usage < $oldsettings{'internal.coursequota'}) { - $newoverquota = 'now'; - } - $r->print('

        '. - &mt('Disk usage [_1] exceeds the quota for this course.',$newoverquota).' '. - &mt('Upload of new portfolio files and assignment of a non-zero Mb quota to new groups in the course will not be possible until some files have been deleted, and total usage is below course quota.')); - } + my %updatedsettings = &Apache::lonnet::get('environment',['internal.anonsurvey_threshold'],$cdom,$cnum); + $r->print(&mt('The responder threshold for display of anonymous survey submissions is now: [_1].',''.$updatedsettings{'internal.anonsurvey_threshold'}.'')); } else { - $r->print(&mt('There was an error')); + $r->print(&mt('An error occurred storing the responder threshold for anonymous submissions display: '). + $putreply); } } } } else { - $r->print(&mt('The new quota requested contained invalid characters, so the quota is unchanged.')); + $r->print(&mt('The proposed responder threshold for display of anonymous submissions contained invalid characters, so the threshold is unchanged.')); } } - $r->print(qq|

        -$lt{'back'}|); + $r->print('

        '. + ''. + &mt('Pick another action').'

        '); $r->print(&hidden_form_elements().'
        '); return; } -sub modify_catsettings { - my ($r,$cdom,$cnum,$cdesc,$domdesc) = @_; - &print_header($r,$cdesc); +sub modify_postsubmit_config { + my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_; + &print_header($r,$type); my %lt = &Apache::lonlocal::texthash( - 'back' => 'Back to options page', - ); - my %desc = &Apache::lonlocal::texthash( - 'hidefromcat' => 'Excluded from course catalog', - 'categories' => 'Assigned categories for this course', - ); + subb => 'Submit button behavior after student makes a submission:', + unch => 'Post submission behavior of the Submit button is unchanged.', + erro => 'An error occurred when saving your proposed changes.', + inva => 'An invalid response was recorded.', + pick => 'Pick another action', + ); + $r->print('
        '."\n". + '

        '.$lt{'subb'}.' ('.$cdesc.')


        '); + my %oldsettings = + &Apache::lonnet::get('environment',['internal.postsubmit','internal.postsubtimeout','internal.coursecode','internal.textbook'],$cdom,$cnum); + my $postsubmit = $env{'form.postsubmit'}; + if ($postsubmit eq '1') { + my $postsubtimeout = $env{'form.postsubtimeout'}; + $postsubtimeout =~ s/[^\d\.]+//g; + if (($oldsettings{'internal.postsubmit'} eq $postsubmit) && ($oldsettings{'internal.postsubtimeout'} eq $postsubtimeout)) { + $r->print($lt{'unch'}); + } else { + my %cenv = ( + 'internal.postsubmit' => $postsubmit, + ); + if ($postsubtimeout eq '') { + my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom,$cnum); + if ($putreply eq 'ok') { + my $defaulttimeout = &domain_postsubtimeout($cdom,$type,\%oldsettings); + $r->print(&mt('The proposed duration for disabling the Submit button post-submission was blank, so the domain default of [quant,_1,second] will be used.',$defaulttimeout)); + if (exists($oldsettings{'internal.postsubtimeout'})) { + &Apache::lonnet::del('environment',['internal.postsubtimeout'],$cdom,$cnum); + } + } else { + $r->print($lt{'erro'}); + } + } else { + $cenv{'internal.postsubtimeout'} = $postsubtimeout; + my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom,$cnum); + if ($putreply eq 'ok') { + if ($postsubtimeout eq '0') { + $r->print(&mt('Submit button will be disabled after student submission until page is reloaded.')); + } else { + $r->print(&mt('Submit button will be disabled after student submission for [quant,_1,second].',$postsubtimeout)); + } + } else { + $r->print($lt{'erro'}); + } + } + } + } elsif ($postsubmit eq '0') { + if ($oldsettings{'internal.postsubmit'} eq $postsubmit) { + $r->print($lt{'unch'}); + } else { + if (exists($oldsettings{'internal.postsubtimeout'})) { + &Apache::lonnet::del('environment',['internal.postsubtimeout'],$cdom,$cnum); + } + my %cenv = ( + 'internal.postsubmit' => $postsubmit, + ); + my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom,$cnum); + if ($putreply eq 'ok') { + $r->print(&mt('Submit button will not be disabled after student submission')); + } else { + $r->print($lt{'erro'}); + } + } + } else { + $r->print($lt{'inva'}.' '.$lt{'unch'}); + } + $r->print('

        '. + ''. + &mt('Pick another action').'

        '); + $r->print(&hidden_form_elements().'
        '); + return; +} + +sub modify_catsettings { + my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_; + &print_header($r,$type); + my ($ccrole,%desc); + if ($type eq 'Community') { + $desc{'hidefromcat'} = &mt('Excluded from community catalog'); + $desc{'categories'} = &mt('Assigned categories for this community'); + $ccrole = 'co'; + } else { + $desc{'hidefromcat'} = &mt('Excluded from course catalog'); + $desc{'categories'} = &mt('Assigned categories for this course'); + $ccrole = 'cc'; + } $r->print('

        '.&mt('Category settings').'

        '); my %domconf = &Apache::lonnet::get_dom('configuration',['coursecategories'],$cdom); - my @cat_params = &catalog_settable($domconf{'coursecategories'}); + my @cat_params = &catalog_settable($domconf{'coursecategories'},$type); if (@cat_params > 0) { my (%cenv,@changes,@nochanges); my %currsettings = @@ -939,6 +2182,13 @@ sub modify_catsettings { if (@changes > 0) { my $putreply = &Apache::lonnet::put('environment',\%cenv,$cdom,$cnum); if ($putreply eq 'ok') { + if ($env{'course.'.$cdom.'_'.$cnum.'.description'} ne '') { + my %newenvhash; + foreach my $item (@changes) { + $newenvhash{'course.'.$cdom.'_'.$cnum.'.'.$item} = $cenv{$item}; + } + &Apache::lonnet::appenv(\%newenvhash); + } my %crsinfo = &Apache::lonnet::courseiddump($cdom,'.',1,'.','.', $cnum,undef,undef,'.'); if (ref($crsinfo{$env{'form.pickedcourse'}}) eq 'HASH') { @@ -951,39 +2201,46 @@ sub modify_catsettings { my $chome = &Apache::lonnet::homeserver($cnum,$cdom); my $putres = &Apache::lonnet::courseidput($cdom,\%crsinfo,$chome,'notime'); } - $r->print(&mt('The following changes occurred').'
          '); + $r->print(&mt('The following changes occurred:').'
            '); foreach my $item (@changes) { - $r->print('
          • '.&mt('[_1] now set to [_2]',$desc{$item},$showitem{$item}).'
          • '); + $r->print('
          • '.&mt('[_1] now set to: [_2]',$desc{$item},$showitem{$item}).'
          • '); } $r->print('

          '); } } if (@nochanges > 0) { - $r->print(&mt('The following were unchanged').'
            '); + $r->print(&mt('The following were unchanged:').'
              '); foreach my $item (@nochanges) { - $r->print('
            • '.&mt('[_1] still set to [_2]',$desc{$item},$showitem{$item}).'
            • '); + $r->print('
            • '.&mt('[_1] still set to: [_2]',$desc{$item},$showitem{$item}).'
            • '); } $r->print('
            '); } } } else { - $r->print(&mt('Category settings for courses in this domain should be modified in course context (via "Set Course Environment").').'
            '); + my $newrole = $ccrole.'./'.$cdom.'/'.$cnum; + my $escuri = &HTML::Entities::encode('/adm/roles?selectrole=1&'.$newrole. + '=1&destinationurl=/adm/courseprefs','&<>"'); + if ($type eq 'Community') { + $r->print(&mt('Category settings for communities in this domain should be modified in community context (via "[_1]Community Configuration[_2]").','','').'
            '); + } else { + $r->print(&mt('Category settings for courses in this domain should be modified in course context (via "[_1]Course Configuration[_2]").','','').'
            '); + } } $r->print('
            '."\n". ''. - $lt{'back'}.''); + &mt('Pick another action').''); $r->print(&hidden_form_elements().''); return; } sub print_header { - my ($r,$cdesc,$javascript_validations) = @_; + my ($r,$type,$javascript_validations) = @_; my $phase = "start"; if ( exists($env{'form.phase'}) ) { $phase = $env{'form.phase'}; } my $js = qq| - + |; if ($phase eq 'setparms') { - $js .= qq| - -|; + $js .= $javascript_validations; } elsif ($phase eq 'courselist') { - $js .= qq| - -|; + +ENDJS } elsif ($phase eq 'setquota') { - $js .= <<'ENDSCRIPT'; - + ENDSCRIPT + } my $starthash; - if ($env{'form.phase'} eq 'ccrole') { + if ($env{'form.phase'} eq 'adhocrole') { $starthash = { - add_entries => {'onload' => "javascript:document.ccrole.submit();"}, + add_entries => {'onload' => "javascript:document.adhocrole.submit();"}, + }; + } elsif ($phase eq 'courselist') { + $starthash = { + add_entries => {'onload' => "hide_searching(); courseSet(document.filterpicker.official, 'load');"}, }; } - $r->print(&Apache::loncommon::start_page('View/Modify Course Settings', - $js,$starthash)); - my $bread_text = "View/Modify Courses"; - if ($cdesc ne '') { - $bread_text = &mt('Course Settings: [_1]',$cdesc); - my $no_mt = 1; - $r->print(&Apache::lonhtmlcommon::breadcrumbs($bread_text,undef,undef, - undef,undef,$no_mt)); + $r->print(&Apache::loncommon::start_page('View/Modify Course/Community Settings', + &Apache::lonhtmlcommon::scripttag($js), + $starthash)); + my $bread_text = "View/Modify Courses/Communities"; + if ($type eq 'Community') { + $bread_text = 'Community Settings'; + } elsif ($type eq 'Placement') { + $bread_text = 'Placement Test Settings'; } else { - $r->print(&Apache::lonhtmlcommon::breadcrumbs($bread_text)); + $bread_text = 'Course Settings'; } + $r->print(&Apache::lonhtmlcommon::breadcrumbs($bread_text)); return; } @@ -1051,42 +2421,50 @@ sub print_footer { } sub check_course { - my ($r,$dom,$domdesc) = @_; - my ($ok_course,$description,$instcode,$owner); - my %args = ( - one_time => 1, - ); - my %coursehash = - &Apache::lonnet::coursedescription($env{'form.pickedcourse'},\%args); - my $cnum = $coursehash{'num'}; - my $cdom = $coursehash{'domain'}; - if ($cdom eq $dom) { - my $description; - my %courseIDs = &Apache::lonnet::courseiddump($cdom,'.',1,'.','.', - $cnum,undef,undef,'.'); - if (keys(%courseIDs) > 0) { - $ok_course = 'ok'; - my ($instcode,$owner); - if (ref($courseIDs{$cdom.'_'.$cnum}) eq 'HASH') { - $description = $courseIDs{$cdom.'_'.$cnum}{'description'}; - $instcode = $courseIDs{$cdom.'_'.$cnum}{'inst_code'}; - $owner = $courseIDs{$cdom.'_'.$cnum}{'owner'}; - } else { - ($description,$instcode,$owner) = - split(/:/,$courseIDs{$cdom.'_'.$cnum}); - } - $description = &unescape($description); - $instcode = &unescape($instcode); - if ($instcode) { - $description .= " ($instcode)"; + my ($dom,$domdesc) = @_; + my ($ok_course,$description,$instcode); + my %coursehash; + if ($env{'form.pickedcourse'} =~ /^$match_domain\_$match_courseid$/) { + my %args; + unless ($env{'course.'.$env{'form.pickedcourse'}.'.description'}) { + %args = ( + 'one_time' => 1, + 'freshen_cache' => 1, + ); + } + %coursehash = + &Apache::lonnet::coursedescription($env{'form.pickedcourse'},\%args); + my $cnum = $coursehash{'num'}; + my $cdom = $coursehash{'domain'}; + $description = $coursehash{'description'}; + $instcode = $coursehash{'internal.coursecode'}; + if ($instcode) { + $description .= " ($instcode)"; + } + if (($cdom eq $dom) && ($cnum =~ /^$match_courseid$/)) { + my %courseIDs = &Apache::lonnet::courseiddump($cdom,'.',1,'.','.', + $cnum,undef,undef,'.'); + if ($courseIDs{$cdom.'_'.$cnum}) { + $ok_course = 'ok'; } - return ($ok_course,$description); } } + return ($ok_course,$description,\%coursehash); } sub course_settings_descrip { - my %longtype = &Apache::lonlocal::texthash( + my ($type) = @_; + my %longtype; + if ($type eq 'Community') { + %longtype = &Apache::lonlocal::texthash( + 'courseowner' => "Username:domain of community owner", + 'co-owners' => "Username:domain of each co-owner", + 'selfenrollmgrdc' => "Community-specific self-enrollment configuration by Domain Coordinator", + 'selfenrollmgrcc' => "Community-specific self-enrollment configuration by Community personnel", + 'mysqltables' => '"Temporary" student performance tables lifetime (seconds)', + ); + } else { + %longtype = &Apache::lonlocal::texthash( 'authtype' => 'Default authentication method', 'autharg' => 'Default authentication parameter', 'autoadds' => 'Automated adds', @@ -1097,10 +2475,18 @@ sub course_settings_descrip { 'default_enrollment_end_date' => 'Date of last student access', 'coursecode' => 'Official course code', 'courseowner' => "Username:domain of course owner", + 'co-owners' => "Username:domain of each co-owner", 'notifylist' => 'Course Coordinators to be notified of enrollment changes', - 'sectionnums' => 'Course section number(:LON-CAPA section)', - 'crosslistings' => 'Crosslisted class(:LON-CAPA section)', - ); + 'sectionnums' => 'Course section number:LON-CAPA section', + 'crosslistings' => 'Crosslisted class:LON-CAPA section', + 'defaultcredits' => 'Credits', + 'autodropfailsafe' => "Failsafe section enrollment count", + 'selfenrollmgrdc' => "Course-specific self-enrollment configuration by Domain Coordinator", + 'selfenrollmgrcc' => "Course-specific self-enrollment configuration by Course personnel", + 'mysqltables' => '"Temporary" student performance tables lifetime (seconds)', + 'nopasswdchg' => 'Disable changing password for users with student role by course owner', + ); + } return %longtype; } @@ -1108,11 +2494,82 @@ sub hidden_form_elements { my $hidden_elements = &Apache::lonhtmlcommon::echo_form_input(['gosearch','updater','coursecode', 'prevphase','numlocalcc','courseowner','login','coursequota','intarg', - 'locarg','krbarg','krbver','counter','hidefromcat','usecategory'])."\n". + 'locarg','krbarg','krbver','counter','hidefromcat','usecategory', + 'threshold','postsubmit','postsubtimeout','defaultcredits','uploadquota', + 'selfenrollmgrdc','selfenrollmgrcc','action','state','currsec_st', + 'sections','newsec','mysqltables','nopasswdchg'], + ['^selfenrollmgr_','^selfenroll_'])."\n". ''; return $hidden_elements; } +sub showcredits { + my ($dom) = @_; + my %domdefaults = &Apache::lonnet::get_domain_defaults($dom); + if ($domdefaults{'officialcredits'} || $domdefaults{'unofficialcredits'} || $domdefaults{'textbookcredits'}) { + return 1; + } +} + +sub get_permission { + my ($dom) = @_; + my ($allowed,%permission); + my %passwdconf = &Apache::lonnet::get_passwdconf($dom); + if (&Apache::lonnet::allowed('ccc',$dom)) { + $allowed = 1; + %permission = ( + setquota => 'edit', + processquota => 'edit', + setanon => 'edit', + processthreshold => 'edit', + setpostsubmit => 'edit', + processpostsubmit => 'edit', + viewparms => 'view', + setparms => 'edit', + processparms => 'edit', + catsettings => 'edit', + processcat => 'edit', + selfenroll => 'edit', + adhocrole => 'coord', + ); + if ($passwdconf{'crsownerchg'}) { + $permission{passwdchg} = 'edit'; + } + } elsif (&Apache::lonnet::allowed('rar',$dom)) { + $allowed = 1; + %permission = ( + setquota => 'view', + viewparms => 'view', + setanon => 'view', + setpostsubmit => 'view', + setparms => 'view', + catsettings => 'view', + selfenroll => 'view', + adhocrole => 'custom', + ); + if ($passwdconf{'crsownerchg'}) { + $permission{passwdchg} = 'view'; + } + } + return ($allowed,\%permission); +} + +sub devalidate_remote_instcats { + if ($modified_dom ne '') { + my %servers = &Apache::lonnet::internet_dom_servers($modified_dom); + my %thismachine; + map { $thismachine{$_} = 1; } &Apache::lonnet::current_machine_ids(); + if (keys(%servers)) { + foreach my $server (keys(%servers)) { + next if ($thismachine{$server}); + &Apache::lonnet::remote_devalidate_cache($server,['instcats:'.$modified_dom]); + } + } + $modified_dom = ''; + } + return; +} + sub handler { my $r = shift; if ($r->header_only) { @@ -1120,9 +2577,14 @@ sub handler { $r->send_http_header; return OK; } + + $registered_cleanup=0; + $modified_dom = ''; + my $dom = $env{'request.role.domain'}; my $domdesc = &Apache::lonnet::domain($dom,'description'); - if (&Apache::lonnet::allowed('ccc',$dom)) { + my ($allowed,$permission) = &get_permission($dom); + if ($allowed) { &Apache::loncommon::content_type($r,'text/html'); $r->send_http_header; @@ -1135,66 +2597,126 @@ sub handler { if ($phase eq '') { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"/adm/modifycourse", - text=>"Course search"}); + text=>"Course/Community search"}); &print_course_search_page($r,$dom,$domdesc); } else { my $firstform = $phase; if ($phase eq 'courselist') { $firstform = 'filterpicker'; + } + my $choose_text; + my $type = $env{'form.type'}; + if ($type eq '') { + $type = 'Course'; + } + if ($type eq 'Community') { + $choose_text = "Choose a community"; + } elsif ($type eq 'Placement') { + $choose_text = "Choose a placement test"; + } else { + $choose_text = "Choose a course"; } &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$firstform,'')", - text=>"Course search"}, + text=>"Course/Community search"}, {href=>"javascript:changePage(document.$phase,'courselist')", - text=>"Choose a course"}); + text=>$choose_text}); if ($phase eq 'courselist') { - &print_course_selection_page($r,$dom,$domdesc); + &print_course_selection_page($r,$dom,$domdesc,$permission); } else { - my ($checked,$cdesc) = &check_course($r,$dom,$domdesc); - my $type = $env{'form.type'}; - if ($type eq '') { - $type = 'Course'; - } + my ($checked,$cdesc,$coursehash) = &check_course($dom,$domdesc); if ($checked eq 'ok') { + my $enter_text; + if ($type eq 'Community') { + $enter_text = 'Enter community'; + } elsif ($type eq 'Placement') { + $enter_text = 'Enter placement test'; + } else { + $enter_text = 'Enter course'; + } if ($phase eq 'menu') { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'menu')", text=>"Pick action"}); - &print_modification_menu($r,$cdesc,$domdesc,$dom); - } elsif ($phase eq 'ccrole') { + &print_modification_menu($r,$cdesc,$domdesc,$dom,$type, + $env{'form.pickedcourse'},$coursehash, + $permission); + } elsif ($phase eq 'adhocrole') { &Apache::lonhtmlcommon::add_breadcrumb - ({href=>"javascript:changePage(document.$phase,'ccrole')", - text=>"Enter course"}); - &print_ccrole_selected($r,$cdesc,$domdesc); + ({href=>"javascript:changePage(document.$phase,'adhocrole')", + text=>$enter_text}); + &print_adhocrole_selected($r,$type,$permission); } else { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'menu')", text=>"Pick action"}); my ($cdom,$cnum) = split(/_/,$env{'form.pickedcourse'}); - if ($phase eq 'setquota') { + my ($readonly,$linktext); + if ($permission->{$phase} eq 'view') { + $readonly = 1; + } + if (($phase eq 'setquota') && ($permission->{'setquota'})) { + if ($permission->{'setquota'} eq 'view') { + $linktext = 'Set quota'; + } else { + $linktext = 'Display quota'; + } &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'$phase')", - text=>"Set quota"}); - &print_setquota($r,$cdom,$cnum,$cdesc,$type); - } elsif ($phase eq 'processquota') { + text=>$linktext}); + &print_setquota($r,$cdom,$cnum,$cdesc,$type,$readonly); + } elsif (($phase eq 'processquota') && ($permission->{'processquota'})) { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'setquota')", text=>"Set quota"}); &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'$phase')", text=>"Result"}); - &modify_quota($r,$cdom,$cnum,$cdesc,$domdesc); - } elsif ($phase eq 'viewparms') { + &modify_quota($r,$cdom,$cnum,$cdesc,$domdesc,$type); + } elsif (($phase eq 'setanon') && ($permission->{'setanon'})) { + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'$phase')", + text=>"Threshold for anonymous submissions display"}); + &print_set_anonsurvey_threshold($r,$cdom,$cnum,$cdesc,$type,$readonly); + } elsif (($phase eq 'processthreshold') && ($permission->{'processthreshold'})) { + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'setanon')", + text=>"Threshold for anonymous submissions display"}); + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'$phase')", + text=>"Result"}); + &modify_anonsurvey_threshold($r,$cdom,$cnum,$cdesc,$domdesc,$type); + } elsif (($phase eq 'setpostsubmit') && ($permission->{'setpostsubmit'})) { + if ($permission->{'setpostsubmit'} eq 'view') { + $linktext = 'Submit button behavior post-submission'; + } else { + $linktext = 'Configure submit button behavior post-submission'; + } + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'$phase')", + text=>$linktext}); + &print_postsubmit_config($r,$cdom,$cnum,$cdesc,$type,$readonly); + } elsif (($phase eq 'processpostsubmit') && ($permission->{'processpostsubmit'})) { + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'$phase')", + text=>"Result"}); + &modify_postsubmit_config($r,$cdom,$cnum,$cdesc,$domdesc,$type); + } elsif (($phase eq 'viewparms') && ($permission->{'viewparms'})) { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'viewparms')", text=>"Display settings"}); - &print_settings_display($r,$cdom,$cnum,$cdesc,$type); - } elsif ($phase eq 'setparms') { + &print_settings_display($r,$cdom,$cnum,$cdesc,$type,$permission); + } elsif (($phase eq 'setparms') && ($permission->{'setparms'})) { + if ($permission->{'setparms'} eq 'view') { + $linktext = 'Display settings'; + } else { + $linktext = 'Change settings'; + } &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'$phase')", - text=>"Change settings"}); - &print_course_modification_page($r,$cdom,$cnum,$cdesc,$domdesc); - } elsif ($phase eq 'processparms') { + text=>$linktext}); + &print_course_modification_page($r,$cdom,$cnum,$cdesc,$type,$readonly); + } elsif (($phase eq 'processparms') && ($permission->{'processparms'})) { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'setparms')", text=>"Change settings"}); @@ -1202,30 +2724,50 @@ sub handler { ({href=>"javascript:changePage(document.$phase,'$phase')", text=>"Result"}); &modify_course($r,$cdom,$cnum,$cdesc,$domdesc,$type); - } elsif ($phase eq 'catsettings') { + } elsif (($phase eq 'catsettings') && ($permission->{'catsettings'})) { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'$phase')", text=>"Catalog settings"}); - &print_catsettings($r,$cdom,$cnum,$cdesc,$type); - } elsif ($phase eq 'processcat') { + &print_catsettings($r,$cdom,$cnum,$cdesc,$type,$readonly); + } elsif (($phase eq 'processcat') && ($permission->{'processcat'})) { &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'catsettings')", text=>"Catalog settings"}); &Apache::lonhtmlcommon::add_breadcrumb ({href=>"javascript:changePage(document.$phase,'$phase')", text=>"Result"}); - &modify_catsettings($r,$cdom,$cnum,$cdesc,$domdesc); + &modify_catsettings($r,$cdom,$cnum,$cdesc,$domdesc,$type); + } elsif (($phase eq 'selfenroll') && ($permission->{'selfenroll'})) { + &Apache::lonhtmlcommon::add_breadcrumb + ({href => "javascript:changePage(document.$phase,'$phase')", + text => "Self-enrollment settings"}); + if (!exists($env{'form.state'})) { + &print_selfenrollconfig($r,$type,$cdesc,$coursehash,$readonly); + } elsif ($env{'form.state'} eq 'done') { + &Apache::lonhtmlcommon::add_breadcrumb + ({href=>"javascript:changePage(document.$phase,'$phase')", + text=>"Result"}); + &modify_selfenrollconfig($r,$type,$cdesc,$coursehash); + } } } } else { - $r->print(''.&mt('The course you selected is not a valid course in this domain')." ($domdesc)".''); + $r->print(''); + if ($type eq 'Community') { + $r->print(&mt('The community you selected is not a valid community in this domain')); + } elsif ($type eq 'Placement') { + $r->print(&mt('The course you selected is not a valid placement test in this domain')); + } else { + $r->print(&mt('The course you selected is not a valid course in this domain')); + } + $r->print(" ($domdesc)"); } } } &print_footer($r); } else { $env{'user.error.msg'}= - "/adm/modifycourse:ccc:0:0:Cannot modify course settings"; + "/adm/modifycourse:ccc:0:0:Cannot modify course/community settings"; return HTTP_NOT_ACCEPTABLE; } return OK;