Annotation of loncom/interface/lonviewclasslist.pm, revision 1.14

1.1       matthew     1: # The LearningOnline Network with CAPA
                      2: # Handler to display the classlist 
                      3: #
1.14    ! raeburn     4: # $Id: lonviewclasslist.pm,v 1.13 2010/03/25 17:38:43 raeburn Exp $
1.1       matthew     5: #
                      6: # Copyright Michigan State University Board of Trustees
                      7: #
                      8: # This file is part of the LearningOnline Network with CAPA (LON-CAPA).
                      9: #
                     10: # LON-CAPA is free software; you can redistribute it and/or modify
                     11: # it under the terms of the GNU General Public License as published by
                     12: # the Free Software Foundation; either version 2 of the License, or
                     13: # (at your option) any later version.
                     14: #
                     15: # LON-CAPA is distributed in the hope that it will be useful,
                     16: # but WITHOUT ANY WARRANTY; without even the implied warranty of
                     17: # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     18: # GNU General Public License for more details.
                     19: #
                     20: # You should have received a copy of the GNU General Public License
                     21: # along with LON-CAPA; if not, write to the Free Software
                     22: # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
                     23: #
                     24: # /home/httpd/html/adm/gpl.txt
                     25: #
                     26: # http://www.lon-capa.org/
                     27: #
                     28: ###############################################################
                     29: ##############################################################
                     30: 
                     31: package Apache::lonviewclasslist;
                     32: 
                     33: use strict;
                     34: use Apache::loncoursedata();
                     35: use Apache::loncommon();
                     36: use Apache::lonhtmlcommon();
1.14    ! raeburn    37: use Apache::courseprefs();
1.1       matthew    38: use Apache::Constants qw(:common :http REDIRECT);
                     39: use Apache::lonlocal;
1.5       albertel   40: use Apache::lonnet;
1.1       matthew    41: 
                     42: 
                     43: ###################################################################
                     44: ###################################################################
                     45: 
                     46: =pod
                     47: 
                     48: =item &handler
                     49: 
                     50: The typical handler you see in all these modules.  Takes $r, the
                     51: http request, as an argument.  
                     52: 
                     53: =cut
                     54: 
                     55: ###################################################################
                     56: ###################################################################
                     57: sub handler {
                     58:     my $r=shift;
                     59:     if ($r->header_only) {
                     60:         &Apache::loncommon::content_type($r,'text/html');
                     61:         $r->send_http_header;
                     62:         return OK;
                     63:     }
1.13      raeburn    64:     &Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'},
1.14    ! raeburn    65:                                  ['register','forceedit','action','symb','todocs']);
1.5       albertel   66:     if (! ($env{'request.course.fn'})) {
                     67:         $env{'user.error.msg'}=
1.1       matthew    68:             "/adm/viewclasslist:not in course role";
                     69:         return HTTP_NOT_ACCEPTABLE; 
                     70:     }
                     71:     &Apache::loncommon::content_type($r,'text/html');
                     72:     $r->send_http_header;
                     73:     #
1.13      raeburn    74:     my $start_page;
                     75:     if ($env{'form.register'}) {
                     76:         $start_page = &Apache::loncommon::start_page('Classlist',undef,
                     77:                              {'force_register' => $env{'form.register'}});
                     78:     } else {
                     79:         my $brcrum = [{'href' => 'adm/viewclasslist',
                     80:                        'text' => 'View Classlist'},];
                     81:         $start_page = &Apache::loncommon::start_page('Classlist',undef,
                     82:                                                      {'bread_crumbs' => $brcrum});
                     83:     }
1.1       matthew    84:     $r->print(<<ENDHEADER);
1.6       albertel   85: $start_page
1.1       matthew    86: ENDHEADER
1.8       raeburn    87: 
                     88:     # Get classlist view settings
                     89:     my %viewsettings = &retrieve_view_settings();
                     90: 
