version 1.65, 2003/03/25 22:20:25
|
version 1.79, 2003/08/27 21:33:33
|
Line 51 Main handler for statistics and chart.
|
Line 51 Main handler for statistics and chart.
|
use Apache::lonproblemstatistics; |
use Apache::lonproblemstatistics; |
use Apache::lonstudentassessment; |
use Apache::lonstudentassessment; |
use Apache::lonpercentage; |
use Apache::lonpercentage; |
|
use Apache::lonmysql; |
=over 4 |
=over 4 |
|
|
=cut |
=cut |
Line 82 use Apache::lonproblemanalysis();
|
Line 82 use Apache::lonproblemanalysis();
|
use Apache::lonproblemstatistics(); |
use Apache::lonproblemstatistics(); |
use Apache::lonstudentassessment(); |
use Apache::lonstudentassessment(); |
use Apache::lonpercentage; |
use Apache::lonpercentage; |
|
use Apache::lonmysql; |
use Time::HiRes; |
use Time::HiRes; |
|
|
####################################################### |
####################################################### |
Line 205 upon the calling context.
|
Line 206 upon the calling context.
|
####################################################### |
####################################################### |
####################################################### |
####################################################### |
sub PrepareClasslist { |
sub PrepareClasslist { |
my $r = shift; |
|
my %Sections; |
my %Sections; |
&clear_classlist_variables(); |
&clear_classlist_variables(); |
# |
# |
Line 229 sub PrepareClasslist {
|
Line 229 sub PrepareClasslist {
|
} |
} |
} |
} |
# |
# |
|
# Deal with instructors with restricted section access |
|
if ($ENV{'request.course.sec'} !~ /^\s*$/) { |
|
@SelectedSections = ($ENV{'request.course.sec'}); |
|
} |
|
# |
# Set up %StudentData |
# Set up %StudentData |
@StudentDataOrder = qw/fullname username domain id section status/; |
@StudentDataOrder = qw/fullname username domain id section status/; |
foreach my $field (@StudentDataOrder) { |
foreach my $field (@StudentDataOrder) { |
Line 237 sub PrepareClasslist {
|
Line 242 sub PrepareClasslist {
|
$StudentData{$field}->{'width'} = |
$StudentData{$field}->{'width'} = |
$StudentData{$field}->{'base_width'}; |
$StudentData{$field}->{'base_width'}; |
} |
} |
|
# |
|
# get the status requested |
|
my $requested_status = 'Active'; |
|
$requested_status = $ENV{'form.Status'} if (exists($ENV{'form.Status'})); |
# |
# |
# Process the classlist |
# Process the classlist |
while (my ($student,$student_data) = each (%$classlist)) { |
while (my ($student,$student_data) = each (%$classlist)) { |
Line 265 sub PrepareClasslist {
|
Line 273 sub PrepareClasslist {
|
# |
# |
# Only put in the list those students we are interested in |
# Only put in the list those students we are interested in |
foreach my $sect (@SelectedSections) { |
foreach my $sect (@SelectedSections) { |
if (($sect eq 'all') || ($section eq $sect)) { |
if ( (($sect eq 'all') || |
|
($section eq $sect)) && |
|
(($studenthash->{'status'} eq $requested_status) || |
|
($requested_status eq 'Any')) |
|
){ |
push (@Students,$studenthash); |
push (@Students,$studenthash); |
last; |
last; |
} |
} |
Line 273 sub PrepareClasslist {
|
Line 285 sub PrepareClasslist {
|
} |
} |
# |
# |
# Put the consolidated section data in the right place |
# Put the consolidated section data in the right place |
@Sections = sort {$a cmp $b} keys(%Sections); |
if ($ENV{'request.course.sec'} !~ /^\s*$/) { |
unshift(@Sections,'all'); # Put 'all' at the front of the list |
@Sections = ($ENV{'request.course.sec'}); |
|
} else { |
|
@Sections = sort {$a cmp $b} keys(%Sections); |
|
unshift(@Sections,'all'); # Put 'all' at the front of the list |
|
} |
# |
# |
# Sort the Students |
# Sort the Students |
my $sortby = 'fullname'; |
my $sortby = 'fullname'; |
Line 284 sub PrepareClasslist {
|
Line 300 sub PrepareClasslist {
|
@Students = @TmpStudents; |
@Students = @TmpStudents; |
# |
# |
# Now deal with that current student thing.... |
# Now deal with that current student thing.... |
if (exists($ENV{'form.StudentAssessmentStudent'})) { |
$curr_student = undef; |
|
if (exists($ENV{'form.SelectedStudent'})) { |
my ($current_uname,$current_dom) = |
my ($current_uname,$current_dom) = |
split(':',$ENV{'form.StudentAssessmentStudent'}); |
split(':',$ENV{'form.SelectedStudent'}); |
my $i; |
my $i; |
for ($i = 0; $i<=$#Students; $i++) { |
for ($i = 0; $i<=$#Students; $i++) { |
next if (($Students[$i]->{'username'} ne $current_uname) || |
next if (($Students[$i]->{'username'} ne $current_uname) || |
Line 294 sub PrepareClasslist {
|
Line 311 sub PrepareClasslist {
|
$curr_student = $Students[$i]; |
$curr_student = $Students[$i]; |
last; # If we get here, we have our student. |
last; # If we get here, we have our student. |
} |
} |
if ($i == 0) { |
if (defined($curr_student)) { |
$prev_student = 'none'; |
if ($i == 0) { |
} else { |
$prev_student = undef; |
$prev_student = $Students[$i-1]; |
} else { |
} |
$prev_student = $Students[$i-1]; |
if ($i == $#Students) { |
} |
$next_student = 'none'; |
if ($i == $#Students) { |
} else { |
$next_student = undef; |
$next_student = $Students[$i+1]; |
} else { |
|
$next_student = $Students[$i+1]; |
|
} |
} |
} |
} |
} |
# |
# |
Line 313 sub PrepareClasslist {
|
Line 332 sub PrepareClasslist {
|
@SelectedStudentData = ($ENV{'form.StudentData'}); |
@SelectedStudentData = ($ENV{'form.StudentData'}); |
} |
} |
} else { |
} else { |
@SelectedStudentData = ('fullname'); |
@SelectedStudentData = ('username'); |
} |
} |
foreach (@SelectedStudentData) { |
foreach (@SelectedStudentData) { |
if ($_ eq 'all') { |
if ($_ eq 'all') { |
Line 325 sub PrepareClasslist {
|
Line 344 sub PrepareClasslist {
|
return; |
return; |
} |
} |
|
|
|
|
|
####################################################### |
|
####################################################### |
|
|
|
=pod |
|
|
|
=item get_students |
|
|
|
Returns a list of the selected students |
|
|
|
=cut |
|
|
|
####################################################### |
|
####################################################### |
|
sub get_students { |
|
if (! @Students) { |
|
&PrepareClasslist() |
|
} |
|
return @Students; |
|
} |
|
|
####################################################### |
####################################################### |
####################################################### |
####################################################### |
|
|
Line 340 selected student.
|
Line 380 selected student.
|
####################################################### |
####################################################### |
####################################################### |
####################################################### |
sub current_student { |
sub current_student { |
if (defined($curr_student)) { |
return $curr_student; |
return $curr_student; |
|
} else { |
|
return 'All Students'; |
|
} |
|
} |
} |
|
|
####################################################### |
####################################################### |
Line 362 in the list of students. Or something.
|
Line 398 in the list of students. Or something.
|
####################################################### |
####################################################### |
####################################################### |
####################################################### |
sub previous_student { |
sub previous_student { |
if (defined($prev_student)) { |
return $prev_student; |
return $prev_student; |
|
} else { |
|
return 'No Student Selected'; |
|
} |
|
} |
} |
|
|
####################################################### |
####################################################### |
Line 384 to be viewed.
|
Line 416 to be viewed.
|
####################################################### |
####################################################### |
####################################################### |
####################################################### |
sub next_student { |
sub next_student { |
if (defined($next_student)) { |
return $next_student; |
return $next_student; |
|
} else { |
|
return 'No Student Selected'; |
|
} |
|
} |
} |
|
|
####################################################### |
####################################################### |
Line 491 sub PrepareCourseData {
|
Line 519 sub PrepareCourseData {
|
my $name_length = length($seq->{'title'}); |
my $name_length = length($seq->{'title'}); |
my $num_parts = $seq->{'num_assess_parts'}; |
my $num_parts = $seq->{'num_assess_parts'}; |
# |
# |
# The number of columns needed for the summation text: |
# Use 3 digits for each the sum and total, which means 7 total... |
# " 1/5" = 1+3 columns, " 10/99" = 1+5 columns |
my $num_col = $num_parts+7; |
my $sum_length = 1+1+2*(length($num_parts)); |
|
my $num_col = $num_parts+$sum_length; |
|
if ($num_col < $name_length) { |
if ($num_col < $name_length) { |
$num_col = $name_length; |
$num_col = $name_length; |
} |
} |
Line 722 sub MapSelect {
|
Line 748 sub MapSelect {
|
Returns html for a selection box allowing the user to choose one (or more) |
Returns html for a selection box allowing the user to choose one (or more) |
of the sections in the course. |
of the sections in the course. |
|
|
|
Uses the package variables @Sections and @SelectedSections |
=over 4 |
=over 4 |
|
|
=item $elementname The name of the HTML form element |
=item $elementname The name of the HTML form element |
Line 730 of the sections in the course.
|
Line 757 of the sections in the course.
|
|
|
=item $numvisible The number of options to be visible |
=item $numvisible The number of options to be visible |
|
|
=item $selected Array ref to the names of the already selected sections. |
|
If undef, $ENV{'form.'.$elementname} is used. |
|
If $ENV{'form.'.$elementname} is also empty, none will be selected. |
|
|
|
=item $restriction Code reference to subroutine which returns true or |
|
false. The code must expect a reference to a sequence data structure. |
|
|
|
=back |
=back |
|
|
=cut |
=cut |
Line 749 sub SectionSelect {
|
Line 769 sub SectionSelect {
|
return; |
return; |
} |
} |
# |
# |
|
# Make sure we have the data we need to continue |
|
if (! @Sections) { |
|
&PrepareClasslist() |
|
} |
|
# |
# Build the form element |
# Build the form element |
my $Str = "\n"; |
my $Str = "\n"; |
$Str .= '<select name="'.$elementname.'" '; |
$Str .= '<select name="'.$elementname.'" '; |
Line 772 sub SectionSelect {
|
Line 797 sub SectionSelect {
|
return $Str; |
return $Str; |
} |
} |
|
|
|
############################################### |
|
############################################### |
|
|
|
=pod |
|
|
|
=item &Gather_Student_Data() |
|
|
|
Ensures all student data is up to date. |
|
|
|
=cut |
|
|
|
############################################### |
|
############################################### |
|
sub Gather_Student_Data { |
|
my ($r) = @_; |
|
my $c = $r->connection(); |
|
# |
|
my @Sequences = &Apache::lonstatistics::Sequences_with_Assess(); |
|
# |
|
my @Students = @Apache::lonstatistics::Students; |
|
# |
|
# Open the progress window |
|
my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin |
|
($r,'Statistics Compilation Status', |
|
'Statistics Compilation Progress', scalar(@Students)); |
|
# |
|
while (my $student = shift @Students) { |
|
return if ($c->aborted()); |
|
my ($status,undef) = &Apache::loncoursedata::ensure_current_data |
|
($student->{'username'},$student->{'domain'}, |
|
$ENV{'request.course.id'}); |
|
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state, |
|
'last student'); |
|
} |
|
&Apache::lonhtmlcommon::Close_PrgWin($r,\%prog_state); |
|
$r->rflush(); |
|
} |
|
|
################################################## |
################################################## |
################################################## |
################################################## |
sub DisplayClasslist { |
sub DisplayClasslist { |
Line 780 sub DisplayClasslist {
|
Line 843 sub DisplayClasslist {
|
my @Fields = ('fullname','username','domain','id','section'); |
my @Fields = ('fullname','username','domain','id','section'); |
# |
# |
my $Str=''; |
my $Str=''; |
|
if (! @Students) { |
|
if ($SelectedSections[0] eq 'all') { |
|
if (lc($ENV{'form.Status'}) eq 'any') { |
|
$Str .= '<h2>There are no students in the course.</h2>'; |
|
} elsif (lc($ENV{'form.Status'}) eq 'active') { |
|
$Str .= '<h2>There are no currently enrolled students in '. |
|
'the course.</h2>'; |
|
} elsif (lc($ENV{'form.Status'}) eq 'expired') { |
|
$Str .= '<h2>There are no previously enrolled '. |
|
'students in the course.</h2>'; |
|
} |
|
} else { |
|
my $sections; |
|
if (@SelectedSections == 1) { |
|
$sections = 'section '.$SelectedSections[0]; |
|
} elsif (@SelectedSections > 2) { |
|
$sections = 'sections '.join(', ',@SelectedSections); |
|
$sections =~ s/, ([^,])*$/, and $1/; |
|
} else { |
|
$sections = 'sections '.join(' and ',@SelectedSections); |
|
} |
|
if (lc($ENV{'form.Status'}) eq 'any') { |
|
$Str .= '<h2>There are no students in '.$sections.'.</h2>'; |
|
} elsif (lc($ENV{'form.Status'}) eq 'active') { |
|
$Str .= '<h2>There are no currently enrolled students '. |
|
'in '.$sections.'.</h2>'; |
|
} elsif (lc($ENV{'form.Status'}) eq 'expired') { |
|
$Str .= '<h2>There are no previously enrolled students '. |
|
'in '.$sections.'.</h2>'; |
|
} |
|
} |
|
$Str.= '<a href="/adm/statistics?reportSelected=student_assessment">'. |
|
'Return to the chart.</a>'; |
|
$r->print($Str); |
|
$r->rflush(); |
|
return; |
|
} |
|
|
|
# "Click" is asinine but it is probably not my place to change the world. |
|
$Str .= '<h2>Click on a students name or username to view their chart</h2>'; |
$Str .= '<table border="0"><tr><td bgcolor="#777777">'."\n"; |
$Str .= '<table border="0"><tr><td bgcolor="#777777">'."\n"; |
$Str .= '<table border="0" cellpadding="3"><tr bgcolor="#e6ffff">'."\n"; |
$Str .= '<table border="0" cellpadding="3"><tr bgcolor="#e6ffff">'."\n"; |
foreach my $field (@Fields) { |
foreach my $field (@Fields) { |
Line 800 sub DisplayClasslist {
|
Line 903 sub DisplayClasslist {
|
# |
# |
foreach my $field (@Fields) { |
foreach my $field (@Fields) { |
$Str .= '<td>'; |
$Str .= '<td>'; |
if ($field eq 'fullname') { |
if ($field eq 'fullname' || $field eq 'username') { |
$Str .= '<a href="/adm/statistics?reportSelected='; |
$Str .= '<a href="/adm/statistics?reportSelected='; |
$Str .= &Apache::lonnet::escape('student_assessment'); |
$Str .= &Apache::lonnet::escape('student_assessment'); |
$Str .= '&StudentAssessmentStudent='; |
$Str .= '&sort='.&Apache::lonnet::escape($ENV{'form.sort'}); |
|
$Str .= '&SelectedStudent='; |
$Str .= &Apache::lonnet::escape($sname).'">'; |
$Str .= &Apache::lonnet::escape($sname).'">'; |
$Str .= $student->{$field}.' '; |
$Str .= $student->{$field}.' '; |
$Str .= '</a>'; |
$Str .= '</a>'; |
Line 829 sub CreateMainMenu {
|
Line 933 sub CreateMainMenu {
|
# |
# |
my $Str = ''; |
my $Str = ''; |
# |
# |
$Str .= '<table border="0"><tbody><tr>'."\n"; |
$Str = '<input type="hidden" name="reportSelected" value="'.$current.'">'; |
$Str .= '<td></td>'."\n"; |
# $Str .= '<table border="0"><tbody><tr>'."\n"; |
$Str .= '<td align="center"><b>Select a Report</b></td>'."\n"; |
# $Str .= '<td align="center"><b>Report:</b></td>'."\n"; |
$Str .= '<td align="center"><b>Student Status</b></td></tr>'."\n"; |
# $Str .= '<td align="center">'; |
$Str .= '<tr>'."\n"; |
# $Str .= '<select name="reportSelected" '. |
# |
# 'onchange="document.Statistics.submit()">'."\n"; |
$Str .= '<td align="center"><input type="submit" name="Refresh" '; |
# foreach (sort(keys(%$reports))) { |
$Str .= 'value="Update Display" /></td>'."\n"; |
# $Str .= '<option value="'.$_.'"'; |
# |
# if($current eq $_) { |
$Str .= '<td align="center">'; |
# $Str .= ' selected'; |
$Str .= '<select name="reportSelected" >'."\n"; |
# } |
foreach (sort(keys(%$reports))) { |
# $Str .= '>'.$reports->{$_}.'</option>'."\n"; |
$Str .= '<option value="'.$_.'"'; |
# } |
if($current eq $_) { |
# $Str .= '</select></td>'."\n"; |
$Str .= ' selected'; |
# # |
} |
# $Str .= '<td>'.(' 'x30).'</td>'; |
$Str .= '>'.$reports->{$_}.'</option>'."\n"; |
# $Str .= '<td align="center">'. |
} |
# '<input type="submit" name="ClearCache" value="Clear Caches" />'. |
$Str .= '</select></td>'."\n"; |
# "</td>\n"; |
# |
# $Str .= '</tr></tbody></table>'."\n"; |
$Str .= '<td align="center">'; |
# $Str .= '<hr>'."\n"; |
$Str .= &Apache::lonhtmlcommon::StatusOptions($status, 'Statistics'); |
|
$Str .= '</td>'."\n"; |
|
# |
|
$Str .= '</tr></tbody></table>'."\n"; |
|
$Str .= '<hr>'."\n"; |
|
# |
# |
return $Str; |
return $Str; |
} |
} |
Line 874 sub handler {
|
Line 973 sub handler {
|
if ($loaderror) { return $loaderror; } |
if ($loaderror) { return $loaderror; } |
# |
# |
# Check for access |
# Check for access |
unless(&Apache::lonnet::allowed('vgr',$ENV{'request.course.id'})) { |
if (! &Apache::lonnet::allowed('vgr',$ENV{'request.course.id'})) { |
$ENV{'user.error.msg'}= |
$ENV{'user.error.msg'}= |
$r->uri.":vgr:0:0:Cannot view grades for complete course"; |
$r->uri.":vgr:0:0:Cannot view grades for complete course"; |
return HTTP_NOT_ACCEPTABLE; |
if (! &Apache::lonnet::allowed('vgr', |
|
$ENV{'request.course.id'}.'/'.$ENV{'request.course.sec'})) { |
|
$ENV{'user.error.msg'}= |
|
$r->uri.":vgr:0:0:Cannot view grades with given role"; |
|
return HTTP_NOT_ACCEPTABLE; |
|
} |
} |
} |
# |
# |
# Set document type for header only |
# Set document type for header only |
Line 899 sub handler {
|
Line 1003 sub handler {
|
# Extract form elements from query string |
# Extract form elements from query string |
&Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'}, |
&Apache::loncommon::get_unprocessed_cgi($ENV{'QUERY_STRING'}, |
['sort','reportSelected', |
['sort','reportSelected', |
'StudentAssessmentStudent']); |
'SelectedStudent']); |
if (! exists($ENV{'form.reportSelected'})) { |
if (! exists($ENV{'form.reportSelected'})) { |
$ENV{'form.reportSelected'} = 'student_assessment'; |
$ENV{'form.reportSelected'} = 'student_assessment'; |
} |
} |
Line 908 sub handler {
|
Line 1012 sub handler {
|
$r->print(&Apache::lonhtmlcommon::Title('Course Statistics and Charts')); |
$r->print(&Apache::lonhtmlcommon::Title('Course Statistics and Charts')); |
$r->rflush(); |
$r->rflush(); |
# |
# |
|
if (! &Apache::lonmysql::verify_sql_connection()) { |
|
my $serveradmin = $r->dir_config('lonAdmEMail'); |
|
$r->print(<<END); |
|
<h2><font color="Red">Unable to connect to database!</font></h2> |
|
<p> |
|
Please notify the server administrator <b>$serveradmin</b>. |
|
</p><p> |
|
Course Statistics and Charts cannot be retrieved until the database is |
|
restarted. Your data is intact but cannot be displayed at this time. |
|
</p> |
|
</body> |
|
</html> |
|
END |
|
return; |
|
} |
|
# |
|
# Clean out the caches |
|
if (exists($ENV{'form.ClearCache'})) { |
|
&Apache::loncoursedata::delete_caches($ENV{'requres.course.id'}); |
|
} |
|
# |
# Set up the statistics and chart environment |
# Set up the statistics and chart environment |
&PrepareClasslist($r); |
&PrepareClasslist(); |
&PrepareCourseData($r); |
&PrepareCourseData($r); |
# |
# |
# Begin form output |
# Begin form output |
Line 919 sub handler {
|
Line 1044 sub handler {
|
# Print main menu |
# Print main menu |
my %reports = ('classlist' => 'Class list', |
my %reports = ('classlist' => 'Class list', |
'problem_statistics' => 'Problem Statistics', |
'problem_statistics' => 'Problem Statistics', |
'student_assessment' => 'Student Assessment', |
'student_assessment' => 'Problem Status Chart', |
'percentage' => 'Correct-problems Plot', |
# 'percentage' => 'Correct-problems Plot', |
'option_response' => 'Option Response Analysis', |
# 'option_response' => 'Option Response Analysis', |
# 'activitylog' => 'Activity Log', |
# 'activitylog' => 'Activity Log', |
); |
); |
$r->print(&CreateMainMenu($ENV{'form.status'}, |
$r->print(&CreateMainMenu($ENV{'form.status'}, |
Line 939 sub handler {
|
Line 1064 sub handler {
|
&Apache::lonstudentassessment::BuildStudentAssessmentPage($r,$c); |
&Apache::lonstudentassessment::BuildStudentAssessmentPage($r,$c); |
} elsif($GoToPage eq 'DoDiffGraph' || $GoToPage eq 'PercentWrongGraph') { |
} elsif($GoToPage eq 'DoDiffGraph' || $GoToPage eq 'PercentWrongGraph') { |
# &Apache::lonproblemstatistics::BuildGraphicChart($r,$c); |
# &Apache::lonproblemstatistics::BuildGraphicChart($r,$c); |
} elsif($GoToPage eq 'classlist') { |
|
&DisplayClasslist($r); |
|
} elsif($GoToPage eq 'Correct-problems Plot') { |
} elsif($GoToPage eq 'Correct-problems Plot') { |
# &Apache::lonpercentage::BuildPercentageGraph($r,$c); |
# &Apache::lonpercentage::BuildPercentageGraph($r,$c); |
} |
} |