# The LearningOnline Network with CAPA
# handler for DC-only modifiable course settings
#
# $Id: lonmodifycourse.pm,v 1.104 2023/12/23 02:17:38 raeburn Exp $
#
# Copyright Michigan State University Board of Trustees
#
# This file is part of the LearningOnline Network with CAPA (LON-CAPA).
#
# LON-CAPA is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# LON-CAPA is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with LON-CAPA; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# /home/httpd/html/adm/gpl.txt
#
# http://www.lon-capa.org/
#
package Apache::lonmodifycourse;
use strict;
use Apache::Constants qw(:common :http);
use Apache::lonnet;
use Apache::loncommon;
use Apache::lonhtmlcommon;
use Apache::lonlocal;
use Apache::lonuserutils;
use Apache::loncreateuser;
use Apache::lonpickcourse;
use lib '/home/httpd/lib/perl';
use LONCAPA qw(:DEFAULT :match);
my $registered_cleanup;
my $modified_dom;
sub get_dc_settable {
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,$type) = @_;
my @settable;
if (ref($confhash) eq 'HASH') {
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'));
}
return @settable;
}
sub get_enrollment_settings {
my ($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'} = '';
foreach my $item (keys(%settings)) {
if ($item =~ m/^internal\.(.+)$/) {
my $type = $1;
if ( ($type eq "autoadds") || ($type eq "autodrops") ) {
if ($settings{$item} == 1) {
$enrollvar{$type} = "ON";
} else {
$enrollvar{$type} = "OFF";
}
} elsif ( ($type eq "autostart") || ($type eq "autoend") ) {
if ( ($type eq "autoend") && ($settings{$item} == 0) ) {
$enrollvar{$type} = &mt('No end date');
} else {
$enrollvar{$type} = &Apache::lonlocal::locallocaltime($settings{$item});
}
} 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 "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};
} else {
if ($settings{$item} ne '') {
$enrollvar{$type} = $settings{$item}.':'.$cdom;
}
}
}
} elsif ($item =~ m/^default_enrollment_(start|end)_date$/) {
my $type = $1;
if ( ($type eq 'end') && ($settings{$item} == 0) ) {
$enrollvar{$item} = &mt('No end date');
} elsif ( ($type eq 'start') && ($settings{$item} eq '') ) {
$enrollvar{$item} = 'When enrolled';
} else {
$enrollvar{$item} = &Apache::lonlocal::locallocaltime($settings{$item});
}
}
}
return %enrollvar;
}
sub print_course_search_page {
my ($r,$dom,$domdesc) = @_;
my $action = '/adm/modifycourse';
my $type = $env{'form.type'};
if (!defined($env{'form.type'})) {
$type = 'Course';
}
&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::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.'
');
if ($roleoption) {
$r->print('
'.$roleoption.'
'."\n");
}
$r->print('
'.$settingsoption.'
'."\n".'
');
return;
}
sub print_course_selection_page {
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");
$lt{'preamble'}
$lt{'cancel'}
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,@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,
$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,$type,$cid,$coursehash,$permission) = @_;
&print_header($r,$type);
my ($ccrole,$categorytitle,$setquota_text,$setuploadquota_text,$cdom,$cnum,
$extendedtype);
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',
'setltiauth' => 'View/Modify re-authentication requirement for LTI launch of deep-linked item',
'setexttool' => 'View/Modify External Tools permissions',
'setcrsauthor' => 'View/Modify In-course Authoring permissions',
);
} 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',
'setltiauth' => 'View re-authentication requirement for LTI launch of deep-linked item',
'setexttool' => 'View External Tools permissions',
'setcrsauthor' => 'View In-course Authoring permissions',
);
}
if ($type eq 'Community') {
if ($permission->{'setparms'} eq 'edit') {
$categorytitle = 'View/Modify Community Settings';
$linktext{'setparms'} = 'View/Modify community owner, self-enrollment and table lifetime';
$linktext{'catsettings'} = 'View/Modify catalog settings for community';
} else {
$categorytitle = 'View Community Settings';
$linktext{'setparms'} = 'View community owner, self-enrollment and table lifetime';
$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 %settings = &Apache::lonnet::get('environment',['internal.coursecode','internal.textbook'],
$cdom,$cnum);
$extendedtype = ucfirst(&Apache::lonuserutils::get_extended_type($cdom,$cnum,$type,\%settings));
}
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.');
my $ltiauth_text = &mt('Override default for requirement for re-authentication for LTI-limited launch of deep-linked item.');
my $exttool_text = &mt('Override default permissions for external tools use for this specific course.');
$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 =
({ categorytitle => $categorytitle,
items => [
{
linktext => $linktext{'setparms'},
url => &phaseurl('setparms'),
permission => $permission->{'setparms'},
#help => '',
icon => 'crsconf.png',
linktitle => ''
},
{
linktext => $linktext{'setquota'},
url => &phaseurl('setquota'),
permission => $permission->{'setquota'},
#help => '',
icon => 'groupportfolioquota.png',
linktitle => ''
},
{
linktext => $linktext{'setanon'},
url => &phaseurl('setanon'),
permission => $permission->{'setanon'},
#help => '',
icon => 'anonsurveythreshold.png',
linktitle => ''
},
{
linktext => $linktext{'catsettings'},
url => &phaseurl('catsettings'),
permission => (($permission->{'catsettings'}) && (@additional_params > 0)),
#help => '',
icon => 'ccatconf.png',
linktitle => ''
},
{
linktext => $linktext{'viewparms'},
url => &phaseurl('viewparms'),
permission => ($permission->{'viewparms'} && ($type ne 'Community') && ($type ne 'Placement')),
#help => '',
icon => 'roles.png',
linktitle => ''
},
{
linktext => $linktext{'selfenroll'},
icon => 'self_enroll.png',
#help => 'Course_Self_Enrollment',
url => &phaseurl('selfenroll'),
permission => &manage_selfenrollment($cdom,$cnum,$type,$coursehash,$permission),
linktitle => 'Configure user self-enrollment.',
},
{
linktext => $linktext{'setpostsubmit'},
icon => 'emblem-readonly.png',
#help => '',
url => &phaseurl('setpostsubmit'),
permission => $permission->{'setpostsubmit'},
linktitle => '',
},
{
linktext => $linktext{'setltiauth'},
icon => 'system-lock-screen.png',
#help => '',
url => &phaseurl('setltiauth'),
permission => $permission->{'setltiauth'},
linktitle => '',
},
{
linktext => $linktext{'setexttool'},
icon => 'exttool.png',
#help => '',
url => &phaseurl('setexttool'),
permission => $permission->{'setexttool'},
linktitle => '',
},
{
linktext => $linktext{'setcrsauthor'},
icon => 'crsauthor.png',
#help => '',
url => &phaseurl('setcrsauthor'),
permission => $permission->{'setcrsauthor'},
linktitle => '',
},
]
},
);
$r->print(
'
'
.&mt($type).': '.$cdesc.''
.'
'."\n");
if ($extendedtype) {
$r->print('
'.&mt('Type').': '.&mt("$extendedtype $type").'
');
}
$r->print(
'');
return;
}
sub print_adhocrole_selected {
my ($r,$type,$permission) = @_;
&print_header($r,$type);
my ($cdom,$cnum) = split(/_/,$env{'form.pickedcourse'});
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,$permission) = @_;
my %enrollvar = &get_enrollment_settings($cdom,$cnum);
my %longtype = &course_settings_descrip($type);
my %lt = &Apache::lonlocal::texthash(
'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 $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($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".
"
\n".
"
$lt{'valu'}
\n".
"
$lt{'dcon'}
\n".
&Apache::loncommon::end_data_table_header_row()."\n";
foreach my $item (@items) {
my $shown = $enrollvar{$item};
if ($item eq 'crosslistings') {
my (@xlists,@lcsecs);
foreach my $entry (split(/,/,$enrollvar{$item})) {
my ($xlist,$lc_sec) = split(/:/,$entry);
push(@xlists,$xlist);
push(@lcsecs,$lc_sec);
}
if (@xlists) {
my $crskey = $cnum.':'.$enrollvar{'coursecode'};
my %reformatted =
&Apache::lonnet::auto_instsec_reformat($cdom,'declutter',
{$crskey => \@xlists});
if (ref($reformatted{$crskey}) eq 'ARRAY') {
my @show;
my @xlcodes = @{$reformatted{$crskey}};
for (my $i=0; $i<@xlcodes; $i++) {
push(@show,$xlcodes[$i].':'.$lcsecs[$i]);
}
if (@show) {
$shown = join(',',@show);
}
}
}
}
$disp_table .= &Apache::loncommon::start_data_table_row()."\n".
"
$longtype{$item}
\n".
"
$shown
\n";
if (grep(/^\Q$item\E$/,@modifiable_params)) {
$disp_table .= '
'.&mt('Yes').'
'."\n";
} else {
$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,$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').'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n".
''."\n");
my @actions =
(''.
$lt{'back'}.'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
}
sub print_setquota {
my ($r,$cdom,$cnum,$cdesc,$type,$readonly) = @_;
my $lctype = lc($type);
my $headline = '
'.&mt("Set disk space quotas for $lctype").'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n";
my %lt = &Apache::lonlocal::texthash(
'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' => 'Pick another action',
);
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 (($uploadquota eq '') || ($coursequota eq '')) {
my %domdefs = &Apache::lonnet::get_domain_defaults($cdom);
my $quotatype = &Apache::lonuserutils::get_extended_type($cdom,$cnum,$type,\%settings);
if ($uploadquota eq '') {
$uploadquota = $domdefs{$quotatype.'quota'};
if ($uploadquota eq '') {
$uploadquota = $staticdefaults{'uploadquota'};
}
}
if ($coursequota eq '') {
$coursequota = $domdefs{$quotatype.'coursequota'};
if ($coursequota eq '') {
$coursequota = $staticdefaults{'coursequota'};
}
}
}
&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(<
'.&Apache::lonhtmlcommon::actionbox(\@actions));
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]',$dctitle);
} else {
return &mt('Course settings modifiable only by [_1]',$dctitle);
}
}
sub gather_authenitems {
my ($cdom,$enrollvar,$readonly) = @_;
my ($krbdef,$krbdefdom)=&Apache::loncommon::get_kerberos_defaults($cdom);
my $curr_authtype = '';
my $curr_authfield = '';
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 %param = ( formname => 'document.'.$env{'form.phase'},
kerb_def_dom => $krbdefdom,
kerb_def_auth => $krbdef,
mode => 'modifycourse',
curr_authtype => $curr_authtype,
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);
$authform{'lti'} = &Apache::loncommon::authform_lti(%param);
foreach my $item ('krb','int','loc','lti') {
if ($authform{$item} ne '') {
$authenitems .= $authform{$item}.' ';
}
}
return($javascript_validations,$authenitems);
}
sub modify_course {
my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_;
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 %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) {
$currattr{$param} = $settings{'internal.'.$param};
}
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.'));
}
}
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 ($changed{'owner'} || $changed{'code'} || $changed{'passwd'}) {
my %crsinfo = &Apache::lonnet::courseiddump($cdom,'.',1,'.','.',$cnum,
undef,undef,'.');
if (ref($crsinfo{$env{'form.pickedcourse'}}) eq 'HASH') {
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 {
$cenv{'internal.'.$param} = $newattr{$param};
push(@changes,$param);
}
}
if (@changes > 0) {
$chgresponse = &mt('The following settings have been changed:').'
';
}
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.').' ';
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.'
'.&mt('[_1] still set to: [_2]',$longtype{$attr},$shown).'
';
}
}
if (($type ne 'Community') && ($type ne 'Placement') && ($changed{'code'} || $changed{'owner'})) {
if ( $newattr{'courseowner'} eq '') {
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'},$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 {
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 {
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 $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') {
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 {
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 {
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 $instxlist = $1;
my $outcome = &Apache::lonnet::auto_new_course($cnum,$cdom,$instxlist,$newattr{'courseowner'},$coowners);
unless ($outcome eq 'ok') {
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));
}
}
}
}
}
}
}
} else {
foreach my $attr (@modifiable_params) {
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).'
';
}
}
if (@changes > 0) {
$chgresponse .= "
";
}
if (@nochanges > 0) {
$nochgresponse .= "
";
}
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).'
'."\n".
''.
' '.&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,$type) = @_;
&print_header($r,$type);
my $lctype = lc($type);
$r->print('
'.&mt("Disk space quotas for $lctype")."
\n".
'
'.&mt($type).' :'.$cdesc.'
'."\n".
'');
my @actions =
(''.
&mt('Pick another action').'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
return;
}
sub modify_anonsurvey_threshold {
my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_;
&print_header($r,$type);
$r->print('
'.&mt('Responder threshold required for display of anonymous survey submissions').'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n".
'');
my @actions =
(''.
&mt('Pick another action').'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
return;
}
sub modify_postsubmit_config {
my ($r,$cdom,$cnum,$cdesc,$domdesc,$type) = @_;
&print_header($r,$type);
my %lt = &Apache::lonlocal::texthash(
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.',
back => 'Pick another action',
);
$r->print('
'.$lt{'subb'}.'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n".
'');
my @actions =
(''.
$lt{'back'}.'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
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').'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n".
'');
my @actions =
(''.
&mt('Pick another action').'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
return;
}
sub modify_default_overrides {
my ($r,$cdom,$cnum,$cdesc,$domdesc,$type,$item) = @_;
my (%titles,$oldcrsval,$newcrsval,%resulttext,$itemvalue,$nochange,$change,$status,$error);
%titles = &default_overrides_titles($type);
my ($title,$domdefdisplay,$oldsettings,$optiontext,$options) =
&default_overrides_common($item,$cdom,$cnum,$type,\%titles);
$oldcrsval = $oldsettings->{'internal.'.$item};
if ($item eq 'ltiauth') {
%resulttext =
&Apache::lonlocal::texthash(
chg => 'Re-authentication requirement for LTI launch of deep-linked changed',
nochg => 'Re-authentication requirement for LTI launch of deep-linked item is unchanged',
);
} elsif ($item eq 'exttool') {
%resulttext =
&Apache::lonlocal::texthash(
chg => 'External Tool permissions changed',
nochg => 'External Tool permissions unchanged',
);
} elsif ($item eq 'crsauthor') {
%resulttext =
&Apache::lonlocal::texthash(
chg => 'In-course authoring permissions changed',
nochg => 'In-course authoring permissions unchanged',
);
}
&print_header($r,$type);
$r->print('
'.$title.'
'."\n".
'
'.&mt($type).': '.$cdesc.'
'."\n".
'');
my @actions =
(''.
$titles{'back'}.'');
$r->print(' '.&Apache::lonhtmlcommon::actionbox(\@actions));
return;
}
sub print_header {
my ($r,$type,$javascript_validations) = @_;
my $phase = "start";
if ( exists($env{'form.phase'}) ) {
$phase = $env{'form.phase'};
}
my $js = qq|
function changePage(formname,newphase) {
formname.phase.value = newphase;
if (newphase == 'processparms') {
return;
}
formname.submit();
}
|;
if ($phase eq 'setparms') {
$js .= $javascript_validations;
} elsif ($phase eq 'courselist') {
$js .= <<"ENDJS";
function hide_searching() {
if (document.getElementById('searching')) {
document.getElementById('searching').style.display = 'none';
}
return;
}
ENDJS
} elsif ($phase eq 'setquota') {
my $invalid = &mt('The quota you entered contained invalid characters.');
my $alert = &mt('You must enter a number');
&js_escape(\$invalid);
&js_escape(\$alert);
my $regexp = '/^\s*(\d+\.?\d*|\.\d+)\s*$/';
$js .= <<"ENDSCRIPT";
function verify_quota() {
var newcoursequota = document.setquota.coursequota.value;
var newuploadquota = document.setquota.uploadquota.value;
var num_reg = $regexp;
if ((num_reg.test(newcoursequota)) && (num_reg.test(newuploadquota))) {
changePage(document.setquota,'processquota');
} else {
alert("$invalid\\n$alert");
return false;
}
return true;
}
ENDSCRIPT
} elsif ($phase eq 'setanon') {
my $invalid = &mt('The responder threshold you entered is invalid.');
my $alert = &mt('You must enter a positive integer.');
&js_escape(\$invalid);
&js_escape(\$alert);
my $regexp = ' /^\s*\d+\s*$/';
$js .= <<"ENDSCRIPT";
function verify_anon_threshold() {
var newthreshold = document.setanon.threshold.value;
var num_reg = $regexp;
if (num_reg.test(newthreshold)) {
if (newthreshold > 0) {
changePage(document.setanon,'processthreshold');
} else {
alert("$invalid\\n$alert");
return false;
}
} else {
alert("$invalid\\n$alert");
return false;
}
return true;
}
ENDSCRIPT
} elsif ($phase eq 'setpostsubmit') {
my $invalid = &mt('The choice entered for disabling the submit button is invalid.');
my $invalidtimeout = &mt('The timeout you entered for disabling the submit button is invalid.');
my $alert = &mt('Enter one of: a positive integer, 0 (for no timeout), or leave blank to use domain default');
&js_escape(\$invalid);
&js_escape(\$invalidtimeout);
&js_escape(\$alert);
my $regexp = ' /^\s*\d+\s*$/';
$js .= <<"ENDSCRIPT";
function verify_postsubmit() {
var optionsElement = document.setpostsubmit.postsubmit;
var verified = '';
if (optionsElement.length) {
var currval;
for (var i=0; i= 0) {
verified = 'ok';
} else {
alert("$invalidtimeout\\n$alert");
return false;
}
} else {
alert("$invalid\\n$alert");
return false;
}
}
} else {
if (currval == 0) {
verified = 'ok';
} else {
alert('$invalid');
return false;
}
}
if (verified == 'ok') {
changePage(document.setpostsubmit,'processpostsubmit');
return true;
}
}
return false;
}
function togglePostsubmit(caller) {
var optionsElement = document.setpostsubmit.postsubmit;
if (document.getElementById(caller)) {
var divitem = document.getElementById(caller);
var optionsElement = document.setpostsubmit.postsubmit;
if (optionsElement.length) {
var currval;
for (var i=0; i {'onload' => "javascript:document.adhocrole.submit();"},
};
} elsif ($phase eq 'courselist') {
$starthash = {
add_entries => {'onload' => "hide_searching(); courseSet(document.filterpicker.official, 'load');"},
};
} elsif ($env{'form.phase'} =~ /^set(ltiauth|exttool|crsauthor)$/) {
$starthash = {
add_entries => {'onload' => "toggleOptions(document.$env{'form.phase'},'$env{'form.phase'}');"},
};
}
$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 {
$bread_text = 'Course Settings';
}
my $helpcomponent;
if ($env{'form.phase'} eq 'menu') {
if ($type eq 'Community') {
$helpcomponent = 'Domain_Modify_Community';
} else {
$helpcomponent = 'Domain_Modify_Course';
}
}
$r->print(&Apache::lonhtmlcommon::breadcrumbs($bread_text,$helpcomponent));
return;
}
sub print_footer {
my ($r) = @_;
$r->print(' '.&Apache::loncommon::end_page());
return;
}
sub check_course {
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,\%coursehash);
}
sub course_settings_descrip {
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',
'autodrops' => 'Automated drops',
'autostart' => 'Date of first automated enrollment',
'autoend' => 'Date of last automated enrollment',
'default_enrollment_start_date' => 'Date of first student access',
'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',
'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;
}
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',
'threshold','postsubmit','postsubtimeout','defaultcredits','uploadquota',
'selfenrollmgrdc','selfenrollmgrcc','action','state','currsec_st',
'sections','newsec','mysqltables','nopasswdchg','ltiauth','ltiauthset',
'exttoolset','exttool','crsauthorset','crsauthor'],
['^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',
setltiauth => 'edit',
processltiauth => 'edit',
setexttool => 'edit',
processexttool => 'edit',
setcrsauthor => 'edit',
processcrsauthor => 'edit',
);
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',
setltiauth => 'view',
setexttool => 'view',
setcrsauthor => 'view',
);
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) {
&Apache::loncommon::content_type($r,'text/html');
$r->send_http_header;
return OK;
}
$registered_cleanup=0;
$modified_dom = '';
my $dom = $env{'request.role.domain'};
my $domdesc = &Apache::lonnet::domain($dom,'description');
my ($allowed,$permission) = &get_permission($dom);
if ($allowed) {
&Apache::loncommon::content_type($r,'text/html');
$r->send_http_header;
&Apache::lonhtmlcommon::clear_breadcrumbs();
my $phase = $env{'form.phase'};
if ($env{'form.updater'}) {
$phase = '';
}
if ($phase eq '') {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"/adm/modifycourse",
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/Community search"},
{href=>"javascript:changePage(document.$phase,'courselist')",
text=>$choose_text});
if ($phase eq 'courselist') {
&print_course_selection_page($r,$dom,$domdesc,$permission);
} else {
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,$type,
$env{'form.pickedcourse'},$coursehash,
$permission);
} elsif ($phase eq 'adhocrole') {
&Apache::lonhtmlcommon::add_breadcrumb
({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'});
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=>$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,$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,$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=>$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"});
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'$phase')",
text=>"Result"});
&modify_course($r,$cdom,$cnum,$cdesc,$domdesc,$type);
} 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,$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,$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);
}
} elsif (($phase eq 'setltiauth') && ($permission->{'setltiauth'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'$phase')",
text=>"Authentication post-LTI launch"});
&print_default_overrides($r,$cdom,$cnum,$cdesc,$type,$readonly,'ltiauth');
} elsif (($phase eq 'processltiauth') && ($permission->{'processltiauth'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'setltiauth')",
text=>"Authentication post-LTI launch"},
{href=>"javascript:changePage(document.$phase,'$phase')",
text=>"Result"});
&modify_default_overrides($r,$cdom,$cnum,$cdesc,$domdesc,$type,'ltiauth');
} elsif (($phase eq 'setexttool') && ($permission->{'setexttool'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'$phase')",
text=>"External Tool permission"});
&print_default_overrides($r,$cdom,$cnum,$cdesc,$type,$readonly,'exttool');
} elsif (($phase eq 'processexttool') && ($permission->{'processexttool'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'setexttool')",
text=>"External Tool permission"},
{href=>"javascript:changePage(document.$phase,'$phase')",
text=>"Result"});
&modify_default_overrides($r,$cdom,$cnum,$cdesc,$domdesc,$type,'exttool');
} elsif (($phase eq 'setcrsauthor') && ($permission->{'setcrsauthor'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'$phase')",
text=>"In-course authoring permission"});
&print_default_overrides($r,$cdom,$cnum,$cdesc,$type,$readonly,'crsauthor');
} elsif (($phase eq 'processcrsauthor') && ($permission->{'processcrsauthor'})) {
&Apache::lonhtmlcommon::add_breadcrumb
({href=>"javascript:changePage(document.$phase,'setcrsauthor')",
text=>"In-course authoring permission"},
{href=>"javascript:changePage(document.$phase,'$phase')",
text=>"Result"});
&modify_default_overrides($r,$cdom,$cnum,$cdesc,$domdesc,$type,'crsauthor');
}
}
} else {
$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/community settings";
return HTTP_NOT_ACCEPTABLE;
}
return OK;
}
1;
__END__