1.14    ! raeburn    91:     if (($env{'form.forceedit'}) || ($env{'form.action'} eq 'setconfig'))  {
        !            92:         if (&Apache::lonnet::allowed('opa',$env{'request.course.id'})) {
        !            93:             my $crstype = &Apache::loncommon::course_type();
        !            94:             my $cdom = $env{'course.'.$env{'request.course.id'}.'.domain'};
        !            95:             my $cnum = $env{'course.'.$env{'request.course.id'}.'.num'};
        !            96:             my $rosterprefs = &roster_prefs($crstype);
        !            97:             my $allitems = {};
        !            98:             if ($env{'form.action'} eq 'setconfig') {
        !            99:                 my %values=&Apache::lonnet::dump('environment',$cdom,$cnum);
        !           100:                 if (keys(%values) > 0) {
        !           101:                     my ($numchanged,%changes,%disallowed);
        !           102:                     my $prefs = {
        !           103:                                     classlists => $rosterprefs,
        !           104:                                 };
        !           105:                     $changes{'classlists'} = {};
        !           106:                     &Apache::courseprefs::process_changes($cdom,'classlists',\%values,
        !           107:                                                           $rosterprefs,
        !           108:                                                           $changes{'classlists'},
        !           109:                                                           $allitems,\%disallowed,$crstype);
        !           110:                     my $message;
        !           111:                     if (keys(%{$changes{'classlists'}}) > 0) {
        !           112:                         my $actions = ['classlists'];
        !           113:                         $message =
        !           114:                             &Apache::courseprefs::store_changes($cdom,$cnum,$actions,
        !           115:                                                                 $actions,$prefs,\%values,
        !           116:                                                                 \%changes,$crstype);
        !           117:                     } else {
        !           118:                         if ($crstype eq 'Community') {
        !           119:                             $message = &mt('No changes made to community settings.');
        !           120:                         } else {
        !           121:                             $message = &mt('No changes made to course settings.');
        !           122:                         }
        !           123:                     }
        !           124:                     $r->print(&Apache::loncommon::confirmwrapper($message));
        !           125:                } else {
        !           126:                     $r->print('<div class="LC_info">'.
        !           127:                               &mt('Unable to retrieve current settings.').'<br />'.
        !           128:                               &mt('No changes saved.').
        !           129:                               '</div>');
        !           130:                }
        !           131:            } else {
        !           132:                my $current = {};
        !           133:                my @settings = ('student_classlist_view','student_classlist_opt_in',
        !           134:                                'student_classlist_portfiles');
        !           135:                foreach my $setting (@settings) {
        !           136:                    $current->{$setting} = $env{"course.$env{'request.course.id'}.$setting"};
        !           137:                }
        !           138:                my ($output,$rowtotal) =
        !           139:                    &Apache::courseprefs::print_config_box($r,$cdom,'display',
        !           140:                                                           'viewableroster',
        !           141:                                                           $rosterprefs,$current,
        !           142:                                                           $allitems,$crstype);
        !           143:                if ($output) {
        !           144:                    $r->print('<form method="post" name="display" action="/adm/viewclasslist">'."\n".
        !           145:                              '<input type="hidden" name="action" value="setconfig" />'."\n".
        !           146:                              '<input type="hidden" name="register" value="'.$env{'form.register'}.'" />'."\n".
        !           147:                              '<input type="hidden" name="forceedit" value="'.$env{'form.forceedit'}.'" />'."\n");
        !           148:                    if ($env{'form.symb'}) {
        !           149:                         $r->print('<input type="hidden" name="symb" value="'.$env{'form.symb'}.'" />'."\n");
        !           150:                    }
        !           151:                    if ($env{'form.symb'}) {
        !           152:                         $r->print('<input type="hidden" name="todocs" value="'.$env{'form.todocs'}.'" />'."\n");
        !           153:                    }
        !           154:                    $r->print('<div class="LC_left_float">'.
        !           155:                              $output.
        !           156:                              '</div><br clear="all" />'.
        !           157:                              '<input type="submit" value="'.&mt('Save').'" />'.
        !           158:                              '</form>');
        !           159:                } else {
        !           160:                    $r->print('<div class="LC_info">'.
        !           161:                              &mt('No student-viewable course roster settings available.').
        !           162:                              '</div>');
        !           163:                }
        !           164:            }
        !           165:         } else {
        !           166:             $r->print('<div class="LC_info">'.
        !           167:                       &mt('You do not have rights to modify student-viewable course roster settings.').
        !           168:                       '</div>');
        !           169:         }
1.8       raeburn   170:     } else {
1.14    ! raeburn   171: 
        !           172:         # Print classlist
        !           173:         if (keys(%viewsettings) > 0) {
        !           174:             $r->print(&html_classlist($r,\%viewsettings));
        !           175:         } else {
        !           176:             $r->print('<div class="LC_info">'.
        !           177:                       &mt("Display of a student-viewable course roster is not currently enabled.").
        !           178:                       '</div>');
        !           179:         }
1.1       matthew   180:     }
                    181:     #
                    182:     # Finish up
