--- loncom/interface/loncreateuser.pm 2009/11/04 14:13:32 1.323 +++ loncom/interface/loncreateuser.pm 2009/11/23 13:03:58 1.328 @@ -1,7 +1,7 @@ # The LearningOnline Network with CAPA # Create a user # -# $Id: loncreateuser.pm,v 1.323 2009/11/04 14:13:32 raeburn Exp $ +# $Id: loncreateuser.pm,v 1.328 2009/11/23 13:03:58 wenzelju Exp $ # # Copyright Michigan State University Board of Trustees # @@ -51,13 +51,14 @@ In LON-CAPA, roles are actually collecti Assistant", "Course Coordinator", and other such roles are really just collection of privileges that are useful in many circumstances. -Creating custom roles can be done by the Domain Coordinator through -the Create User functionality. That screen will show all privileges -that can be assigned to users. For a complete list of privileges, -please see C</home/httpd/lonTabs/rolesplain.tab>. +Custom roles can be defined by a Domain Coordinator, Course Coordinator +or Community Coordinator via the Manage User functionality. +The custom role editor screen will show all privileges which can be +assigned to users. For a complete list of privileges, please see +C</home/httpd/lonTabs/rolesplain.tab>. -Custom role definitions are stored in the C<roles.db> file of the role -author. +Custom role definitions are stored in the C<roles.db> file of the creator +of the role. =cut @@ -487,6 +488,11 @@ sub print_username_entry_form { '// ]]>'."\n". '</script>'."\n"; + my %existingroles=&Apache::lonuserutils::my_custom_roles($crstype); + if (($env{'form.action'} eq 'custom') && (keys(%existingroles) > 0) + && (&Apache::lonnet::allowed('mcr','/'))) { + $jscript .= &customrole_javascript(); + } my %loaditems = ( 'onload' => "javascript:setFormElements(document.$formtoset)", ); @@ -512,31 +518,53 @@ sub print_username_entry_form { } my $crumbs = &Apache::lonhtmlcommon::breadcrumbs('User Management', $helpitem); - my %existingroles=&Apache::lonuserutils::my_custom_roles(); - my $choice=&Apache::loncommon::select_form('make new role','rolename', - ('make new role' => 'Generate new role ...',%existingroles)); my %lt=&Apache::lonlocal::texthash( 'srst' => 'Search for a user and enroll as a student', 'srme' => 'Search for a user and enroll as a member', 'srad' => 'Search for a user and modify/add user information or roles', 'usr' => "Username", 'dom' => "Domain", - 'ecrp' => "Edit Custom Role Privileges", - 'nr' => "Name of Role", + 'ecrp' => "Define or Edit Custom Role", + 'nr' => "role name", 'cre' => "Next", ); $r->print($start_page."\n".$crumbs); if ($env{'form.action'} eq 'custom') { if (&Apache::lonnet::allowed('mcr','/')) { - $r->print(<<ENDCUSTOM); -<form action="/adm/createuser" method="post" name="docustom"> -<input type="hidden" name="action" value="$env{'form.action'}" /> -<input type="hidden" name="phase" value="selected_custom_edit" /> -<h3>$lt{'ecrp'}</h3> -$choice $lt{'nr'}: <input type="text" size="15" name="newrolename" /><br /> -<input name="customeditor" type="submit" value="$lt{'cre'}" /> -</form> -ENDCUSTOM + my $newroletext = &mt('Define new custom role:'); + $r->print('<form action="/adm/createuser" method="post" name="docustom">'. + '<input type="hidden" name="action" value="'.$env{'form.action'}.'" />'. + '<input type="hidden" name="phase" value="selected_custom_edit" />'. + '<h3>'.$lt{'ecrp'}.'</h3>'. + &Apache::loncommon::start_data_table(). + &Apache::loncommon::start_data_table_row(). + '<td>'); + if (keys(%existingroles) > 0) { + $r->print('<br /><label><input type="radio" name="customroleaction" value="new" checked="checked" onclick="setCustomFields();" /><b>'.$newroletext.'</b></label>'); + } else { + $r->print('<br /><input type="hidden" name="customroleaction" value="new" /><b>'.$newroletext.'</b>'); + } + $r->print('</td><td align="center">'.$lt{'nr'}.'<br /><input type="text" size="15" name="newrolename" onfocus="setCustomAction('."'new'".');" /></td>'. + &Apache::loncommon::end_data_table_row()); + if (keys(%existingroles) > 0) { + $r->print(&Apache::loncommon::start_data_table_row().'<td><br />'. + '<label><input type="radio" name="customroleaction" value="edit" onclick="setCustomFields();"/><b>'. + &mt('View/Modify existing role:').'</b></label></td>'. + '<td align="center"><br />'. + '<select name="rolename" onchange="setCustomAction('."'edit'".');">'. + '<option value="" selected="selected">'. + &mt('Select')); + foreach my $role (sort(keys(%existingroles))) { + $r->print('<option value="'.$role.'">'.$role.'</option>'); + } + $r->print('</select>'. + '</td>'. + &Apache::loncommon::end_data_table_row()); + } + $r->print(&Apache::loncommon::end_data_table().'<p>'. + '<input name="customeditor" type="submit" value="'. + $lt{'cre'}.'" /></p>'. + '</form>'); } } else { my $actiontext = $lt{'srad'}; @@ -547,8 +575,7 @@ ENDCUSTOM $actiontext = $lt{'srst'}; } } - $r->print(" -<h3>$actiontext</h3>"); + $r->print("<h3>$actiontext</h3>"); if ($env{'form.origform'} ne 'crtusername') { $r->print("\n".$response); } @@ -557,6 +584,44 @@ ENDCUSTOM $r->print(&Apache::loncommon::end_page()); } +sub customrole_javascript { + my $js = <<"END"; +<script type="text/javascript"> +// <![CDATA[ + +function setCustomFields() { + if (document.docustom.customroleaction.length > 0) { + for (var i=0; i<document.docustom.customroleaction.length; i++) { + if (document.docustom.customroleaction[i].checked) { + if (document.docustom.customroleaction[i].value == 'new') { + document.docustom.rolename.selectedIndex = 0; + } else { + document.docustom.newrolename.value = ''; + } + } + } + } + return; +} + +function setCustomAction(caller) { + if (document.docustom.customroleaction.length > 0) { + for (var i=0; i<document.docustom.customroleaction.length; i++) { + if (document.docustom.customroleaction[i].value == caller) { + document.docustom.customroleaction[i].checked = true; + } + } + } + setCustomFields(); + return; +} + +// ]]> +</script> +END + return $js; +} + sub entry_form { my ($dom,$srch,$forcenewuser,$context,$responsemsg,$crstype) = @_; my %domconf = &Apache::lonnet::get_dom('configuration',['usercreation'],$dom); @@ -3478,18 +3543,27 @@ sub build_roles { sub custom_role_editor { my ($r) = @_; - my $rolename=$env{'form.rolename'}; - - if ($rolename eq 'make new role') { - $rolename=$env{'form.newrolename'}; + my $action = $env{'form.customroleaction'}; + my $rolename; + if ($action eq 'new') { + $rolename=$env{'form.newrolename'}; + } else { + $rolename=$env{'form.rolename'}; } $rolename=~s/[^A-Za-z0-9]//gs; - if (!$rolename || $env{'form.phase'} eq 'pickrole') { &print_username_entry_form($r); return; } + my ($crstype,$context); + if ($env{'request.course.id'}) { + $crstype = &Apache::loncommon::course_type(); + $context = 'course'; + } else { + $context = 'domain'; + $crstype = $env{'form.templatecrstype'}; + } # ------------------------------------------------------- What can be assigned? my %full=(); my %courselevel=(); @@ -3498,7 +3572,6 @@ sub custom_role_editor { my $dompriv=''; my $coursepriv=''; my $body_top; - my ($disp_dummy,$disp_roles) = &Apache::lonnet::get('roles',["st"]); my ($rdummy,$roledef)= &Apache::lonnet::get('roles',["rolesdef_$rolename"]); # ------------------------------------------------------- Does this role exist? @@ -3507,6 +3580,9 @@ sub custom_role_editor { $body_top .= &mt('Existing Role').' "'; # ------------------------------------------------- Get current role privileges ($syspriv,$dompriv,$coursepriv)=split(/\_/,$roledef); + if ($crstype eq 'Community') { + $syspriv =~ s/bre\&S//; + } } else { $body_top .= &mt('New Role').' "'; $roledef=''; @@ -3548,20 +3624,50 @@ sub custom_role_editor { my $head_script = "\n"; $head_script .= '<script type="text/javascript">'."\n" .'// <![CDATA['."\n"; - my $crstype; - if ($env{'request.course.id'}) { - $crstype = &Apache::loncommon::course_type(); + my @template_roles = ("in","ta","ep"); + if ($context eq 'domain') { + push(@template_roles,"ad"); } - my @template_roles = ("in","ta","ep","st"); + push(@template_roles,"st"); if ($crstype eq 'Community') { unshift(@template_roles,'co'); } else { unshift(@template_roles,'cc'); } foreach my $role (@template_roles) { - $head_script .= &make_script_template($role); + $head_script .= &make_script_template($role,$crstype); $button_code .= &make_button_code($role,$crstype).' '; } + my $context_code; + if ($context eq 'domain') { + my $checkedCommunity = ''; + my $checkedCourse = ' checked="checked"'; + if ($env{'form.templatecrstype'} eq 'Community') { + $checkedCommunity = $checkedCourse; + $checkedCourse = ''; + } + $context_code = '<label>'. + '<input type="radio" name="templatecrstype" value="Course"'.$checkedCourse.' onclick="this.form.submit();">'. + &mt('Course'). + '</label>'.(' ' x2). + '<label>'. + '<input type="radio" name="templatecrstype" value="Community"'.$checkedCommunity.' onclick="this.form.submit();">'. + &mt('Community'). + '</label>'. + '</fieldset>'. + '<input type="hidden" name="customroleaction" value="'. + $action.'" />'; + if ($env{'form.customroleaction'} eq 'new') { + $context_code .= '<input type="hidden" name="newrolename" value="'. + $rolename.'" />'; + } else { + $context_code .= '<input type="hidden" name="rolename" value="'. + $rolename.'" />'; + } + $context_code .= '<input type="hidden" name="action" value="custom" />'. + '<input type="hidden" name="phase" value="selected_custom_edit" />'; + } + $head_script .= "\n".$jsback."\n" .'// ]]>'."\n" .'</script>'."\n"; @@ -3583,13 +3689,22 @@ sub custom_role_editor { 'dml' => "Domain Level", 'ssl' => "System Level"); - $r->print('<div>' + + $r->print('<div class="LC_left_float">' .'<form action=""><fieldset>' .'<legend>'.&mt('Select a Template').'</legend>' .$button_code - .'</fieldset></form>' - .'</div>' - ); + .'</fieldset></form></div>'); + if ($context_code) { + $r->print('<div class="LC_left_float">' + .'<form action="/adm/createuser" method="post"><fieldset>' + .'<legend>'.&mt('Context').'</legend>' + .$context_code + .'</form>' + .'</div>' + ); + } + $r->print('<br clear="all" />'); $r->print(<<ENDCCF); <form name="form1" method="post"> @@ -3601,7 +3716,7 @@ ENDCCF '<th>'.$lt{'prv'}.'</th><th>'.$lt{'crl'}.'</th><th>'.$lt{'dml'}. '</th><th>'.$lt{'ssl'}.'</th>'. &Apache::loncommon::end_data_table_header_row()); - foreach my $priv (sort keys %full) { + foreach my $priv (sort(keys(%full))) { my $privtext = &Apache::lonnet::plaintext($priv,$crstype); $r->print(&Apache::loncommon::start_data_table_row(). '<td>'.$privtext.'</td><td>'. @@ -3610,11 +3725,15 @@ ENDCCF '</td><td>'. ($domainlevel{$priv}?'<input type="checkbox" name="'.$priv.'_d"'. ($domainlevelcurrent{$priv}?' checked="checked"':'').' />':' '). - '</td><td>'. - ($systemlevel{$priv}?'<input type="checkbox" name="'.$priv.'_s"'. - ($systemlevelcurrent{$priv}?' checked="checked"':'').' />':' '). - '</td>'. - &Apache::loncommon::end_data_table_row()); + '</td><td>'); + if ($priv eq 'bre' && $crstype eq 'Community') { + $r->print(' '); + } else { + $r->print($systemlevel{$priv}?'<input type="checkbox" name="'.$priv.'_s"'. + ($systemlevelcurrent{$priv}?' checked="checked"':'').' />':' '); + } + $r->print('</td>'. + &Apache::loncommon::end_data_table_row()); } $r->print(&Apache::loncommon::end_data_table(). '<input type="hidden" name="action" value="'.$env{'form.action'}.'" />'. @@ -3626,7 +3745,7 @@ ENDCCF } # -------------------------------------------------------- sub make_script_template { - my ($role) = @_; + my ($role,$crstype) = @_; my %full_c=(); my %full_d=(); my %full_s=(); @@ -3640,6 +3759,7 @@ sub make_script_template { $full_d{$priv}=1; } foreach my $item (split(/\:/,$Apache::lonnet::pr{'cr:s'})) { + next if (($crstype eq 'Community') && ($item eq 'bre&S')); my ($priv,$restrict)=split(/\&/,$item); $full_s{$priv}=1; } @@ -4297,7 +4417,7 @@ sub print_main_menu { linktitle => $linktitle{$crstype}{'groups'}, }, { - linktext => 'Change Logs', + linktext => 'Change Log', icon => 'document-properties.png', #help => 'Course_User_Logs', url => '/adm/createuser?action=changelogs', @@ -5041,7 +5161,6 @@ sub print_userchangelogs_display { my %roleslog=&Apache::lonnet::dump('nohist_rolelog',$cdom,$cnum); if ((keys(%roleslog))[0]=~/^error\:/) { undef(%roleslog); } - $r->print('<form action="/adm/createuser" method="post" name="'.$formname.'">'); my %saveable_parameters = ('show' => 'scalar',); &Apache::loncommon::store_course_settings('roles_log', \%saveable_parameters); @@ -5076,7 +5195,6 @@ sub print_userchangelogs_display { } my (%whodunit,%changed,$version); ($version) = ($r->dir_config('lonVersion') =~ /^([\d\.]+)\-/); - $r->print(&role_display_filter($formname,$cdom,$cnum,\%curr,$version,$crstype)); my ($minshown,$maxshown); $minshown = 1; my $count = 0; @@ -5087,8 +5205,29 @@ sub print_userchangelogs_display { } } - # Collect user change log data - my $content = ''; + # Form Header + $r->print('<form action="/adm/createuser" method="post" name="'.$formname.'">'. + &role_display_filter($formname,$cdom,$cnum,\%curr,$version,$crstype)); + + # Create navigation + my ($nav_script,$nav_links) = &userlogdisplay_nav($formname,\%curr,$more_records); + my $showntableheader = 0; + + # Table Header + my $tableheader = + &Apache::loncommon::start_data_table_header_row() + .'<th> </th>' + .'<th>'.&mt('When').'</th>' + .'<th>'.&mt('Who made the change').'</th>' + .'<th>'.&mt('Changed User').'</th>' + .'<th>'.&mt('Role').'</th>' + .'<th>'.&mt('Section').'</th>' + .'<th>'.&mt('Context').'</th>' + .'<th>'.&mt('Start').'</th>' + .'<th>'.&mt('End').'</th>' + .&Apache::loncommon::end_data_table_header_row(); + + # Display user change log data foreach my $id (sort { $roleslog{$b}{'exe_time'}<=>$roleslog{$a}{'exe_time'} } (keys(%roleslog))) { next if (($roleslog{$id}{'exe_time'} < $curr{'rolelog_start_date'}) || ($roleslog{$id}{'exe_time'} > $curr{'rolelog_end_date'})); @@ -5110,7 +5249,14 @@ sub print_userchangelogs_display { } $count ++; next if ($count < $minshown); - + unless ($showntableheader) { + $r->print($nav_script + .$nav_links + .&Apache::loncommon::start_data_table() + .$tableheader); + $r->rflush(); + $showntableheader = 1; + } if ($whodunit{$roleslog{$id}{'exe_uname'}.':'.$roleslog{$id}{'exe_udom'}} eq '') { $whodunit{$roleslog{$id}{'exe_uname'}.':'.$roleslog{$id}{'exe_udom'}} = &Apache::loncommon::plainname($roleslog{$id}{'exe_uname'},$roleslog{$id}{'exe_udom'}); @@ -5149,7 +5295,7 @@ sub print_userchangelogs_display { if ($chgcontext ne '' && $lt{$chgcontext} ne '') { $chgcontext = $lt{$chgcontext}; } - $content .= + $r->print( &Apache::loncommon::start_data_table_row() .'<td>'.$count.'</td>' .'<td>'.&Apache::lonlocal::locallocaltime($roleslog{$id}{'exe_time'}).'</td>' @@ -5160,30 +5306,34 @@ sub print_userchangelogs_display { .'<td>'.$chgcontext.'</td>' .'<td>'.$rolestart.'</td>' .'<td>'.$roleend.'</td>' - .&Apache::loncommon::end_data_table_row(); + .&Apache::loncommon::end_data_table_row()."\n"); } - # Form Footer - my $form_footer = - '<input type="hidden" name="page" value="'.$curr{'page'}.'" />' - .'<input type="hidden" name="action" value="changelogs" />' - .'</form>'; - - # Only display table, if content is available (has been collected above) - if (!$content) { + if ($showntableheader) { # Table footer, if content displayed above + $r->print(&Apache::loncommon::end_data_table() + .$nav_links); + } else { # No content displayed above $r->print('<p class="LC_info">' .&mt('There are no records to display.') .'</p>' ); - $r->print($form_footer); - return; } - # Content to display, so create navigation and display table + # Form Footer + $r->print( + '<input type="hidden" name="page" value="'.$curr{'page'}.'" />' + .'<input type="hidden" name="action" value="changelogs" />' + .'</form>'); + return; +} - # Create Navigation: - # Navigation Script - my $nav_script = <<"ENDSCRIPT"; +sub userlogdisplay_nav { + my ($formname,$curr,$more_records) = @_; + my ($nav_script,$nav_links); + if (ref($curr) eq 'HASH') { + # Create Navigation: + # Navigation Script + $nav_script = <<"ENDSCRIPT"; <script type="text/javascript"> // <![CDATA[ function chgPage(caller) { @@ -5193,57 +5343,31 @@ function chgPage(caller) { if (caller == 'next') { document.$formname.page.value ++; } - document.$formname.submit(); + document.$formname.submit(); return; } // ]]> </script> ENDSCRIPT - # Navigation Buttons - my $nav_links; - $nav_links = '<p>'; - if (($curr{'page'} > 1) || ($more_records)) { - if ($curr{'page'} > 1) { - $nav_links .= '<input type="button"' - .' onclick="javascript:chgPage('."'previous'".');"' - .' value="'.&mt('Previous [_1] changes',$curr{'show'}) - .'" /> '; - } - if ($more_records) { - $nav_links .= '<input type="button"' - .' onclick="javascript:chgPage('."'next'".');"' - .' value="'.&mt('Next [_1] changes',$curr{'show'}) - .'" />'; + # Navigation Buttons + $nav_links = '<p>'; + if (($curr->{'page'} > 1) || ($more_records)) { + if ($curr->{'page'} > 1) { + $nav_links .= '<input type="button"' + .' onclick="javascript:chgPage('."'previous'".');"' + .' value="'.&mt('Previous [_1] changes',$curr->{'show'}) + .'" /> '; + } + if ($more_records) { + $nav_links .= '<input type="button"' + .' onclick="javascript:chgPage('."'next'".');"' + .' value="'.&mt('Next [_1] changes',$curr->{'show'}) + .'" />'; + } } + $nav_links .= '</p>'; } - $nav_links .= '</p>'; - - # Table Header - my $tableheader = - &Apache::loncommon::start_data_table_header_row() - .'<th> </th>' - .'<th>'.&mt('When').'</th>' - .'<th>'.&mt('Who made the change').'</th>' - .'<th>'.&mt('Changed User').'</th>' - .'<th>'.&mt('Role').'</th>' - .'<th>'.&mt('Section').'</th>' - .'<th>'.&mt('Context').'</th>' - .'<th>'.&mt('Start').'</th>' - .'<th>'.&mt('End').'</th>' - .&Apache::loncommon::end_data_table_header_row(); - - # Print Content - $r->print( - $nav_script - .$nav_links - .&Apache::loncommon::start_data_table() - .$tableheader - .$content - .&Apache::loncommon::end_data_table() - .$nav_links - .$form_footer - ); - return; + return ($nav_script,$nav_links); } sub role_display_filter { @@ -5882,7 +6006,8 @@ sub course_level_table { } } if (&Apache::lonnet::allowed('ccr',$thiscourse)) { - foreach my $cust (sort keys %customroles) { + foreach my $cust (sort(keys(%customroles))) { + next if ($crstype eq 'Community' && $customroles{$cust} =~ /bre\&S/); my $role = 'cr_cr_'.$env{'user.domain'}.'_'.$env{'user.name'}.'_'.$cust; $table .= &course_level_row($protectedcourse,$role,$area,$domain, $cust,\%sections_count,\%lt);