version 1.54.10.4, 2011/11/18 22:35:33
|
version 1.71, 2014/03/28 15:07:01
|
Line 30 use strict;
|
Line 30 use strict;
|
use Apache::lonnet; |
use Apache::lonnet; |
use Apache::loncommon(); |
use Apache::loncommon(); |
use Apache::lonhtmlcommon(); |
use Apache::lonhtmlcommon(); |
|
use Apache::lonquickgrades(); |
use Apache::loncoursedata(); |
use Apache::loncoursedata(); |
use Apache::lonstatistics; |
use Apache::lonstatistics; |
use Apache::lonlocal; |
use Apache::lonlocal; |
use Apache::lonstathelpers; |
use Apache::lonstathelpers; |
|
use Apache::lonmsgdisplay(); |
use HTML::Entities(); |
use HTML::Entities(); |
use Time::Local(); |
use Time::Local(); |
use Spreadsheet::WriteExcel(); |
use Spreadsheet::WriteExcel(); |
use Crypt::PasswdMD5; |
|
use lib '/home/httpd/lib/perl/'; |
use lib '/home/httpd/lib/perl/'; |
use LONCAPA; |
use LONCAPA; |
|
|
Line 62 sub BuildStudentSubmissionsPage {
|
Line 63 sub BuildStudentSubmissionsPage {
|
# |
# |
&Apache::lonstatistics::PrepareClasslist(); |
&Apache::lonstatistics::PrepareClasslist(); |
# |
# |
|
$r->print( &Apache::lonhtmlcommon::breadcrumbs('Student Submission Reports')); |
|
&Apache::lonquickgrades::startGradeScreen($r,'statistics'); |
$r->print(&CreateInterface()); |
$r->print(&CreateInterface()); |
# |
# |
my @Students = @Apache::lonstatistics::Students; |
my @Students = @Apache::lonstatistics::Students; |
Line 77 sub BuildStudentSubmissionsPage {
|
Line 80 sub BuildStudentSubmissionsPage {
|
'<div class="LC_info">'.&mt('Loading student data...').'</div>'); |
'<div class="LC_info">'.&mt('Loading student data...').'</div>'); |
$r->rflush(); |
$r->rflush(); |
# |
# |
if ((exists($env{'form.problemchoice'}) || (exists($env{'form.allproblems'}))) && |
my %anoncounter = |
|
&Apache::lonnet::dump('nohist_anonsurveys', |
|
$env{'course.'.$env{'request.course.id'}.'.domain'}, |
|
$env{'course.'.$env{'request.course.id'}.'.num'}); |
|
if (exists($env{'form.problemchoice'}) && |
! exists($env{'form.SelectAnother'})) { |
! exists($env{'form.SelectAnother'})) { |
foreach my $button (@SubmitButtons) { |
foreach my $button (@SubmitButtons) { |
if ($button->{'name'} eq 'break') { |
if ($button->{'name'} eq 'break') { |
Line 95 sub BuildStudentSubmissionsPage {
|
Line 102 sub BuildStudentSubmissionsPage {
|
$r->print('<hr />'.$/); |
$r->print('<hr />'.$/); |
$r->rflush(); |
$r->rflush(); |
# |
# |
|
|
# |
|
# Get resource objects |
|
my $navmap = Apache::lonnavmaps::navmap->new(); |
|
|
|
# Determine which problems we are to analyze |
# Determine which problems we are to analyze |
my @Symbs = |
my @Symbs = |
&Apache::lonstathelpers::get_selected_symbs('problemchoice'); |
&Apache::lonstathelpers::get_selected_symbs('problemchoice'); |
if (defined($navmap)) { |
|
if ($env{'form.allproblems'}) { |
# If there are multi-part problems with anonymous survey and named |
my $iterator = $navmap->getIterator(undef, undef, undef, 1); |
# parts check if named was picked for display. |
while (my $curRes = $iterator->next()) { |
# |
next if (! ref($curRes)); |
my %mixed_named; |
if ($curRes->is_problem) { |
foreach my $envkey (%env) { |
my $symb = $curRes->symb; |
if ($envkey =~ /^form\.mixed_(\d+:\d+)$/) { |
unless(grep(/^\Q$symb\E$/,@Symbs)) { |
my $item = $1; |
push(@Symbs,$symb); |
if ($env{$envkey} =~ /^symb_(.+)$/) { |
} |
my $symb = &unescape($1); |
|
if (ref($mixed_named{$symb}) eq 'ARRAY') { |
|
push(@{$mixed_named{$symb}},$item); |
|
} else { |
|
@{$mixed_named{$symb}} = ($item); |
} |
} |
} |
} |
} |
} |
} |
} |
foreach my $selected (@Symbs) { |
# |
$r->print('<input type="hidden" name="problemchoice" value="'. |
# Get resource objects |
$selected.'" />'.$/); |
my $navmap = Apache::lonnavmaps::navmap->new(); |
} |
|
if (!defined($navmap)) { |
if (!defined($navmap)) { |
|
foreach my $selected (@Symbs) { |
|
$r->print('<input type="hidden" name="problemchoice" value="'. |
|
&escape($selected).'" />'.$/); |
|
if (ref($mixed_named{$selected}) eq 'ARRAY') { |
|
foreach my $item (@{$mixed_named{$selected}}) { |
|
$r->print('<input type="hidden" name="mixed_'.$item.'" value="'.&escape($selected).'" />'.$/); |
|
} |
|
} |
|
} |
$r->print('<div class="LC_error">'.&mt("Internal error").'</div>'); |
$r->print('<div class="LC_error">'.&mt("Internal error").'</div>'); |
return; |
return; |
} |
} |
my %already_seen; |
my %already_seen; |
my (@Problems,$show_named); |
my (@Problems,@anonProbs,@namedProbs,$show_named); |
unless (&Apache::loncommon::needs_gci_custom()) { |
|
unless ($env{'form.anonymized'} eq 'true') { |
|
$show_named = 1; |
|
} |
|
} |
|
foreach my $symb (@Symbs) { |
foreach my $symb (@Symbs) { |
my $resource = $navmap->getBySymb($symb); |
my $resource = $navmap->getBySymb($symb); |
push(@Problems,$resource); |
my ($hasanon,$hasnamed); |
|
if (ref($resource)) { |
|
foreach my $partid (@{$resource->parts}) { |
|
if (($anoncounter{$symb."\0".$partid}) || ($resource->is_anonsurvey($partid))) { |
|
unless (exists($mixed_named{$symb})) { |
|
$hasanon = 1; |
|
} |
|
} else { |
|
$hasnamed = 1; |
|
} |
|
} |
|
if ($hasanon) { |
|
push(@anonProbs,$resource); |
|
} elsif ($hasnamed) { |
|
push(@namedProbs,$resource); |
|
} |
|
} |
|
} |
|
if (@namedProbs > 0) { |
|
@Problems = @namedProbs; |
|
$show_named = 1; |
|
} elsif (@anonProbs > 0) { |
|
@Problems = @anonProbs; |
|
} |
|
foreach my $selected (@Symbs) { |
|
$r->print('<input type="hidden" name="problemchoice" value="'. |
|
&escape($selected).'" />'.$/); |
|
if (ref($mixed_named{$selected}) eq 'ARRAY') { |
|
foreach my $item (@{$mixed_named{$selected}}) { |
|
$r->print('<input type="hidden" name="mixed_'.$item.'" value="'.&escape($selected).'" />'.$/); |
|
} |
|
} |
} |
} |
# If these are to be anonymized, do a random shuffle of @Students. |
# If these are to be anonymized, do a random shuffle of @Students. |
unless ($show_named) { |
unless ($show_named) { |
&array_shuffle(\@Students); |
&array_shuffle(\@Students); |
} |
} |
|
# |
|
|
my $threshold = $env{'course.'.$env{'request.course.id'}.'.internal.anonsurvey_threshold'}; |
my $threshold = $env{'course.'.$env{'request.course.id'}.'.internal.anonsurvey_threshold'}; |
if ($threshold eq '') { |
if ($threshold eq '') { |
my %domconfig = |
my %domconfig = |
Line 154 sub BuildStudentSubmissionsPage {
|
Line 196 sub BuildStudentSubmissionsPage {
|
$threshold = 10; |
$threshold = 10; |
} |
} |
} |
} |
unless (&Apache::loncommon::needs_gci_custom()) { |
$r->print('<p>'. |
$threshold = 0; |
|
} |
|
# |
|
$r->print('<h4>'. |
|
&Apache::lonstatistics::section_and_enrollment_description(). |
&Apache::lonstatistics::section_and_enrollment_description(). |
'</h4>'); |
'</p>'); |
if (! scalar(@Problems) || ! defined($Problems[0])) { |
if (! scalar(@Problems) || ! defined($Problems[0])) { |
$r->print('resource is undefined'); |
$r->print(&mt('resource is undefined')); |
} elsif (!$show_named && @Students < $threshold) { |
} elsif (!$show_named && @Students < $threshold) { |
$r->print(&mt('The number of students matching the selection criteria is too few for display of submission data for anonymous surveys.').'<br />'.&mt('There must be at least [quant,_1,student].',$threshold).' '.&mt('Contact the LON-CAPA [_1]Helpdesk[_2] if you need the threshold to be changed for this course.','<a href="/adm/helpdesk?origurl=%2fadm%2fstatistics>','</a>')); |
$r->print(&mt('The number of students matching the selection criteria is too few for display of submission data for anonymous surveys.').'<br />'.&mt('There must be at least [quant,_1,student].',$threshold).' '.&mt('Contact a Domain Coordinator if you need the threshold to be changed for this course.')); |
} else { |
} else { |
if (scalar(@Problems) == 1) { |
if (scalar(@Problems) == 1) { |
my $resource = $Problems[0]; |
my $resource = $Problems[0]; |
Line 176 sub BuildStudentSubmissionsPage {
|
Line 214 sub BuildStudentSubmissionsPage {
|
} |
} |
} |
} |
if ($env{'form.output'} eq 'excel') { |
if ($env{'form.output'} eq 'excel') { |
&prepare_excel_output($r,\@Problems,\@Students,$show_named); |
&prepare_excel_output($r,\@Problems,\@Students,\%anoncounter,$show_named); |
} elsif ($env{'form.output'} eq 'csv') { |
} elsif ($env{'form.output'} eq 'csv') { |
&prepare_csv_output($r,\@Problems,\@Students,$show_named); |
&prepare_csv_output($r,\@Problems,\@Students,\%anoncounter,$show_named); |
} else { |
} else { |
&prepare_html_output($r,\@Problems,\@Students,$show_named); |
&prepare_html_output($r,\@Problems,\@Students,\%anoncounter,$show_named); |
} |
} |
} |
} |
$r->print('<hr />'); |
$r->print('<hr />'); |
Line 189 sub BuildStudentSubmissionsPage {
|
Line 227 sub BuildStudentSubmissionsPage {
|
&mt('Prepare Report').'" />'); |
&mt('Prepare Report').'" />'); |
$r->print(' 'x5); |
$r->print(' 'x5); |
$r->print('<p>'. |
$r->print('<p>'. |
&mt('Computing correct answers greatly increasese the amount of time required to prepare a report.'). |
&mt('Computing correct answers greatly increases the amount of time required to prepare a report.'). |
'</p>'); |
'</p>'); |
$r->print('<p>'. |
$r->print('<p>'. |
&mt('Please select problems and use the [_1]Prepare Report[_2] button to continue.','<b>','</b>'). |
&mt('Please select problems and use the [_1]Prepare Report[_2] button to continue.','<b>','</b>'). |
'</p>'); |
'</p>'); |
$r->print(&Apache::lonstathelpers::MultipleProblemSelector |
$r->print(&Apache::lonstathelpers::MultipleProblemSelector |
(undef,'problemchoice','Statistics')); |
(undef,'problemchoice','Statistics',\%anoncounter)); |
} |
} |
} |
} |
|
|
Line 216 sub array_shuffle {
|
Line 254 sub array_shuffle {
|
## |
## |
## get_extra_response_headers |
## get_extra_response_headers |
## |
## |
|
|
sub get_extra_response_headers { |
sub get_extra_response_headers { |
my ($show_named) = @_; |
my ($show_named) = @_; |
my @extra_resp_headers; |
my @extra_resp_headers; |
Line 274 sub get_headers {
|
Line 311 sub get_headers {
|
######################################################### |
######################################################### |
######################################################### |
######################################################### |
sub prepare_html_output { |
sub prepare_html_output { |
my ($r,$problems,$students,$show_named) = @_; |
my ($r,$problems,$students,$anoncounter,$show_named) = @_; |
my $c = $r->connection(); |
my $c = $r->connection(); |
my $salt = '$1$'.$Apache::lonnet::perlvar{'AnonymousSalt'}; |
|
# |
# |
# Set a flag for the case when there is just one problem |
# Set a flag for the case when there is just one problem |
my $single_response = 0; |
my $single_response = 0; |
Line 289 sub prepare_html_output {
|
Line 325 sub prepare_html_output {
|
my @extra_resp_headers = &get_extra_response_headers($show_named); |
my @extra_resp_headers = &get_extra_response_headers($show_named); |
# |
# |
# Create the table header |
# Create the table header |
my @student_columns; |
my @student_columns = &get_student_columns($show_named); |
if ($show_named) { |
|
@student_columns = @Apache::lonstatistics::SelectedStudentData; |
|
if (grep(/^all$/,@student_columns)) { |
|
@student_columns = qw(fullname username domain id section status groups comments); |
|
} |
|
} else { |
|
@student_columns = ('username'); |
|
} |
|
# |
|
my %headers; |
my %headers; |
my $student_column_count = scalar(@student_columns); |
my $student_column_count = scalar(@student_columns); |
$headers{'problem'} = qq{<th colspan="$student_column_count">\ </th>}; |
$headers{'problem'} = qq{<th colspan="$student_column_count">\ </th>}; |
foreach (@student_columns) { |
foreach my $field (@student_columns) { |
$headers{'student'}.= '<th>'.ucfirst($_).'</th>'; |
$headers{'student'}.= '<th>'.&mt(ucfirst($field)).'</th>'; |
} |
} |
# |
# |
# we put the headers into the %headers hash |
# we put the headers into the %headers hash |
my $total_col = scalar(@student_columns); |
my $total_col = $student_column_count; |
my $nonempty_part_headers = 0; |
my $nonempty_part_headers = 0; |
# |
# |
my %problem_analysis; |
my %problem_analysis; |
foreach my $prob (@$problems) { |
foreach my $prob (@$problems) { |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
$problem_analysis{$prob->src}=\%analysis; |
$problem_analysis{$prob->src}=\%analysis; |
|
my $symb = $prob->symb(); |
# |
# |
my $prob_span = 0; |
my $prob_span = 0; |
my $single_part = 0; |
my $single_part = 0; |
if (scalar(@{$prob->parts}) == 1) { |
if (scalar(@{$prob->parts}) == 1) { |
$single_part = 1; |
$single_part = 1; |
} |
} |
|
my $shown_parts = 0; |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
|
$shown_parts ++; |
my $part_span = 0; |
my $part_span = 0; |
my $responses = [$prob->responseIds($partid)]; |
my $responses = [$prob->responseIds($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
Line 361 sub prepare_html_output {
|
Line 396 sub prepare_html_output {
|
} |
} |
$prob_span += $part_span; |
$prob_span += $part_span; |
} |
} |
|
next if (!$shown_parts); |
my $title = $prob->compTitle; |
my $title = $prob->compTitle; |
unless (&Apache::loncommon::needs_gci_custom()) { |
|
($title) = ($prob->src =~ m{/([^/]+)$}); |
|
} |
|
if ($prob_span > 0) { |
if ($prob_span > 0) { |
$headers{'problem'}.= qq{<th colspan="$prob_span">$title</th>}; |
$headers{'problem'}.= qq{<th colspan="$prob_span">$title</th>}; |
} elsif ($single_response) { |
} elsif ($single_response) { |
Line 381 sub prepare_html_output {
|
Line 414 sub prepare_html_output {
|
qq{<th colspan="$student_column_count">\ </th>}. |
qq{<th colspan="$student_column_count">\ </th>}. |
$headers{'response'}; |
$headers{'response'}; |
} |
} |
my $full_header = $/.'<table>'.$/; |
#my $full_header = $/.'<table>'.$/; |
|
my $full_header = $/.&Apache::loncommon::start_data_table().$/; |
$full_header .= '<tr align="left">'.$headers{'problem'}.'</tr>'.$/; |
$full_header .= '<tr align="left">'.$headers{'problem'}.'</tr>'.$/; |
if ($nonempty_part_headers) { |
if ($nonempty_part_headers) { |
$full_header .= '<tr align="left">'.$headers{'part'}.'</tr>'.$/; |
$full_header .= '<tr align="left">'.$headers{'part'}.'</tr>'.$/; |
Line 392 sub prepare_html_output {
|
Line 426 sub prepare_html_output {
|
# Main loop |
# Main loop |
my $count; |
my $count; |
$r->print($/.$full_header.$/); |
$r->print($/.$full_header.$/); |
my $row_class = 'odd'; # css |
|
foreach my $student (@$students) { |
foreach my $student (@$students) { |
my $student_row_data; |
my $student_row_data; |
if ($count++ >= 30) { |
if ($count++ >= 30) { |
$r->print('</table>'.$/.$full_header.$/); |
$r->print(&Apache::loncommon::end_data_table().$/.$full_header.$/); |
$count = 0; |
$count = 0; |
} |
} |
last if ($c->aborted()); |
last if ($c->aborted()); |
Line 414 sub prepare_html_output {
|
Line 447 sub prepare_html_output {
|
$student_row_data .= '</td>'; |
$student_row_data .= '</td>'; |
} |
} |
} else { |
} else { |
my $anonid = &Crypt::PasswdMD5::unix_md5_crypt($student->{'username'}, |
$student_row_data = '<td valign="top" colspan="'.$student_column_count.'">'.&mt('Anonymized').'</td>'; |
$salt); |
|
$anonid = substr($anonid,length($salt)+1); |
|
$student_row_data = '<td valign="top" colspan="'.$student_column_count.'">'. |
|
$anonid.'</td>'; |
|
} |
} |
# |
# |
# Figure out what it is we need to output for this student |
# Figure out what it is we need to output for this student |
Line 426 sub prepare_html_output {
|
Line 455 sub prepare_html_output {
|
my %prob_data; |
my %prob_data; |
my $maxrow; |
my $maxrow; |
foreach my $prob (@$problems) { |
foreach my $prob (@$problems) { |
$prob_data{$prob->symb}={}; |
my $symb = $prob->symb; |
|
$prob_data{$symb}={}; |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
my @responses = $prob->responseIds($partid); |
my @responses = $prob->responseIds($partid); |
my @response_type = $prob->responseType($partid); |
my @response_type = $prob->responseType($partid); |
for (my $i=0;$i<=$#responses;$i++) { |
for (my $i=0;$i<=$#responses;$i++) { |
Line 492 sub prepare_html_output {
|
Line 527 sub prepare_html_output {
|
next if (! $maxrow && ! scalar(@essays)); |
next if (! $maxrow && ! scalar(@essays)); |
# |
# |
# Go through the problem data and output a row. |
# Go through the problem data and output a row. |
if ($row_class eq 'even') { |
|
$row_class = 'odd'; |
|
} else { |
|
$row_class = 'even'; |
|
} |
|
my $printed_something; |
my $printed_something; |
for (my $rows_output = 0;$rows_output<$maxrow;$rows_output++) { |
for (my $rows_output = 0;$rows_output<$maxrow;$rows_output++) { |
my $html; |
my $html; |
Line 520 sub prepare_html_output {
|
Line 550 sub prepare_html_output {
|
} |
} |
} |
} |
if (! $no_data) { |
if (! $no_data) { |
$r->print(qq{<tr class="$row_class">$student_row_data$html</tr>}.$/); |
$r->print(&Apache::loncommon::start_data_table_row().$student_row_data.$html. |
|
&Apache::loncommon::end_data_table_row().$/); |
$printed_something=1; |
$printed_something=1; |
} |
} |
} |
} |
if (@essays) { |
if (@essays) { |
my $tr = qq{<tr class="$row_class">}; |
my $tr = &Apache::loncommon::start_data_table_row(); |
my $td = qq{<td valign="top" class="essay" colspan="$total_col">}; |
my $td = qq{<td valign="top" class="essay" colspan="$total_col">}; |
if (! $printed_something) { |
if (! $printed_something) { |
$r->print($tr.$student_row_data.'</tr>'.$/); |
$r->print($tr.$student_row_data.&Apache::loncommon::end_data_table_row().$/); |
} |
} |
$r->print($tr.$td. |
$r->print($tr.$td. |
join('</td></tr>'.$/.$tr.$td,@essays).'</td></tr>'.$/); |
join('</td></tr>'.$/.$tr.$td,@essays).'</td></tr>'.$/); |
undef(@essays); |
undef(@essays); |
} |
} |
} # end of student loop |
} # end of student loop |
$r->print('</table>'.$/); |
$r->print(&Apache::loncommon::end_data_table().$/); |
return; |
return; |
} |
} |
|
|
Line 700 sub html_non_essay_results {
|
Line 731 sub html_non_essay_results {
|
push(@values,$response->{$original_header}); |
push(@values,$response->{$original_header}); |
} |
} |
} |
} |
|
# FIXME: Bug #6700 - Properly process multiple answer fields |
|
# Would be a good place here but the data needs to be properly |
|
# compiled somewhere before. |
|
# elsif (($resptype =~ /^(numerical|formula|custom)$/) && |
|
# (submission) && (ref)) { |
|
# de-ref...: push(@values,$response->{$original_header}); } |
} else { |
} else { |
@values = map { $response->{$_}; } @$headers; |
foreach my $original_header (@$headers) { |
|
if ($original_header eq 'Time') { |
|
push(@values,&Apache::lonlocal::locallocaltime($response->{$original_header})); |
|
} elsif (($original_header eq 'Submission') && |
|
!($resptype =~ /^(radiobutton|option|match|rank)$/)) { |
|
# encode all submissions which have not been encoded above |
|
push(@values,&HTML::Entities::encode($response->{$original_header},'<>&"')); |
|
} elsif (($original_header eq 'Correct') && |
|
($resptype eq 'radiobutton')) { |
|
# encode foil separators |
|
push(@values,&HTML::Entities::encode($response->{$original_header},'&')); |
|
} else { |
|
# A normal column |
|
push(@values,$response->{$original_header}); |
|
} |
|
} |
} |
} |
my $td = '<td valign="top">'; |
my $td = '<td valign="top">'; |
my $str = $td.join('</td>'.$td,@values).'</td>'; |
my $str = $td.join('</td>'.$td,@values).'</td>'; |
Line 717 sub html_non_essay_results {
|
Line 769 sub html_non_essay_results {
|
######################################################### |
######################################################### |
######################################################### |
######################################################### |
sub prepare_excel_output { |
sub prepare_excel_output { |
my ($r,$Problems,$Students,$show_named) = @_; |
my ($r,$Problems,$Students,$anoncounter,$show_named) = @_; |
my $c = $r->connection(); |
my $c = $r->connection(); |
my $salt = '$1$'.$Apache::lonnet::perlvar{'AnonymousSalt'}; |
|
# |
# |
# |
# |
# Determine the number of columns in the spreadsheet |
# Determine the number of columns in the spreadsheet |
my $columncount = 3; # username, domain, id |
my $columncount = 3; # username, domain, id |
my @extra_resp_headers = &get_extra_response_headers(); |
my @extra_resp_headers = &get_extra_response_headers($show_named); |
my $lastprob; |
my $lastprob; |
my %problem_analysis; |
my %problem_analysis; |
foreach my $prob (@$Problems) { |
foreach my $prob (@$Problems) { |
|
my $symb = $prob->symb(); |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
$problem_analysis{$prob->src}=\%analysis; |
$problem_analysis{$prob->src}=\%analysis; |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
|
|
my $responses = [$prob->responseIds($partid)]; |
my $responses = [$prob->responseIds($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
for (my $i=0;$i<scalar(@$responses);$i++) { |
for (my $i=0;$i<scalar(@$responses);$i++) { |
Line 746 sub prepare_excel_output {
|
Line 804 sub prepare_excel_output {
|
$lastprob = $prob; |
$lastprob = $prob; |
} |
} |
if ($columncount > 255) { |
if ($columncount > 255) { |
$r->print('<h1>'.&mt('Unable to complete request').'</h1>'.$/. |
$r->print('<p class="LC_error">'.&mt('Unable to complete request').'</p>'.$/. |
'<p>'.&mt('LON-CAPA is unable to produce your Excel spreadsheet because your selections will result in more than 255 columns. Excel allows only 255 columns in a spreadsheet.').'</p>'.$/. |
'<p class="LC_warning">'.&mt('LON-CAPA is unable to produce your Excel spreadsheet because your selections will result in more than 255 columns. Excel allows only 255 columns in a spreadsheet.').'</p>'.$/. |
'<p>'.&mt('Consider selecting fewer problems to generate reports on, or reducing the number of items per problem. Or use HTML or CSV output.').'</p>'.$/. |
'<p class="LC_warning">'.&mt('Consider selecting fewer problems to generate reports on, or reducing the number of items per problem. Or use HTML or CSV output.').'</p>'.$/); |
'<p>'.&mt('The last problem that will fit in the current spreadsheet is [_1].',$lastprob->compTitle).'</p>'); |
if (ref($lastprob)) { |
|
$r->print('<p>'.&mt('The last problem that will fit in the current spreadsheet is [_1].',$lastprob->compTitle).'</p>'); |
|
} |
$r->rflush(); |
$r->rflush(); |
return; |
return; |
} |
} |
# |
# |
# Print out a message telling them what we are doing |
# Print out a message telling them what we are doing |
if (scalar(@$Problems) > 1) { |
if (scalar(@$Problems) > 1) { |
$r->print('<h2>'. |
$r->print('<p class="LC_info">'. |
&mt('Preparing Excel spreadsheet of student responses to [_1] problems', |
&mt('Preparing Excel spreadsheet of student responses to [_1] problems', |
scalar(@$Problems)). |
scalar(@$Problems)). |
'</h2>'); |
'</p>'); |
} else { |
} else { |
$r->print('<h2>'. |
$r->print('<p class="LC_info">'. |
&mt('Preparing Excel spreadsheet of student responses'). |
&mt('Preparing Excel spreadsheet of student responses'). |
'</h2>'); |
'</p>'); |
} |
} |
$r->rflush(); |
$r->rflush(); |
# |
# |
Line 771 sub prepare_excel_output {
|
Line 831 sub prepare_excel_output {
|
my ($workbook,$filename,$format) = |
my ($workbook,$filename,$format) = |
&Apache::loncommon::create_workbook($r); |
&Apache::loncommon::create_workbook($r); |
return if (! defined($workbook)); |
return if (! defined($workbook)); |
my $worksheet = $workbook->addworksheet('Student Submission Data'); |
|
|
# Worksheet name |
|
# The length of a worksheet name in Excel 95 is limited to 31 characters. |
|
# Let's make sure to not exceed the maximum length in the translation files. |
|
# Otherwise, no Excel file would be created. |
|
my $worksheetname = &mt('Student Submission Data'); |
|
if (length($worksheetname) > 31) { |
|
# nice way to cut off too long name |
|
$worksheetname = substr($worksheetname,0,28,).'...'; |
|
} |
|
my $worksheet = $workbook->addworksheet($worksheetname); |
# |
# |
# Add headers to the worksheet |
# Add headers to the worksheet |
my $rows_output = 0; |
my $rows_output = 0; |
Line 784 sub prepare_excel_output {
|
Line 854 sub prepare_excel_output {
|
my $partid_row = $rows_output++; |
my $partid_row = $rows_output++; |
my $respid_row = $rows_output++; |
my $respid_row = $rows_output++; |
my $header_row = $rows_output++; |
my $header_row = $rows_output++; |
$worksheet->write($title_row ,0,'Problem Title',$format->{'bold'}); |
$worksheet->write($title_row ,0,&mt('Problem Title'),$format->{'bold'}); |
$worksheet->write($partid_row,0,'Part ID',$format->{'bold'}); |
$worksheet->write($partid_row,0,&mt('Part ID'),$format->{'bold'}); |
$worksheet->write($respid_row,0,'Response ID',$format->{'bold'}); |
$worksheet->write($respid_row,0,&mt('Response ID'),$format->{'bold'}); |
# Student headers |
# Student headers |
my @StudentColumns; |
my @StudentColumns = &get_student_columns($show_named); |
if ($show_named) { |
foreach my $field (@StudentColumns) { |
@StudentColumns = qw(username domain id section); |
$worksheet->write($header_row,$cols_output++,&mt(ucfirst($field)), |
} else { |
|
@StudentColumns = qw(username); |
|
} |
|
foreach (@StudentColumns) { |
|
$worksheet->write($header_row,$cols_output++,ucfirst($_), |
|
$format->{'bold'}); |
$format->{'bold'}); |
} |
} |
# Problem headers |
# Problem headers |
my %start_col; |
my %start_col; |
foreach my $prob (@$Problems) { |
foreach my $prob (@$Problems) { |
my $title = $prob->compTitle; |
my $title = $prob->compTitle; |
unless (&Apache::loncommon::needs_gci_custom()) { |
my $symb = $prob->symb(); |
($title) = ($prob->src =~ m{/([^/]+)$}); |
|
} |
|
$worksheet->write($title_row,$cols_output, |
$worksheet->write($title_row,$cols_output, |
$title,$format->{'h3'}); |
$title,$format->{'h3'}); |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
$worksheet->write($partid_row,$cols_output, |
$worksheet->write($partid_row,$cols_output, |
$prob->part_display($partid)); |
$prob->part_display($partid)); |
my $responses = [$prob->responseIds($partid)]; |
my $responses = [$prob->responseIds($partid)]; |
Line 834 sub prepare_excel_output {
|
Line 902 sub prepare_excel_output {
|
} |
} |
# |
# |
# Populate the worksheet with the student data |
# Populate the worksheet with the student data |
my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin |
my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin($r,scalar(@$Students)); |
($r,'Excel File Compilation Status', |
|
'Excel File Compilation Progress', |
|
scalar(@$Students),'inline',undef,'Statistics','stats_status'); |
|
my $max_row = $rows_output; |
my $max_row = $rows_output; |
foreach my $student (@$Students) { |
foreach my $student (@$Students) { |
last if ($c->aborted()); |
last if ($c->aborted()); |
$cols_output = 0; |
$cols_output = 0; |
my $student_row = $max_row; |
my $student_row = $max_row; |
my $anonid = &Crypt::PasswdMD5::unix_md5_crypt($student->{'username'}, |
|
$salt); |
|
$anonid = substr($anonid,length($salt)+1); |
|
foreach my $prob (@$Problems) { |
foreach my $prob (@$Problems) { |
|
my $symb = $prob->symb(); |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
my @Response = $prob->responseIds($partid); |
my @Response = $prob->responseIds($partid); |
my @ResponseType = $prob->responseType($partid); |
my @ResponseType = $prob->responseType($partid); |
for (my $i=0;$i<=$#Response;$i++) { |
for (my $i=0;$i<=$#Response;$i++) { |
Line 894 sub prepare_excel_output {
|
Line 962 sub prepare_excel_output {
|
} |
} |
} |
} |
} |
} |
# Fill in the remaining rows with the students data |
# Prepend current student's user information to all rows |
for (my $row = $student_row;$row<$max_row;$row++) { |
for (my $row = $student_row;$row<$max_row;$row++) { |
my $cols = 0; |
my $cols = 0; |
foreach my $field (@StudentColumns) { |
foreach my $field (@StudentColumns) { |
if ($show_named) { |
if ($show_named) { |
$worksheet->write($row,$cols++, |
my $value = $student->{$field}; |
$student->{$field}); |
if ($field eq 'comments') { |
|
$value = &Apache::lonmsgdisplay::retrieve_instructor_comments |
|
($student->{'username'},$student->{'domain'}); |
|
} |
|
$worksheet->write($row,$cols++,$value); |
} else { |
} else { |
$worksheet->write($row,$cols++, |
$worksheet->write($row,$cols++, |
$anonid); |
&mt('Anonymized')); |
} |
} |
} |
} |
} |
} |
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state, |
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state,'last student'); |
'last student'); |
|
} |
} |
&Apache::lonhtmlcommon::Close_PrgWin($r,\%prog_state); |
&Apache::lonhtmlcommon::Close_PrgWin($r,\%prog_state); |
# |
# |
Line 1014 sub excel_format_item {
|
Line 1085 sub excel_format_item {
|
######################################################### |
######################################################### |
######################################################### |
######################################################### |
sub prepare_csv_output { |
sub prepare_csv_output { |
my ($r,$problems,$students,$show_named) = @_; |
my ($r,$problems,$students,$anoncounter,$show_named) = @_; |
my $c = $r->connection(); |
my $c = $r->connection(); |
my $salt = '$1$'.$Apache::lonnet::perlvar{'AnonymousSalt'}; |
|
# |
# |
$r->print('<h2>'. |
$r->print('<p class="LC_info">'. |
&mt('Generating CSV report of student responses').'</h2>'); |
&mt('Generating CSV report of student responses').'</p>'); |
# |
# |
# Progress window |
# Progress window |
my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin |
my %prog_state=&Apache::lonhtmlcommon::Create_PrgWin($r,scalar(@$students)); |
($r,'CSV File Compilation Status', |
|
'CSV File Compilation Progress', |
|
scalar(@$students),'inline',undef,'Statistics','stats_status'); |
|
|
|
$r->rflush(); |
$r->rflush(); |
# |
# |
Line 1036 sub prepare_csv_output {
|
Line 1103 sub prepare_csv_output {
|
time.'_'.rand(1000000000).'.csv'; |
time.'_'.rand(1000000000).'.csv'; |
unless ($outputfile = Apache::File->new('>/home/httpd'.$filename)) { |
unless ($outputfile = Apache::File->new('>/home/httpd'.$filename)) { |
$r->log_error("Couldn't open $filename for output $!"); |
$r->log_error("Couldn't open $filename for output $!"); |
$r->print('<div class="LC_error">' |
$r->print( |
.&mt('Problems occurred in writing the CSV file. ' |
'<p class="LC_error">' |
.'This error has been logged. ' |
.&mt('Problems occurred in writing the CSV file.') |
.'Please alert your LON-CAPA administrator.') |
.' '.&mt('This error has been logged.') |
.'</div>'); |
.' '.&mt('Please alert your LON-CAPA administrator.') |
|
.'</p>' |
|
); |
$outputfile = undef; |
$outputfile = undef; |
} |
} |
# |
# |
Line 1048 sub prepare_csv_output {
|
Line 1117 sub prepare_csv_output {
|
my @extra_resp_headers = &get_extra_response_headers($show_named); |
my @extra_resp_headers = &get_extra_response_headers($show_named); |
# |
# |
# Create the table header |
# Create the table header |
my @student_columns; |
my @student_columns = &get_student_columns($show_named); |
if ($show_named) { |
my $student_column_count = scalar(@student_columns); |
@student_columns = qw(username domain id section); |
|
} else { |
|
@student_columns = qw(username); |
|
} |
|
# |
# |
my %headers; |
my %headers; |
push(@{$headers{'student'}},@student_columns); |
push(@{$headers{'student'}},@student_columns); |
Line 1065 sub prepare_csv_output {
|
Line 1130 sub prepare_csv_output {
|
# we put the headers into the %headers hash |
# we put the headers into the %headers hash |
my %problem_analysis; |
my %problem_analysis; |
my %start_col; |
my %start_col; |
my $max_column = scalar(@student_columns); |
my $max_column = $student_column_count; |
foreach my $prob (@$problems) { |
foreach my $prob (@$problems) { |
|
my $symb = $prob->symb(); |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
my %analysis = &Apache::lonstathelpers::get_problem_data($prob->src); |
$problem_analysis{$prob->src}=\%analysis; |
$problem_analysis{$prob->src}=\%analysis; |
my $title = $prob->compTitle; |
$headers{'problem'}->[$max_column] = $prob->compTitle; |
unless (&Apache::loncommon::needs_gci_custom()) { |
|
($title) = ($prob->src =~ m{/([^/]+)$}); |
|
} |
|
$headers{'problem'}->[$max_column] = $title; |
|
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
$headers{'part'}->[$max_column] = $prob->part_display($partid); |
$headers{'part'}->[$max_column] = $prob->part_display($partid); |
my $responses = [$prob->responseIds($partid)]; |
my $responses = [$prob->responseIds($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
my $resptypes = [$prob->responseType($partid)]; |
Line 1106 sub prepare_csv_output {
|
Line 1173 sub prepare_csv_output {
|
# Main loop |
# Main loop |
foreach my $student (@$students) { |
foreach my $student (@$students) { |
last if ($c->aborted()); |
last if ($c->aborted()); |
my $anonid = &Crypt::PasswdMD5::unix_md5_crypt($student->{'username'}, |
|
$salt); |
|
$anonid = substr($anonid,length($salt)+1); |
|
my @rows; |
my @rows; |
foreach my $prob (@$problems) { |
foreach my $prob (@$problems) { |
|
my $symb = $prob->symb; |
foreach my $partid (@{$prob->parts}) { |
foreach my $partid (@{$prob->parts}) { |
|
if (($prob->is_anonsurvey($partid)) || ($anoncounter->{$symb."\0".$partid})) { |
|
next if ($show_named); |
|
} else { |
|
next unless ($show_named); |
|
} |
my @responses = $prob->responseIds($partid); |
my @responses = $prob->responseIds($partid); |
my @response_type = $prob->responseType($partid); |
my @response_type = $prob->responseType($partid); |
for (my $i=0;$i<=$#responses;$i++) { |
for (my $i=0;$i<=$#responses;$i++) { |
Line 1151 sub prepare_csv_output {
|
Line 1221 sub prepare_csv_output {
|
foreach my $row (@rows) { |
foreach my $row (@rows) { |
my $student_row_data = ''; |
my $student_row_data = ''; |
if ($show_named) { |
if ($show_named) { |
$student_row_data = '"'.join('","', |
foreach my $field (@student_columns) { |
map { $student->{$_}; } |
my $value = $student->{$field}; |
@student_columns).'"'; |
if ($field eq 'comments') { |
|
$value = &Apache::lonmsgdisplay::retrieve_instructor_comments |
|
($student->{'username'},$student->{'domain'}); |
|
} |
|
$student_row_data .= '"'.&Apache::loncommon::csv_translate($value).'",'; |
|
} |
|
$student_row_data =~ s/,$//; |
} else { |
} else { |
$student_row_data = '"'.$anonid.'"'; |
$student_row_data = '"'.&mt('Anonymized').'"'; |
} |
} |
print $outputfile $student_row_data; |
print $outputfile $student_row_data; |
for (my $i=scalar(@student_columns);$i<$max_column;$i++) { |
for (my $i=$student_column_count;$i<$max_column;$i++) { |
my $value = &Apache::loncommon::csv_translate($row->[$i]); |
my $value = &Apache::loncommon::csv_translate($row->[$i]); |
$value ||=''; |
$value ||=''; |
print $outputfile ',"'.$value.'"'; |
print $outputfile ',"'.$value.'"'; |
Line 1166 sub prepare_csv_output {
|
Line 1242 sub prepare_csv_output {
|
print $outputfile $/; |
print $outputfile $/; |
} |
} |
undef(@rows); |
undef(@rows); |
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state, |
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state,'last student'); |
'last student'); |
|
} |
} |
close($outputfile); |
close($outputfile); |
# |
# |
Line 1184 sub prepare_csv_output {
|
Line 1259 sub prepare_csv_output {
|
sub csv_format_item { |
sub csv_format_item { |
my ($item,$type) = @_; |
my ($item,$type) = @_; |
if ($type eq 'Time') { |
if ($type eq 'Time') { |
$item = localtime($item); |
$item = &Apache::lonlocal::locallocaltime($item); |
} |
} |
$item =&Apache::loncommon::csv_translate($item); |
$item =&Apache::loncommon::csv_translate($item); |
return $item; |
return $item; |
Line 1204 sub CreateInterface {
|
Line 1279 sub CreateInterface {
|
foreach ('HTML','Excel','CSV') { |
foreach ('HTML','Excel','CSV') { |
$output_selector .= ' <option value="'.lc($_).'"'; |
$output_selector .= ' <option value="'.lc($_).'"'; |
if ($env{'form.output'} eq lc($_)) { |
if ($env{'form.output'} eq lc($_)) { |
$output_selector .= ' selected="selected" '; |
$output_selector .= ' selected="selected"'; |
} |
} |
$output_selector .='>'.&mt($_).'</option>'.$/; |
$output_selector .='>'.&mt($_).'</option>'.$/; |
} |
} |
Line 1212 sub CreateInterface {
|
Line 1287 sub CreateInterface {
|
## |
## |
## Environment variable initialization |
## Environment variable initialization |
my $Str = ''; |
my $Str = ''; |
$Str .= &Apache::lonhtmlcommon::breadcrumbs('Student Submission Reports'); |
|
$Str .= '<br />'; |
$Str .= '<br />'; |
$Str .= &Apache::loncommon::start_data_table(); |
$Str .= &Apache::loncommon::start_data_table(); |
$Str .= &Apache::loncommon::start_data_table_header_row(); |
$Str .= &Apache::loncommon::start_data_table_header_row(); |
$Str .= '<th>'.&mt('Sections').'</th>'; |
$Str .= '<th>'.&mt('Sections').'</th>'; |
$Str .= '<th>'.&mt('Groups').'</th>'; |
$Str .= '<th>'.&mt('Groups').'</th>'; |
|
$Str .= '<th>'.&mt('Student Data').&Apache::loncommon::help_open_topic("Chart_Student_Data").'</th>'; |
$Str .= '<th>'.&mt('Access Status').'</th>'; |
$Str .= '<th>'.&mt('Access Status').'</th>'; |
$Str .= '<th>'.&mt('Options').'</th>'; |
$Str .= '<th>'.&mt('Options').'</th>'; |
$Str .= '<th>'.&mt('Output Format').'</th>'; |
$Str .= '<th>'.&mt('Output Format').'</th>'; |
Line 1232 sub CreateInterface {
|
Line 1307 sub CreateInterface {
|
$Str .= &Apache::lonstatistics::GroupSelect('Group','multiple',5); |
$Str .= &Apache::lonstatistics::GroupSelect('Group','multiple',5); |
$Str .= '</td>'; |
$Str .= '</td>'; |
# |
# |
|
$Str .= '<td align="center">'."\n"; |
|
$Str .= &Apache::lonstatistics::StudentDataSelect('StudentData','multiple', 5,undef); |
|
$Str .= '</td>'; |
|
# |
$Str .= '<td align="center">'; |
$Str .= '<td align="center">'; |
$Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5); |
$Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5); |
$Str .= '</td>'; |
$Str .= '</td>'; |
Line 1265 sub CreateInterface {
|
Line 1344 sub CreateInterface {
|
$prob_status_checkbox .= 'checked="checked" '; |
$prob_status_checkbox .= 'checked="checked" '; |
} |
} |
$prob_status_checkbox .= 'value="true" />'; |
$prob_status_checkbox .= 'value="true" />'; |
my $anon_checkbox; |
# |
unless (&Apache::loncommon::needs_gci_custom()) { |
$Str .= |
# |
'<td valign="top">' |
# anonymized checkbox |
.'<label>' |
# |
.$prob_checkbox.&mt('Show problem') |
$anon_checkbox = '<input type="checkbox" name="anonymized" '; |
.'</label><br />' |
if (exists($env{'form.anonymized'}) && |
.'<label>' |
$env{'form.anonymized'} eq 'true') { |
.' '.$ans_checkbox.&mt('Show correct answers') |
$anon_checkbox .= 'checked="checked" '; |
.'</label><br />' |
} |
.'<label>' |
$anon_checkbox .= ' value="true" />'; |
.$all_sub_checkbox.&mt('Show all submissions') |
} |
.'</label><br />' |
|
.'<label>' |
$Str .= '<td align="right" valign="top">'. |
.$prob_status_checkbox.&mt('Show problem grading') |
'<label><b>'. |
.'</label>' |
&mt('Show problem').' '.$prob_checkbox.'</b></label><br />'. |
.'</td>'; |
'<label><b>'. |
|
&mt('Show correct answers').' '.$ans_checkbox.'</b></label><br />'. |
|
'<label><b>'. |
|
&mt('Show all submissions').' '.$all_sub_checkbox. |
|
'</b></label><br />'. |
|
'<label><b>'. |
|
&mt('Show problem grading').' '.$prob_status_checkbox. |
|
'</b></label><br />'; |
|
|
|
unless (&Apache::loncommon::needs_gci_custom()) { |
|
$Str .= '<label><b>'. |
|
&mt('Anonymized').' '.$anon_checkbox. |
|
'</b></label><br />'; |
|
} |
|
$Str .= '</td>'; |
|
# |
# |
$Str .= '<td align="center" valign="top">'.$output_selector.'</td>'; |
$Str .= '<td align="center" valign="top">'.$output_selector.'</td>'; |
# |
# |
$Str .= &Apache::loncommon::end_data_table_row(); |
$Str .= &Apache::loncommon::end_data_table_row(); |
$Str .= &Apache::loncommon::end_data_table(); |
$Str .= &Apache::loncommon::end_data_table(); |
# |
# |
$Str .= '<p><span class="LC_nobreak">' |
|
.&mt('Status: [_1]', |
|
'<input type="text" name="stats_status"' |
|
.' size="60" value="" readonly="readonly" />') |
|
.'</span></p>'; |
|
## |
|
return $Str; |
return $Str; |
} |
} |
|
|
|
sub get_student_columns { |
|
my ($show_named) = @_; |
|
my @student_columns; |
|
if ($show_named) { |
|
@student_columns = @Apache::lonstatistics::SelectedStudentData; |
|
if (grep(/^all$/,@student_columns)) { |
|
@student_columns = qw(fullname username domain id section status groups comments); |
|
} |
|
} else { |
|
@student_columns = ('username'); |
|
} |
|
return @student_columns; |
|
} |
|
|
1; |
1; |
|
|
__END__ |
__END__ |