1.6       albertel  183:     $r->print(&Apache::loncommon::end_page());
1.1       matthew   184:     return OK;
                    185: }
                    186: 
1.8       raeburn   187: sub retrieve_view_settings {
                    188:     my %viewsettings;
                    189:     if (exists($env{'request.course.id'})) {
                    190:         my $cid = $env{'request.course.id'};
                    191:         my $viewpermission = 'course.'.$cid.'.student_classlist_view';
                    192:         my $student_opt_in = 'course.'.$cid.'.student_classlist_opt_in';
                    193:         my $portfiles_link = 'course.'.$cid.'.student_classlist_portfiles';
                    194:         if (exists($env{$viewpermission}) &&
                    195:             $env{$viewpermission} =~ /^(all|section)$/) {
                    196:             $viewsettings{'permission'} = $env{$viewpermission};
                    197:             if ($viewsettings{'permission'} =~ /^section$/i) {
                    198:                 $viewsettings{'limit_to_section'} = 1;
                    199:             } else {
                    200:                 $viewsettings{'limit_to_section'} = 0;
                    201:             }
                    202:             $viewsettings{'student_opt_in'} = $env{$student_opt_in};
                    203:             $viewsettings{'portfiles_link'} = $env{$portfiles_link};
                    204:         }
1.1       matthew   205:     }
1.8       raeburn   206:     return %viewsettings;
1.1       matthew   207: }
                    208: 
1.14    ! raeburn   209: sub roster_prefs {
        !           210:     my ($crstype) = @_;
        !           211:     my %lt;
        !           212:     if ($crstype eq 'Community') {
        !           213:         %lt = &Apache::lonlocal::texthash (
        !           214:                   stuv => 'Member-viewable membership list options',
        !           215:                   stul => 'Member agreement needed to be listed',
        !           216:         );
        !           217:     } else {
        !           218:         %lt = &Apache::lonlocal::texthash(
        !           219:                   stuv => 'Student-viewable classlist options',
        !           220:                   stul => 'Student agreement needed to be listed',
        !           221:         );
        !           222:     }
        !           223:     $lt{'incl'} = &mt('Include link to accessible portfolio files');
        !           224: 
        !           225:     return 
        !           226:         { text => 'Student-viewable roster settings',
        !           227:           header => [ {col1 => 'Setting',
        !           228:                        col2 => $lt{'stuv'}}],
        !           229:           ordered => ['student_classlist_view',
        !           230:                       'student_classlist_opt_in',
        !           231:                       'student_classlist_portfiles'],
        !           232:           itemtext => {
        !           233:                        student_classlist_view        => $lt{'stuv'},
        !           234:                        student_classlist_opt_in      => $lt{'stul'},
        !           235:                        student_classlist_portfiles   => $lt{'incl'},
        !           236:                       },
        !           237:         };
        !           238: }
        !           239: 
