version 1.32, 2005/02/02 23:05:26
|
version 1.40, 2005/04/07 06:56:24
|
Line 27
|
Line 27
|
package Apache::lonstudentsubmissions; |
package Apache::lonstudentsubmissions; |
|
|
use strict; |
use strict; |
use Apache::lonnet(); |
use Apache::lonnet; |
use Apache::loncommon(); |
use Apache::loncommon(); |
use Apache::lonhtmlcommon(); |
use Apache::lonhtmlcommon(); |
use Apache::loncoursedata(); |
use Apache::loncoursedata(); |
Line 71 sub BuildStudentSubmissionsPage {
|
Line 71 sub BuildStudentSubmissionsPage {
|
'<h3>'.&mt('Loading student data').'</h3>'); |
'<h3>'.&mt('Loading student data').'</h3>'); |
$r->rflush(); |
$r->rflush(); |
# |
# |
if (exists($ENV{'form.problemchoice'}) && |
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') { |
$r->print("<br />\n"); |
$r->print("<br />\n"); |
Line 110 sub BuildStudentSubmissionsPage {
|
Line 110 sub BuildStudentSubmissionsPage {
|
push(@Problems,$resource); |
push(@Problems,$resource); |
} |
} |
# |
# |
|
$r->print('<h4>'. |
|
&Apache::lonstatistics::section_and_enrollment_description(). |
|
'</h4>'); |
if (! scalar(@Problems) || ! defined($Problems[0])) { |
if (! scalar(@Problems) || ! defined($Problems[0])) { |
$r->print('resource is undefined'); |
$r->print('resource is undefined'); |
} else { |
} else { |
Line 117 sub BuildStudentSubmissionsPage {
|
Line 120 sub BuildStudentSubmissionsPage {
|
my $resource = $Problems[0]; |
my $resource = $Problems[0]; |
$r->print('<h1>'.$resource->title.'</h1>'); |
$r->print('<h1>'.$resource->title.'</h1>'); |
$r->print('<h3>'.$resource->src.'</h3>'); |
$r->print('<h3>'.$resource->src.'</h3>'); |
if ($ENV{'form.renderprob'} eq 'true') { |
if ($env{'form.renderprob'} eq 'true') { |
$r->print( |
$r->print(&Apache::lonstathelpers::render_resource($resource)); |
&Apache::lonstathelpers::render_resource({src => $resource->src}) |
|
); |
|
$r->rflush(); |
$r->rflush(); |
} |
} |
} |
} |
if ($ENV{'form.output'} eq 'excel') { |
if ($env{'form.output'} eq 'excel') { |
&prepare_excel_output($r,\@Problems,\@Students); |
&prepare_excel_output($r,\@Problems,\@Students); |
} elsif ($ENV{'form.output'} eq 'csv') { |
} elsif ($env{'form.output'} eq 'csv') { |
&prepare_csv_output($r,\@Problems,\@Students); |
&prepare_csv_output($r,\@Problems,\@Students); |
} else { |
} else { |
&prepare_html_output($r,\@Problems,\@Students); |
&prepare_html_output($r,\@Problems,\@Students); |
Line 153 sub BuildStudentSubmissionsPage {
|
Line 154 sub BuildStudentSubmissionsPage {
|
## |
## |
sub get_extra_response_headers { |
sub get_extra_response_headers { |
my @extra_resp_headers; |
my @extra_resp_headers; |
if ($ENV{'form.correctans'} eq 'true') { |
if ($env{'form.correctans'} eq 'true') { |
push(@extra_resp_headers,'Correct'); |
push(@extra_resp_headers,'Correct'); |
} |
} |
if ($ENV{'form.prob_status'} eq 'true') { |
if ($env{'form.prob_status'} eq 'true') { |
push(@extra_resp_headers,'Award Detail'); |
push(@extra_resp_headers,'Award Detail'); |
push(@extra_resp_headers,'Time'); |
push(@extra_resp_headers,'Time'); |
push(@extra_resp_headers,'Attempt'); |
push(@extra_resp_headers,'Attempt'); |
Line 217 sub prepare_html_output {
|
Line 218 sub prepare_html_output {
|
my @extra_resp_headers = &get_extra_response_headers(); |
my @extra_resp_headers = &get_extra_response_headers(); |
# |
# |
# Create the table header |
# Create the table header |
my @student_columns = ('username','domain','id'); |
my @student_columns = ('username','domain','id','section'); |
# |
# |
my %headers; |
my %headers; |
my $student_column_count = scalar(@student_columns); |
my $student_column_count = scalar(@student_columns); |
Line 267 sub prepare_html_output {
|
Line 268 sub prepare_html_output {
|
if ($partid =~/^\d+$/) { |
if ($partid =~/^\d+$/) { |
$tmpname = $prob->part_display($partid); |
$tmpname = $prob->part_display($partid); |
} |
} |
|
if ($tmpname !~ /^part/) { |
|
$tmpname = 'Part '.$tmpname; |
|
} |
$headers{'part'} .= qq{<th colspan="$part_span">$tmpname</th>}; |
$headers{'part'} .= qq{<th colspan="$part_span">$tmpname</th>}; |
$nonempty_part_headers = 1; |
$nonempty_part_headers = 1; |
} else { |
} else { |
Line 346 sub prepare_html_output {
|
Line 350 sub prepare_html_output {
|
$maxrow = scalar(@$results); |
$maxrow = scalar(@$results); |
} |
} |
for (my $j=scalar(@$results)-1;$j>=0;$j--) { |
for (my $j=scalar(@$results)-1;$j>=0;$j--) { |
if ($ENV{'form.all_sub'} ne 'true') { |
if ($env{'form.all_sub'} ne 'true') { |
next if ($j ne scalar(@$results)-1); |
next if ($j ne scalar(@$results)-1); |
} |
} |
my $response = &hashify_response($results->[$j], |
my $response = &hashify_response($results->[$j], |
Line 426 sub prepare_html_output {
|
Line 430 sub prepare_html_output {
|
sub hashify_response { |
sub hashify_response { |
my ($response,$prob,$student,$partid,$respid) =@_; |
my ($response,$prob,$student,$partid,$respid) =@_; |
my $resp_hash = {}; |
my $resp_hash = {}; |
if ($ENV{'form.correctans'} eq 'true') { |
if ($env{'form.correctans'} eq 'true') { |
$resp_hash->{'Correct'} = |
$resp_hash->{'Correct'} = |
&Apache::lonstathelpers::get_student_answer |
&Apache::lonstathelpers::get_student_answer |
($prob,$student->{'username'},$student->{'domain'}, |
($prob,$student->{'username'},$student->{'domain'}, |
Line 550 sub html_non_essay_results {
|
Line 554 sub html_non_essay_results {
|
$option = $submission{$header}; |
$option = $submission{$header}; |
} |
} |
push(@values,&HTML::Entities::encode($option)); |
push(@values,&HTML::Entities::encode($option)); |
|
} elsif ($original_header eq 'Time') { |
|
push(@values,&Apache::lonlocal::locallocaltime($response->{$original_header})); |
} else { |
} else { |
# A normal column |
# A normal column |
push(@values,$response->{$original_header}); |
push(@values,$response->{$original_header}); |
Line 622 sub prepare_excel_output {
|
Line 628 sub prepare_excel_output {
|
$r->rflush(); |
$r->rflush(); |
# |
# |
# Create the excel spreadsheet |
# Create the excel spreadsheet |
my $filename = '/prtspool/'. |
my ($workbook,$filename,$format) = |
$ENV{'user.name'}.'_'.$ENV{'user.domain'}.'_'. |
&Apache::loncommon::create_workbook($r); |
time.'_'.rand(1000000000).'.xls'; |
return if (! defined($workbook)); |
my $workbook = Spreadsheet::WriteExcel->new('/home/httpd'.$filename); |
|
if (! defined($workbook)) { |
|
$r->log_error("Error creating excel spreadsheet $filename: $!"); |
|
$r->print('<p>'.&mt("Unable to create new Excel file. ". |
|
"This error has been logged. ". |
|
"Please alert your LON-CAPA administrator"). |
|
'</p>'); |
|
return undef; |
|
} |
|
# |
|
$workbook->set_tempdir('/home/httpd/perl/tmp'); |
|
# |
|
my $format = &Apache::loncommon::define_excel_formats($workbook); |
|
my $worksheet = $workbook->addworksheet('Student Submission Data'); |
my $worksheet = $workbook->addworksheet('Student Submission Data'); |
# |
# |
# Add headers to the worksheet |
# Add headers to the worksheet |
my $rows_output = 0; |
my $rows_output = 0; |
$worksheet->write($rows_output++,0, |
$worksheet->write($rows_output++,0, |
$ENV{'course.'.$ENV{'request.course.id'}.'.description'}, |
$env{'course.'.$env{'request.course.id'}.'.description'}, |
$format->{'h1'}); |
$format->{'h1'}); |
$rows_output++; |
$rows_output++; |
my $cols_output = 0; |
my $cols_output = 0; |
Line 655 sub prepare_excel_output {
|
Line 648 sub prepare_excel_output {
|
$worksheet->write($partid_row,0,'Part ID',$format->{'bold'}); |
$worksheet->write($partid_row,0,'Part ID',$format->{'bold'}); |
$worksheet->write($respid_row,0,'Response ID',$format->{'bold'}); |
$worksheet->write($respid_row,0,'Response ID',$format->{'bold'}); |
# Student headers |
# Student headers |
my @StudentColumns = ('username','domain','id'); |
my @StudentColumns = ('username','domain','id','section'); |
foreach (@StudentColumns) { |
foreach (@StudentColumns) { |
$worksheet->write($header_row,$cols_output++,ucfirst($_), |
$worksheet->write($header_row,$cols_output++,ucfirst($_), |
$format->{'bold'}); |
$format->{'bold'}); |
Line 732 sub prepare_excel_output {
|
Line 725 sub prepare_excel_output {
|
my $response_start_col = $start_col{$prob->symb}->{$partid}->{$respid}; |
my $response_start_col = $start_col{$prob->symb}->{$partid}->{$respid}; |
for (my $j=scalar(@$results)-1;$j>=0;$j--) { |
for (my $j=scalar(@$results)-1;$j>=0;$j--) { |
$cols_output = $response_start_col; |
$cols_output = $response_start_col; |
if ($ENV{'form.all_sub'} ne 'true') { |
if ($env{'form.all_sub'} ne 'true') { |
next if ($j ne scalar(@$results)-1); |
next if ($j ne scalar(@$results)-1); |
} |
} |
my $response = &hashify_response($results->[$j], |
my $response = &hashify_response($results->[$j], |
Line 741 sub prepare_excel_output {
|
Line 734 sub prepare_excel_output {
|
$partid, |
$partid, |
$respid); |
$respid); |
my @response_data = |
my @response_data = |
&excel_response_data(\@headers,$prob,$partid, |
&compile_response_data(\@headers,$response, |
$respid,$response,$resptype); |
$prob,$partid,$respid, |
|
$resptype, |
|
\&excel_format_item); |
$worksheet->write_row($rows_output++,$cols_output, |
$worksheet->write_row($rows_output++,$cols_output, |
\@response_data); |
\@response_data); |
$cols_output+=scalar(@response_data); |
$cols_output+=scalar(@response_data); |
Line 754 sub prepare_excel_output {
|
Line 749 sub prepare_excel_output {
|
} |
} |
} |
} |
# Fill in the remaining rows with the students data |
# Fill in the remaining rows with the students data |
for (my $row = $student_row+1;$row<=$max_row;$row++) { |
for (my $row = $student_row+1;$row<$max_row;$row++) { |
my $cols = 0; |
my $cols = 0; |
foreach my $field (@StudentColumns) { |
foreach my $field (@StudentColumns) { |
$worksheet->write($row,$cols++, |
$worksheet->write($row,$cols++, |
$student->{$field}); |
$student->{$field}); |
} |
} |
} |
} |
$rows_output++; |
|
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state, |
&Apache::lonhtmlcommon::Increment_PrgWin($r,\%prog_state, |
'last student'); |
'last student'); |
} |
} |
Line 782 sub prepare_excel_output {
|
Line 776 sub prepare_excel_output {
|
return; |
return; |
} |
} |
|
|
sub excel_response_data { |
sub compile_response_data { |
my ($headers,$prob,$partid,$respid,$response,$resptype) = @_; |
my ($headers,$response,$prob,$partid,$respid,$resptype,$format) = @_; |
if (! ref($headers) || ref($headers) ne 'ARRAY' || ! scalar(@$headers)) { |
if (! ref($headers) || ref($headers) ne 'ARRAY' || ! scalar(@$headers)) { |
return (); |
return (); |
} |
} |
|
if (ref($format) ne 'CODE') { |
|
$format = sub { return $_[0]; }; |
|
} |
# |
# |
my $submission = &HTML::Entities::decode($response->{'Submission'}); |
my $submission = |
|
&HTML::Entities::decode |
|
(&Apache::lonnet::unescape($response->{'Submission'})); |
return () if (! defined($submission) || $submission eq ''); |
return () if (! defined($submission) || $submission eq ''); |
$submission =~ s/\\\"/\"/g; |
$submission =~ s/\\\"/\"/g; |
$submission =~ s/\\\'/\'/g; |
$submission =~ s/\\\'/\'/g; |
Line 825 sub excel_response_data {
|
Line 824 sub excel_response_data {
|
} elsif (exists($submission{$header})) { |
} elsif (exists($submission{$header})) { |
$option = $submission{$header}; |
$option = $submission{$header}; |
} |
} |
push(@values,&excel_format_item($option,$header)); |
push(@values,&{$format}($option,$header)); |
} else { |
} else { |
# A normal column |
# A normal column |
push(@values, |
push(@values,&{$format}($response->{$original_header}, |
&excel_format_item($response->{$original_header}, |
|
$original_header)); |
$original_header)); |
} |
} |
} |
} |
} else { |
} else { |
@values = map { &excel_format_item($response->{$_},$_); } @$headers; |
@values = map { &{$format}($response->{$_},$_); } @$headers; |
} |
} |
return @values; |
return @values; |
} |
} |
Line 842 sub excel_response_data {
|
Line 840 sub excel_response_data {
|
sub excel_format_item { |
sub excel_format_item { |
my ($item,$type) = @_; |
my ($item,$type) = @_; |
if ($type eq 'Time') { |
if ($type eq 'Time') { |
&Apache::lonstathelpers::calc_serial($item); |
$item = &Apache::lonstathelpers::calc_serial($item); |
} else { |
} else { |
if ($item =~ m/^=/) { |
if ($item =~ m/^=/) { |
$item = ' '.$item; |
$item = ' '.$item; |
Line 880 sub prepare_csv_output {
|
Line 878 sub prepare_csv_output {
|
# Open a file |
# Open a file |
my $outputfile; |
my $outputfile; |
my $filename = '/prtspool/'. |
my $filename = '/prtspool/'. |
$ENV{'user.name'}.'_'.$ENV{'user.domain'}.'_'. |
$env{'user.name'}.'_'.$env{'user.domain'}.'_'. |
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 $!"); |
Line 894 sub prepare_csv_output {
|
Line 892 sub prepare_csv_output {
|
my @extra_resp_headers = &get_extra_response_headers(); |
my @extra_resp_headers = &get_extra_response_headers(); |
# |
# |
# Create the table header |
# Create the table header |
my @student_columns = ('username','domain','id'); |
my @student_columns = ('username','domain','id','section'); |
# |
# |
my %headers; |
my %headers; |
push(@{$headers{'student'}},@student_columns); |
push(@{$headers{'student'}},@student_columns); |
Line 941 sub prepare_csv_output {
|
Line 939 sub prepare_csv_output {
|
} |
} |
# |
# |
# Main loop |
# Main loop |
my $mycount = 10; |
|
foreach my $student (@$students) { |
foreach my $student (@$students) { |
last if ($mycount-- <0); |
|
last if ($c->aborted()); |
last if ($c->aborted()); |
my @rows; |
my @rows; |
foreach my $prob (@$problems) { |
foreach my $prob (@$problems) { |
Line 964 sub prepare_csv_output {
|
Line 960 sub prepare_csv_output {
|
$results = []; |
$results = []; |
} |
} |
for (my $j=0; $j<scalar(@$results);$j++) { |
for (my $j=0; $j<scalar(@$results);$j++) { |
if ($ENV{'form.all_sub'} ne 'true') { |
if ($env{'form.all_sub'} ne 'true') { |
next if ($j != 0); |
next if ($j != 0); |
} |
} |
my $idx = scalar(@$results) - $j - 1; |
my $idx = scalar(@$results) - $j - 1; |
Line 973 sub prepare_csv_output {
|
Line 969 sub prepare_csv_output {
|
$partid,$respid); |
$partid,$respid); |
my @data = &compile_response_data(\@headers,$response, |
my @data = &compile_response_data(\@headers,$response, |
$prob,$partid, |
$prob,$partid, |
$respid,$resptype); |
$respid,$resptype, |
|
\&csv_format_item); |
my $resp_start_idx = |
my $resp_start_idx = |
$start_col{$prob->symb}->{$partid}->{$respid}; |
$start_col{$prob->symb}->{$partid}->{$respid}; |
for (my $k=0;$k<=$#data;$k++) { |
for (my $k=0;$k<=$#data;$k++) { |
Line 1010 sub prepare_csv_output {
|
Line 1007 sub prepare_csv_output {
|
return; |
return; |
} |
} |
|
|
sub compile_response_data { |
|
my ($headers,$response,$prob,$partid,$respid,$resptype) = @_; |
|
if (! ref($headers) || ref($headers) ne 'ARRAY' || ! scalar(@$headers)) { |
|
return (); |
|
} |
|
# |
|
my $submission = |
|
&HTML::Entities::decode |
|
(&Apache::lonnet::unescape($response->{'Submission'})); |
|
return () if (! defined($submission) || $submission eq ''); |
|
$response->{'Submission'} = $submission; |
|
my @values; |
|
if ($resptype =~ /^(option|match|rank)$/) { |
|
my %submission = |
|
map { |
|
my ($foil,$value) = split('=',&Apache::lonnet::unescape($_)); |
|
($foil,$value); |
|
} split('&',$response->{'Submission'}); |
|
my %correct; |
|
if (exists($response->{'Correct'})) { |
|
%correct = |
|
map { |
|
my ($foil,$value)=split('=',&Apache::lonnet::unescape($_)); |
|
($foil,$value); |
|
} split('&',$response->{'Correct'}); |
|
} |
|
# |
|
foreach my $original_header (@$headers) { |
|
if ($original_header =~ /^_/) { |
|
# '_' denotes a foil column |
|
my ($header) = ($original_header =~ m/^_(.*)$/); |
|
my $option = ''; |
|
if ( my ($foil) = ($header =~ /(.*) Correct$/)) { |
|
if (exists($correct{$foil})) { |
|
$option = $correct{$foil}; |
|
} |
|
} elsif (exists($submission{$header})) { |
|
$option = $submission{$header}; |
|
} |
|
push(@values,&csv_format_item($option,$header)); |
|
} else { |
|
# A normal column |
|
push(@values, |
|
&csv_format_item($response->{$original_header}, |
|
$original_header)); |
|
} |
|
} |
|
} else { |
|
@values = map { &csv_format_item($response->{$_},$_); } @$headers; |
|
} |
|
return @values; |
|
} |
|
|
|
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 = localtime($item); |
} |
} |
$item =&Apache::loncommon::csv_translate($item); |
$item =&Apache::loncommon::csv_translate($item); |
return $item; |
return $item; |
} |
} |
Line 1085 sub CreateInterface {
|
Line 1029 sub CreateInterface {
|
my $output_selector = $/.'<select name="output">'.$/; |
my $output_selector = $/.'<select name="output">'.$/; |
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 '; |
$output_selector .= ' selected '; |
} |
} |
$output_selector .='>'.&mt($_).'</option>'.$/; |
$output_selector .='>'.&mt($_).'</option>'.$/; |
Line 1114 sub CreateInterface {
|
Line 1058 sub CreateInterface {
|
# |
# |
# Render problem checkbox |
# Render problem checkbox |
my $prob_checkbox = '<input type="checkbox" name="renderprob" '; |
my $prob_checkbox = '<input type="checkbox" name="renderprob" '; |
if (exists($ENV{'form.renderprob'}) && $ENV{'form.renderprob'} eq 'true') { |
if (exists($env{'form.renderprob'}) && $env{'form.renderprob'} eq 'true') { |
$prob_checkbox .= 'checked '; |
$prob_checkbox .= 'checked '; |
} |
} |
$prob_checkbox .= 'value="true" />'; |
$prob_checkbox .= 'value="true" />'; |
# |
# |
# Compute correct answers checkbox |
# Compute correct answers checkbox |
my $ans_checkbox = '<input type="checkbox" name="correctans" '; |
my $ans_checkbox = '<input type="checkbox" name="correctans" '; |
if (exists($ENV{'form.correctans'}) && $ENV{'form.correctans'} eq 'true') { |
if (exists($env{'form.correctans'}) && $env{'form.correctans'} eq 'true') { |
$ans_checkbox .= 'checked '; |
$ans_checkbox .= 'checked '; |
} |
} |
$ans_checkbox .= 'value="true" />'; |
$ans_checkbox .= 'value="true" />'; |
# |
# |
# Show all submissions checkbox |
# Show all submissions checkbox |
my $all_sub_checkbox = '<input type="checkbox" name="all_sub" '; |
my $all_sub_checkbox = '<input type="checkbox" name="all_sub" '; |
if (exists($ENV{'form.all_sub'}) && |
if (exists($env{'form.all_sub'}) && |
$ENV{'form.all_sub'} eq 'true') { |
$env{'form.all_sub'} eq 'true') { |
$all_sub_checkbox .= 'checked '; |
$all_sub_checkbox .= 'checked '; |
} |
} |
$all_sub_checkbox.= 'value="true" />'; |
$all_sub_checkbox.= 'value="true" />'; |
# |
# |
# problem status checkbox |
# problem status checkbox |
my $prob_status_checkbox = '<input type="checkbox" name="prob_status" '; |
my $prob_status_checkbox = '<input type="checkbox" name="prob_status" '; |
if (exists($ENV{'form.prob_status'}) && |
if (exists($env{'form.prob_status'}) && |
$ENV{'form.prob_status'} eq 'true') { |
$env{'form.prob_status'} eq 'true') { |
$prob_status_checkbox .= 'checked '; |
$prob_status_checkbox .= 'checked '; |
} |
} |
$prob_status_checkbox .= 'value="true" />'; |
$prob_status_checkbox .= 'value="true" />'; |
# |
# |
$Str .= '<td align="right" halign="top">'. |
$Str .= '<td align="right" valign="top">'. |
'<label><b>'. |
'<label><b>'. |
&mt('Show problem [_1]',$prob_checkbox).'</b></label><br />'. |
&mt('Show problem [_1]',$prob_checkbox).'</b></label><br />'. |
'<label><b>'. |
'<label><b>'. |