version 1.103, 2005/03/27 05:08:03
|
version 1.113, 2008/09/11 14:47:22
|
Line 50 Excel files, and plots.
|
Line 50 Excel files, and plots.
|
package Apache::lonproblemstatistics; |
package Apache::lonproblemstatistics; |
|
|
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 60 use Apache::lonlocal;
|
Line 60 use Apache::lonlocal;
|
use Spreadsheet::WriteExcel; |
use Spreadsheet::WriteExcel; |
use Apache::lonstathelpers(); |
use Apache::lonstathelpers(); |
use Time::HiRes; |
use Time::HiRes; |
|
use LONCAPA; |
|
|
|
|
my @StatsArray; |
my @StatsArray; |
my %SeqStat; # keys are symbs, values are hash refs |
my %SeqStat; # keys are symbs, values are hash refs |
Line 516 my %SelectedFields;
|
Line 518 my %SelectedFields;
|
sub parse_field_selection { |
sub parse_field_selection { |
# |
# |
# Pull out the defaults |
# Pull out the defaults |
if (! defined($ENV{'form.fieldselections'})) { |
if (! defined($env{'form.fieldselections'})) { |
$ENV{'form.fieldselections'} = []; |
$env{'form.fieldselections'} = []; |
foreach my $field (@Fields) { |
foreach my $field (@Fields) { |
next if ($field->{'selectable'} ne 'yes'); |
next if ($field->{'selectable'} ne 'yes'); |
if ($field->{'defaultselected'} eq 'yes') { |
if ($field->{'defaultselected'} eq 'yes') { |
push(@{$ENV{'form.fieldselections'}},$field->{'name'}); |
push(@{$env{'form.fieldselections'}},$field->{'name'}); |
} |
} |
} |
} |
} |
} |
# |
# |
# Make sure the data we are plotting is there |
# Make sure the data we are plotting is there |
my %NeededFields; |
my %NeededFields; |
if (exists($ENV{'form.plot'}) && $ENV{'form.plot'} ne '' && |
if (exists($env{'form.plot'}) && $env{'form.plot'} ne '' && |
$ENV{'form.plot'} ne 'none') { |
$env{'form.plot'} ne 'none') { |
if ($ENV{'form.plot'} eq 'degrees') { |
if ($env{'form.plot'} eq 'degrees') { |
$NeededFields{'deg_of_diff'}++; |
$NeededFields{'deg_of_diff'}++; |
$NeededFields{'deg_of_disc'}++; |
$NeededFields{'deg_of_disc'}++; |
} elsif ($ENV{'form.plot'} eq 'tries statistics') { |
} elsif ($env{'form.plot'} eq 'tries statistics') { |
$NeededFields{'mean_tries'}++; |
$NeededFields{'mean_tries'}++; |
$NeededFields{'std_tries'}++; |
$NeededFields{'std_tries'}++; |
$NeededFields{'problem_num'}++; |
$NeededFields{'problem_num'}++; |
} else { |
} else { |
$NeededFields{$ENV{'form.plot'}}++; |
$NeededFields{$env{'form.plot'}}++; |
} |
} |
} |
} |
# |
# |
# This should not happen, but in case it does... |
# This should not happen, but in case it does... |
if (ref($ENV{'form.fieldselections'}) ne 'ARRAY') { |
if (ref($env{'form.fieldselections'}) ne 'ARRAY') { |
$ENV{'form.fieldselections'} = [$ENV{'form.fieldselections'}]; |
$env{'form.fieldselections'} = [$env{'form.fieldselections'}]; |
} |
} |
# |
# |
# Set the field data and the selected fields (for easier checking) |
# Set the field data and the selected fields (for easier checking) |
Line 559 sub parse_field_selection {
|
Line 561 sub parse_field_selection {
|
$field->{'selected'} = 'yes'; |
$field->{'selected'} = 'yes'; |
$SelectedFields{$field->{'name'}}++; |
$SelectedFields{$field->{'name'}}++; |
} |
} |
foreach my $selection (@{$ENV{'form.fieldselections'}}) { |
foreach my $selection (@{$env{'form.fieldselections'}}) { |
if ($selection eq $field->{'name'} || $selection eq 'all') { |
if ($selection eq $field->{'name'} || $selection eq 'all') { |
$field->{'selected'} = 'yes'; |
$field->{'selected'} = 'yes'; |
$SelectedFields{$field->{'name'}}++; |
$SelectedFields{$field->{'name'}}++; |
Line 608 sub CreateInterface {
|
Line 610 sub CreateInterface {
|
&parse_field_selection(); |
&parse_field_selection(); |
# |
# |
my $Str = ''; |
my $Str = ''; |
$Str .= &Apache::lonhtmlcommon::breadcrumbs |
$Str .= &Apache::lonhtmlcommon::breadcrumbs('Overall Problem Statistics', |
(undef,'Overall Problem Statistics','Statistics_Overall_Key'); |
'Statistics_Overall_Key'); |
$Str .= '<table cellspacing="5">'."\n"; |
$Str .= '<table cellspacing="5">'."\n"; |
$Str .= '<tr>'; |
$Str .= '<tr>'; |
$Str .= '<td align="center"><b>'.&mt('Sections').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Sections').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Enrollment Status').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Groups').'</b></td>'; |
|
$Str .= '<td align="center"><b>'.&mt('Access Status').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Sequences and Folders').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Sequences and Folders').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Statistics').'</b></td>'; |
$Str .= '<td align="center"><b>'.&mt('Statistics').'</b></td>'; |
$Str .= '<td rowspan="2">'. |
$Str .= '<td rowspan="2">'. |
Line 623 sub CreateInterface {
|
Line 626 sub CreateInterface {
|
$Str .= '<tr><td align="center">'."\n"; |
$Str .= '<tr><td align="center">'."\n"; |
$Str .= &Apache::lonstatistics::SectionSelect('Section','multiple',5); |
$Str .= &Apache::lonstatistics::SectionSelect('Section','multiple',5); |
$Str .= '</td><td align="center">'; |
$Str .= '</td><td align="center">'; |
|
$Str .= &Apache::lonstatistics::GroupSelect('Group','multiple',5); |
|
$Str .= '</td><td align="center">'; |
$Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5); |
$Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5); |
$Str .= '</td><td align="center">'; |
$Str .= '</td><td align="center">'; |
# |
# |
Line 661 Main interface to problem statistics.
|
Line 666 Main interface to problem statistics.
|
my $navmap; |
my $navmap; |
my @sequences; |
my @sequences; |
|
|
|
sub clean_up { |
|
undef($navmap); |
|
undef(@sequences); |
|
} |
|
|
sub BuildProblemStatisticsPage { |
sub BuildProblemStatisticsPage { |
my ($r,$c)=@_; |
my ($r,$c)=@_; |
undef($navmap); |
undef($navmap); |
Line 669 sub BuildProblemStatisticsPage {
|
Line 679 sub BuildProblemStatisticsPage {
|
my %Saveable_Parameters = ('Status' => 'scalar', |
my %Saveable_Parameters = ('Status' => 'scalar', |
'statsoutputmode' => 'scalar', |
'statsoutputmode' => 'scalar', |
'Section' => 'array', |
'Section' => 'array', |
|
'Groups' => 'array', |
'StudentData' => 'array', |
'StudentData' => 'array', |
'Maps' => 'array', |
'Maps' => 'array', |
'fieldselections'=> 'array'); |
'fieldselections'=> 'array'); |
Line 686 sub BuildProblemStatisticsPage {
|
Line 697 sub BuildProblemStatisticsPage {
|
# Finally let the user know we are here |
# Finally let the user know we are here |
my $interface = &CreateInterface($r); |
my $interface = &CreateInterface($r); |
$r->print($interface); |
$r->print($interface); |
$r->print('<input type="hidden" name="sortby" value="'.$ENV{'form.sortby'}. |
$r->print('<input type="hidden" name="sortby" value="'.$env{'form.sortby'}. |
'" />'); |
'" />'); |
# |
# |
my @CacheButtonHTML = |
my @CacheButtonHTML = |
Line 697 sub BuildProblemStatisticsPage {
|
Line 708 sub BuildProblemStatisticsPage {
|
} |
} |
# |
# |
$r->print($Str); |
$r->print($Str); |
if (! exists($ENV{'form.firstrun'})) { |
if (! exists($env{'form.firstrun'})) { |
$r->print('<h3>'. |
$r->print('<h3>'. |
&mt('Press "Generate Statistics" when you are ready.'). |
&mt('Press "Generate Statistics" when you are ready.'). |
'</h3><p>'. |
'</h3><p>'. |
&mt('It may take some time to update the student data '. |
&mt('It may take some time to update the student data '. |
'for the first analysis. Future analysis this session '. |
'for the first analysis. Future analysis this session '. |
' will not have this delay.'). |
'will not have this delay.'). |
'</p>'); |
'</p>'); |
|
&clean_up(); |
return; |
return; |
} |
} |
$r->rflush(); |
$r->rflush(); |
Line 716 sub BuildProblemStatisticsPage {
|
Line 728 sub BuildProblemStatisticsPage {
|
($navmap,@sequences) = |
($navmap,@sequences) = |
&Apache::lonstatistics::selected_sequences_with_assessments(); |
&Apache::lonstatistics::selected_sequences_with_assessments(); |
if (! ref($navmap)) { |
if (! ref($navmap)) { |
$r->print('<h1>'.&mt('A course-wide error occured.').'</h1>'. |
$r->print('<h1>'.&mt('A course-wide error occurred.').'</h1>'. |
'<h3>'.$navmap.'</h3>'); |
'<h3>'.$navmap.'</h3>'); |
|
&clean_up(); |
return; |
return; |
} |
} |
if (exists($ENV{'form.Excel'})) { |
if (exists($env{'form.Excel'})) { |
$r->print('<h4>'. |
$r->print('<h4>'. |
&Apache::lonstatistics::section_and_enrollment_description(). |
&Apache::lonstatistics::section_and_enrollment_description(). |
'</h4>'); |
'</h4>'); |
Line 748 sub BuildProblemStatisticsPage {
|
Line 761 sub BuildProblemStatisticsPage {
|
$r->rflush(); |
$r->rflush(); |
} |
} |
# |
# |
my $sortby = $ENV{'form.sortby'}; |
my $sortby = $env{'form.sortby'}; |
$sortby = 'container' if (! defined($sortby) || $sortby =~ /^\s*$/); |
$sortby = 'container' if (! defined($sortby) || $sortby =~ /^\s*$/); |
my $plot = $ENV{'form.plot'}; |
my $plot = $env{'form.plot'}; |
if ($plot eq '' || $plot eq 'none') { |
if ($plot eq '' || $plot eq 'none') { |
undef($plot); |
undef($plot); |
} |
} |
Line 765 sub BuildProblemStatisticsPage {
|
Line 778 sub BuildProblemStatisticsPage {
|
&output_sequence_statistics($r); |
&output_sequence_statistics($r); |
} |
} |
} |
} |
|
&clean_up(); |
return; |
return; |
} |
} |
|
|
Line 825 sub output_html_stats {
|
Line 839 sub output_html_stats {
|
my ($r)=@_; |
my ($r)=@_; |
&compute_all_statistics($r); |
&compute_all_statistics($r); |
$r->print(&html_preamble()); |
$r->print(&html_preamble()); |
&sort_data($ENV{'form.sortby'}); |
&sort_data($env{'form.sortby'}); |
# |
# |
my $count=0; |
my $count=0; |
foreach my $data (@StatsArray) { |
foreach my $data (@StatsArray) { |
Line 847 sub output_html_stats {
|
Line 861 sub output_html_stats {
|
sub html_preamble { |
sub html_preamble { |
my $Str=''; |
my $Str=''; |
$Str .= "<h2>". |
$Str .= "<h2>". |
$ENV{'course.'.$ENV{'request.course.id'}.'.description'}. |
$env{'course.'.$env{'request.course.id'}.'.description'}. |
"</h2>\n"; |
"</h2>\n"; |
my ($starttime,$endtime) = &Apache::lonstathelpers::get_time_limits(); |
my ($starttime,$endtime) = &Apache::lonstathelpers::get_time_limits(); |
if (defined($starttime) || defined($endtime)) { |
if (defined($starttime) || defined($endtime)) { |
Line 971 sub sequence_html_output {
|
Line 985 sub sequence_html_output {
|
sub make_plot { |
sub make_plot { |
my ($r,$plot) = @_; |
my ($r,$plot) = @_; |
&compute_all_statistics($r); |
&compute_all_statistics($r); |
&sort_data($ENV{'form.sortby'}); |
&sort_data($env{'form.sortby'}); |
if ($plot eq 'degrees') { |
if ($plot eq 'degrees') { |
°rees_plot($r); |
°rees_plot($r); |
} elsif ($plot eq 'tries statistics') { |
} elsif ($plot eq 'tries statistics') { |
Line 1225 END
|
Line 1239 END
|
sub plot_dropdown { |
sub plot_dropdown { |
my $current = ''; |
my $current = ''; |
# |
# |
if (defined($ENV{'form.plot'})) { |
if (defined($env{'form.plot'})) { |
$current = $ENV{'form.plot'}; |
$current = $env{'form.plot'}; |
} |
} |
# |
# |
my @Additional_Plots = ( |
my @Additional_Plots = ( |
Line 1279 sub Excel_output {
|
Line 1293 sub Excel_output {
|
return if (! defined($excel_workbook)); |
return if (! defined($excel_workbook)); |
# |
# |
# Add a worksheet |
# Add a worksheet |
my $sheetname = $ENV{'course.'.$ENV{'request.course.id'}.'.description'}; |
my $sheetname = $env{'course.'.$env{'request.course.id'}.'.description'}; |
if (length($sheetname) > 31) { |
if (length($sheetname) > 31) { |
$sheetname = substr($sheetname,0,31); |
$sheetname = substr($sheetname,0,31); |
} |
} |
Line 1292 sub Excel_output {
|
Line 1306 sub Excel_output {
|
# |
# |
# Put the course description in the header |
# Put the course description in the header |
$excel_sheet->write($rows_output,$cols_output++, |
$excel_sheet->write($rows_output,$cols_output++, |
$ENV{'course.'.$ENV{'request.course.id'}.'.description'}, |
$env{'course.'.$env{'request.course.id'}.'.description'}, |
$format->{'h1'}); |
$format->{'h1'}); |
$cols_output += 3; |
$cols_output += 3; |
# |
# |
Line 1302 sub Excel_output {
|
Line 1316 sub Excel_output {
|
&Apache::lonstatistics::section_and_enrollment_description('plaintext'), |
&Apache::lonstatistics::section_and_enrollment_description('plaintext'), |
$format->{'h3'}); |
$format->{'h3'}); |
$cols_output += scalar(&Apache::lonstatistics::get_selected_sections()); |
$cols_output += scalar(&Apache::lonstatistics::get_selected_sections()); |
|
$cols_output += scalar(&Apache::lonstatistics::get_selected_groups()); |
# |
# |
# Time restrictions |
# Time restrictions |
my $time_string; |
my $time_string; |
Line 1539 sub get_statistics {
|
Line 1554 sub get_statistics {
|
# |
# |
my ($starttime,$endtime) = &Apache::lonstathelpers::get_time_limits(); |
my ($starttime,$endtime) = &Apache::lonstathelpers::get_time_limits(); |
my $symb = $resource->symb; |
my $symb = $resource->symb; |
my $courseid = $ENV{'request.course.id'}; |
my $courseid = $env{'request.course.id'}; |
# |
# |
my $data = &Apache::loncoursedata::get_problem_statistics |
my $data = &Apache::loncoursedata::get_problem_statistics |
([&Apache::lonstatistics::get_selected_sections()], |
([&Apache::lonstatistics::get_selected_sections()], |
|
[&Apache::lonstatistics::get_selected_groups()], |
$Apache::lonstatistics::enrollment_status, |
$Apache::lonstatistics::enrollment_status, |
$symb,$part,$courseid,$starttime,$endtime); |
$symb,$part,$courseid,$starttime,$endtime); |
$data->{'symb'} = $symb; |
$data->{'symb'} = $symb; |
Line 1551 sub get_statistics {
|
Line 1567 sub get_statistics {
|
$data->{'container'} = $sequence->compTitle; |
$data->{'container'} = $sequence->compTitle; |
$data->{'title'} = $resource->compTitle; |
$data->{'title'} = $resource->compTitle; |
$data->{'title.link'} = $resource->src.'?symb='. |
$data->{'title.link'} = $resource->src.'?symb='. |
&Apache::lonnet::escape($resource->symb); |
&escape($resource->symb); |
# |
# |
if ($SelectedFields{'deg_of_disc'}) { |
if ($SelectedFields{'deg_of_disc'}) { |
$data->{'deg_of_disc'} = |
$data->{'deg_of_disc'} = |
Line 1564 sub get_statistics {
|
Line 1580 sub get_statistics {
|
my $sections = '"'.join(' ',@Sections).'"'; |
my $sections = '"'.join(' ',@Sections).'"'; |
$sections =~ s/&+/_/g; # Ensure no special characters |
$sections =~ s/&+/_/g; # Ensure no special characters |
$data->{'sections'}=$sections; |
$data->{'sections'}=$sections; |
$data->{'course'} = $ENV{'request.course.id'}; |
$data->{'course'} = $env{'request.course.id'}; |
my $urlres=(&Apache::lonnet::decode_symb($resource->symb))[2]; |
my $urlres=(&Apache::lonnet::decode_symb($resource->symb))[2]; |
$data->{'urlres'}=$urlres; |
$data->{'urlres'}=$urlres; |
my %storestats = |
my %storestats = |
&LONCAPA::lonmetadata::dynamic_metadata_storage($data); |
&LONCAPA::lonmetadata::dynamic_metadata_storage($data); |
my ($dom,$user) = $urlres=~/^(\w+)\/(\w+)/; |
my ($dom,$user) = ($urlres=~m{^($LONCAPA::domain_re)/($LONCAPA::username_re)}); |
&Apache::lonnet::put('nohist_resevaldata',\%storestats,$dom,$user); |
&Apache::lonnet::put('nohist_resevaldata',\%storestats,$dom,$user); |
} |
} |
# |
# |
Line 1577 sub get_statistics {
|
Line 1593 sub get_statistics {
|
($data->{'num_solved'}+0.1); |
($data->{'num_solved'}+0.1); |
# |
# |
# Get the due date for research purposes (commented out most of the time) |
# Get the due date for research purposes (commented out most of the time) |
# $data->{'duedate'} = |
# my $duedate = &Apache::lonnet::EXT('resource.'.$part.'.duedate',$symb);; |
# &Apache::lonnet::EXT('resource.'.$part.'.duedate',$symb); |
# my $opendate = &Apache::lonnet::EXT('resource.'.$part.'.opendate',$symb); |
# $data->{'opendate'} = |
# my $maxtries = &Apache::lonnet::EXT('resource.'.$part.'.maxtries',$symb); |
# &Apache::lonnet::EXT('resource.'.$part.'.opendate',$symb); |
# my $hinttries = &Apache::lonnet::EXT('resource.'.$part.'.hinttries',$symb); |
# $data->{'maxtries'} = |
my $weight = &Apache::lonnet::EXT('resource.'.$part.'.weight',$symb); |
# &Apache::lonnet::EXT('resource.'.$part.'.maxtries',$symb); |
$data->{'weight'} = $weight; |
# $data->{'hinttries'} = |
# $data->{'duedate'} = $duedate; |
# &Apache::lonnet::EXT('resource.'.$part.'.hinttries',$symb); |
# $data->{'opendate'} = $opendate; |
$data->{'weight'} = |
# $data->{'maxtries'} = $maxtries; |
&Apache::lonnet::EXT('resource.'.$part.'.weight',$symb); |
# $data->{'hinttries'} = $hinttries; |
# $data->{'resptypes'} = join(',',@{$resource->{'partdata'}->{$part}->{'ResponseTypes'}}); |
# $data->{'resptypes'} = join(',',@{$resource->{'partdata'}->{$part}->{'ResponseTypes'}}); |
return $data; |
return $data; |
} |
} |
Line 1621 sub compute_discrimination_factor {
|
Line 1637 sub compute_discrimination_factor {
|
&Apache::loncoursedata::rank_students_by_scores_on_resources |
&Apache::loncoursedata::rank_students_by_scores_on_resources |
(\@Resources, |
(\@Resources, |
[&Apache::lonstatistics::get_selected_sections()], |
[&Apache::lonstatistics::get_selected_sections()], |
|
[&Apache::lonstatistics::get_selected_groups()], |
$Apache::lonstatistics::enrollment_status,undef, |
$Apache::lonstatistics::enrollment_status,undef, |
$starttime,$endtime); |
$starttime,$endtime, $symb); |
# |
# |
# compute their percent scores on the problems in the sequence, |
# compute their percent scores on the problems in the sequence, |
my $number_to_grab = int(scalar(@{$ranking})/4); |
my $number_to_grab = int(scalar(@{$ranking})/4); |
Line 1632 sub compute_discrimination_factor {
|
Line 1649 sub compute_discrimination_factor {
|
my @TopSet = |
my @TopSet = |
map { |
map { |
$_->[&Apache::loncoursedata::RNK_student()]; |
$_->[&Apache::loncoursedata::RNK_student()]; |
} @{$ranking}[($num_students-$number_to_grab)..($num_students-1)]; |
} @{$ranking}[-$number_to_grab..0]; |
if (! @BottomSet || (@BottomSet == 1 && $BottomSet[0] eq '') || |
if (! @BottomSet || (@BottomSet == 1 && $BottomSet[0] eq '') || |
! @TopSet || (@TopSet == 1 && $TopSet[0] eq '')) { |
! @TopSet || (@TopSet == 1 && $TopSet[0] eq '')) { |
return 'nan'; |
return 'nan'; |
Line 1686 sub compute_sequence_statistics {
|
Line 1703 sub compute_sequence_statistics {
|
my ($smin,$smax,$sMean,$sSTD,$scount,$sMAX) = |
my ($smin,$smax,$sMean,$sSTD,$scount,$sMAX) = |
&Apache::loncoursedata::score_stats |
&Apache::loncoursedata::score_stats |
([&Apache::lonstatistics::get_selected_sections()], |
([&Apache::lonstatistics::get_selected_sections()], |
|
[&Apache::lonstatistics::get_selected_groups()], |
$Apache::lonstatistics::enrollment_status, |
$Apache::lonstatistics::enrollment_status, |
\@Resources,$starttime,$endtime,undef); |
\@Resources,$starttime,$endtime,undef); |
$SeqStat{$symb}->{'title'} = $seq->compTitle; |
$SeqStat{$symb}->{'title'} = $seq->compTitle; |
Line 1701 sub compute_sequence_statistics {
|
Line 1719 sub compute_sequence_statistics {
|
my ($cmin,$cmax,$cMean,$cSTD,$ccount)= |
my ($cmin,$cmax,$cMean,$cSTD,$ccount)= |
&Apache::loncoursedata::count_stats |
&Apache::loncoursedata::count_stats |
([&Apache::lonstatistics::get_selected_sections()], |
([&Apache::lonstatistics::get_selected_sections()], |
|
[&Apache::lonstatistics::get_selected_groups()], |
$Apache::lonstatistics::enrollment_status, |
$Apache::lonstatistics::enrollment_status, |
\@Resources,$starttime,$endtime,undef); |
\@Resources,$starttime,$endtime,undef); |
my $K = $part_count; |
my $K = $part_count; |