1.1       matthew   240: sub html_classlist {
1.8       raeburn   241:     my ($r,$viewsettings) = @_;
                    242:     my ($Str,$title,$secdisplay,$cid,$cdom,$cnum,$listtype,%publicroster);
                    243:     my $fullroster = &Apache::loncoursedata::get_classlist();
                    244:     my $classlist;
                    245: 
                    246:     if ($env{'form.action'} eq 'setenv') {
                    247:         $Str .= &process_student_prefs();
                    248:     }
                    249:     $Str .= '<h3>'.&mt('Student-viewable course roster').'</h3>';
                    250: 
                    251:     $cid = $env{'request.course.id'};
                    252:     $cdom = $env{'course.'.$cid.'.domain'};
                    253:     $cnum = $env{'course.'.$cid.'.num'};
                    254: 
                    255:     if ($viewsettings->{'limit_to_section'}) {
1.5       albertel  256:         if ($env{'request.course.sec'} eq '') {
1.8       raeburn   257:             $title = '<h4>'.&mt('Students with no section').'</h4>';
                    258:             $listtype = 'without a section';
                    259:         } else {
                    260:             $title ='<h4>'.&mt('Students in section "[_1]"',
                    261:                                $env{'request.course.sec'}).'</h4>';
                    262:             $listtype = 'in the section';
                    263:             $secdisplay = " ($env{'request.course.sec'}) ";
                    264:         }
                    265:     } else {
                    266:         $title .= '<h4>'.&mt('Students in any section').'</h4>';
                    267:         $listtype = 'in the course';
                    268:     }
                    269: 
                    270:     if ($viewsettings->{'student_opt_in'}) {
                    271:         if ($env{'request.role'} =~ /^st/)  {
                    272:             $Str .= &print_roster_form();
                    273:         }
                    274:         %publicroster = &Apache::lonnet::dump('publicroster',$cdom,$cnum);
                    275:     }
                    276: 
                    277:     $Str .= $title;
                    278: 
                    279:     my $fullcount = 0;
                    280:     my $publiccount = 0;
                    281:     my $displaycount = 0;
                    282:     my $sectionidx  = &Apache::loncoursedata::CL_SECTION();
                    283:     my $statusidx   = &Apache::loncoursedata::CL_STATUS();
                    284: 
                    285:     foreach my $student (keys(%{$fullroster})) {
                    286:         my $section  = $fullroster->{$student}->[$sectionidx];
                    287:         my $status   = $fullroster->{$student}->[$statusidx];
                    288:         next if (lc($status) ne 'active');
                    289:         if ($viewsettings->{'limit_to_section'}) {
                    290:             next if ($section ne $env{'request.course.sec'});
                    291:         }
                    292:         $fullcount ++;
                    293:         if ($viewsettings->{'student_opt_in'}) {
                    294:             if ($publicroster{$student}) {
                    295:                 $classlist->{$student} = $fullroster->{$student};
                    296:                 $publiccount ++;
                    297:             }
                    298:         } else {
                    299:             $classlist->{$student} = $fullroster->{$student};
                    300:         }
                    301:     }
                    302:     if ($viewsettings->{'student_opt_in'}) {
                    303:         $displaycount = $publiccount;
                    304:         if ($fullcount > $publiccount) {
                    305:             if ($publiccount) {
                    306:                 $Str .= &mt('Only students who have opted to be listed in the roster ([_1] out of [_2] students) are shown.',$publiccount,$fullcount).'<br />';
                    307:             } else {
                    308:                 if ($fullcount == 1) {
                    309:                     $Str .= &mt('The single student '.$listtype.'[_1] has opted not to be listed in the roster.',$secdisplay);
                    310:                 } else {
                    311:                     $Str .= &mt('None of the [_1] students '.$listtype.'[_2] have opted to be listed in the roster.',$fullcount,$secdisplay);
                    312:                 }
                    313:                 return $Str;
                    314:             }
1.1       matthew   315:         } else {
1.8       raeburn   316:             if ($fullcount > 1) {
                    317:                 $Str .= &mt('All [_1] students '.$listtype.'[_2] have opted to be listed in the roster.',$fullcount,$secdisplay);
                    318:             } elsif ($fullcount == 1) {
                    319:                 $Str .= &mt('The single student '.$listtype.'[_1] has opted to be listed in the roster.',$secdisplay);
                    320:             }
                    321:         }
                    322:     } else {
                    323:         $displaycount = $fullcount;
                    324:         if ($fullcount > 1) {
                    325:             $Str .= &mt('All [_1] students '.$listtype.'[_2] are listed in the roster.',$fullcount,$secdisplay);
                    326:         } elsif ($fullcount == 1) {
                    327:             $Str .= &mt('There is only a single student '.$listtype.'[_1]',$secdisplay);
1.1       matthew   328:         }
                    329:     }
1.8       raeburn   330:     undef($fullroster);
                    331: 
                    332:     if (!$displaycount) {
                    333:         $Str .= &mt('There are currently no students to display.');
                    334:         return $Str;
                    335:     }
                    336: 
1.1       matthew   337:     # Set up a couple variables.
                    338:     my $usernameidx = &Apache::loncoursedata::CL_SNAME();
                    339:     my $domainidx   = &Apache::loncoursedata::CL_SDOM();
                    340:     my $fullnameidx = &Apache::loncoursedata::CL_FULLNAME();
1.8       raeburn   341: 
1.1       matthew   342:     # Sort the students
                    343:     my $sortby = $fullnameidx;
                    344:     my @Sorted_Students = sort {
                    345:         lc($classlist->{$a}->[$sortby])  cmp lc($classlist->{$b}->[$sortby])
                    346:         } (keys(%$classlist));
1.8       raeburn   347:     $Str .= '<br />'.&Apache::loncommon::start_data_table()."\n".
                    348:             &Apache::loncommon::start_data_table_header_row()."\n".
1.1       matthew   349:         '<th></th>'. # for the count
                    350:         '<th>'.&mt('Student').'</th>'.
                    351:         '<th>'.&mt('Username').'</th>';
1.8       raeburn   352:     if (! $viewsettings->{'limit_to_section'}) {
1.1       matthew   353:         $Str .= '<th>'.&mt('Section').'</th>';
                    354:     }
1.8       raeburn   355:     if ($viewsettings->{'portfiles_link'}) {
                    356:         $Str .= '<th>'.&mt('Available Portfolio files').'</th>';
                    357:     }
                    358:     $Str .= &Apache::loncommon::end_data_table_header_row();
1.1       matthew   359:     my $count ++;
                    360:     foreach my $student (@Sorted_Students) {
                    361:         my $username = $classlist->{$student}->[$usernameidx];
                    362:         my $domain   = $classlist->{$student}->[$domainidx];
                    363:         my $fullname = $classlist->{$student}->[$fullnameidx];
                    364:         if ($fullname =~ /^\s*$/) {
                    365:             $fullname = &mt('Name not given');
                    366:         }
                    367:         my $section  = $classlist->{$student}->[$sectionidx];
1.8       raeburn   368:         if ($section eq '') {
                    369:             $section = &mt('none');
1.1       matthew   370:         }
1.8       raeburn   371:         $Str .= &Apache::loncommon::start_data_table_row()."\n".
1.1       matthew   372:             '<td>'.$count++.'</td>'.
                    373:             '<td>'.&Apache::loncommon::aboutmewrapper($fullname,
                    374:                                                       $username,
                    375:                                                       $domain).'</td>'.
                    376:             '<td>'.('&nbsp;'x2).
                    377:             &Apache::loncommon::messagewrapper
1.2       raeburn   378:             ('<img src="/adm/lonIcons/mailto.gif" border="0" />&nbsp;'.
1.11      raeburn   379:              $username.':'.$domain,$username,$domain).'</td>';
1.8       raeburn   380:         if (! $viewsettings->{'limit_to_section'}) {
1.1       matthew   381:             $Str .= '<td>'.$section.'</td>';
                    382:         }
1.8       raeburn   383:         if ($viewsettings->{'portfiles_link'}) {
                    384:             my $filecounts = &Apache::lonaboutme::portfolio_files($r,'showlink',undef,undef,$domain,$username,$fullname);
                    385:             my $link;
                    386:             if (ref($filecounts) eq 'HASH') {
                    387:                 $link = &mt('[quant,_1,file,files,No files]',$filecounts->{'both'});
                    388:                 if ($filecounts->{'both'} > 0) {
                    389:                     $link = '<a href="/adm/'.$domain.'/'.$username.'/aboutme/portfolio?classlist">'.$link.'</a>'; 
                    390:                 }
                    391:             } else {
1.9       albertel  392:                 $link = '<span class="LC_error">'.&mt("Error retrieving file information.").'</span>';
1.8       raeburn   393:             }
                    394:             $Str .= '<td>'.$link.'</td>';
                    395:         }
                    396:         $Str .= &Apache::loncommon::end_data_table_row()."\n";
1.1       matthew   397:     }
1.8       raeburn   398:     $Str .= &Apache::loncommon::end_data_table();
1.1       matthew   399:     return $Str;
                    400: }
                    401: 
1.8       raeburn   402: sub print_roster_form {
                    403:     my $cid = $env{'request.course.id'};
                    404:     my $showinroster = $env{'environment.internal.'.$cid.'.showinroster'};
                    405:     my ($showoff,$showon);
                    406:     if ($showinroster) {
                    407:         $showon = ' checked="checked" ';
                    408:         $showoff = ' ';
                    409:     } else {
                    410:         $showoff = ' checked="checked" ';
                    411:         $showon = ' ';
                    412:     }
                    413:     my $output = '<hr /><h4>'.&mt('Your roster setting').'</h4>';
                    414:     if ($showinroster) {
                    415:         $output .= &mt('You are currently listed in the student-viewable roster.');
                    416:     } else {
                    417:         $output .=  &mt('You are currently <b>not</b> listed in the student-viewable roster.');
                    418:     }
                    419:     $output .= '<br />'.&mt('Include yourself in the roster?').'&nbsp;&nbsp;'.
                    420:         '<form name="studentparm" method="post">'.
                    421:         '<span class="LC_nobreak"><label><input type="radio" name="showinroster" value="1"'.$showon.'/>'.&mt('Yes').'</label>&nbsp;&nbsp;<label>'.
                    422:         '<input type="radio" name="showinroster" value="0"'.$showoff.'/>'.&mt('No').
                    423:         '</label></span><br /><br />'.
                    424:         '<input type="hidden" name="action" value="setenv" />'.
                    425:         '<input type="submit" name="studentsubmit" value="'.&mt('Save').'" /></form><hr />';
                    426:     return $output;
                    427: }
                    428: 
                    429: sub process_student_prefs {
                    430:     my $cid = $env{'request.course.id'};
                    431:     my $cdom = $env{'course.'.$cid.'.domain'};
                    432:     my $cnum = $env{'course.'.$cid.'.num'};
                    433:     my $uname = $env{'user.name'};
                    434:     my $udom = $env{'user.domain'};
                    435:     my $student = $uname.':'.$udom;
                    436:     my %pubroster = &Apache::lonnet::get('publicroster',[$student],$cdom,$cnum);
                    437:     my $visibility = &mt('off');
                    438:     my $showinroster = $env{'form.showinroster'};
                    439:     if ($showinroster) {
                    440:         $visibility = &mt('on');
                    441:     }
                    442:     my $sturoster = 0;
                    443:     if ($pubroster{$student}) {
                    444:         $sturoster = 1;
                    445:     }
                    446:     my $output;
                    447:     if ($sturoster ne $showinroster) {
                    448:         my %changeHash = (
                    449:             'environment.internal.'.$cid.'.showinroster' => $showinroster,
                    450:         );
                    451:         my $putresult = &Apache::lonnet::put('environment',
                    452:                                              \%changeHash,$udom,$uname);
                    453:         if ($putresult eq 'ok') {
1.10      raeburn   454:             &Apache::lonnet::appenv(\%changeHash);
1.8       raeburn   455:             my $result = &Apache::lonnet::put('publicroster',{$student => $showinroster,},$cdom,$cnum);
                    456:             if ($result eq 'ok') {
                    457:                 $output .= &mt('Display of your name in the student-viewable roster set to <b>[_1]</b>.',$visibility);
                    458:             } else {
1.9       albertel  459:                 $output .= '<span class="LC_error">'.&mt('Error occurred saving display setting.').'</span>';
1.8       raeburn   460:             }
                    461:         } else {
1.9       albertel  462:             $output .= '<span class="LC_error">'.&mt('Error occurred saving display setting.').'</span>';
1.8       raeburn   463:         }
                    464:     } else {
                    465:         $output .= &mt('Display of your name in the student-viewable roster unchanged (set to <b>[_1]</b>).',$visibility);
                    466:     }
                    467:     return $output;
                    468: }
                    469: 
                    470: 
                    471: 
                    472: 
1.1       matthew   473: ###################################################################
                    474: ###################################################################
                    475: 
                    476: 1;
                    477: __END__
                    478: 
                    479: 

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