Diff for /loncom/interface/statistics/lonproblemanalysis.pm between versions 1.113 and 1.116

version 1.113, 2005/02/22 05:28:21 version 1.116, 2005/02/28 23:07:32
Line 97  sub BuildProblemAnalysisPage { Line 97  sub BuildProblemAnalysisPage {
             $r->print($html.(' 'x5));              $r->print($html.(' 'x5));
         }          }
         #          #
           $r->print(&Apache::lonstathelpers::submission_report_form
                                                         ('problem_analysis'));
           #
         $r->print('<hr />');          $r->print('<hr />');
         $r->rflush();          $r->rflush();
         #          #
Line 104  sub BuildProblemAnalysisPage { Line 107  sub BuildProblemAnalysisPage {
         my $current_problem = &Apache::lonstathelpers::get_target_from_id          my $current_problem = &Apache::lonstathelpers::get_target_from_id
             ($ENV{'form.problemchoice'});              ($ENV{'form.problemchoice'});
         #          #
         my ($prev,$curr,$next) =           my ($navmap,$prev,$curr,$next) = 
             &Apache::lonstathelpers::get_prev_curr_next($current_problem,              &Apache::lonstathelpers::get_prev_curr_next($current_problem,
                                                         $problem_types,                                                          $problem_types,
                                                         'response',                                                          'response',
Line 127  sub BuildProblemAnalysisPage { Line 130  sub BuildProblemAnalysisPage {
             $r->print('resource is undefined');              $r->print('resource is undefined');
         } else {          } else {
             my $resource = $current_problem->{'resource'};              my $resource = $current_problem->{'resource'};
             $r->print('<h1>'.$resource->{'title'}.'</h1>');              $r->print('<h1>'.$resource->compTitle.'</h1>');
             $r->print('<h3>'.$resource->{'src'}.'</h3>');              $r->print('<h3>'.$resource->src.'</h3>');
             if ($ENV{'form.show_prob'} eq 'true') {              if ($ENV{'form.show_prob'} eq 'true') {
                 $r->print(&Apache::lonstathelpers::render_resource($resource));                  $r->print(&Apache::lonstathelpers::render_resource($resource));
             }              }
             $r->rflush();              $r->rflush();
             my %Data = &Apache::lonstathelpers::get_problem_data              my %Data = &Apache::lonstathelpers::get_problem_data
                 ($resource->{'src'});                  ($resource->src);
             my $problem_data = $Data{$current_problem->{'part'}.              my $problem_data = $Data{$current_problem->{'part'}.
                                     '.'.                                      '.'.
                                     $current_problem->{'respid'}};                                      $current_problem->{'respid'}};
Line 159  sub BuildProblemAnalysisPage { Line 162  sub BuildProblemAnalysisPage {
                   &mt('Analyze Problem').'" />');                    &mt('Analyze Problem').'" />');
         $r->print('&nbsp;'x5);          $r->print('&nbsp;'x5);
         $r->print('<h3>'.&mt('Please select a problem to analyze').'</h3>');          $r->print('<h3>'.&mt('Please select a problem to analyze').'</h3>');
         $r->print(&Apache::lonstathelpers::ProblemSelector          $r->print(&Apache::lonstathelpers::problem_selector($problem_types));
                   ($problem_types));  
     }      }
 }  }
   
Line 186  sub numerical_response_analysis { Line 188  sub numerical_response_analysis {
     my $response_data = &Apache::loncoursedata::get_response_data      my $response_data = &Apache::loncoursedata::get_response_data
         (\@Apache::lonstatistics::SelectedSections,          (\@Apache::lonstatistics::SelectedSections,
          $Apache::lonstatistics::enrollment_status,           $Apache::lonstatistics::enrollment_status,
          $resource->{'symb'},$respid);           $resource->symb,$respid);
     #      #
     $problem_analysis->{'answercomputed'} = 1;      $problem_analysis->{'answercomputed'} = 1;
     if ($problem_analysis->{'answercomputed'}) {      if ($problem_analysis->{'answercomputed'}) {
Line 303  sub numerical_plot_percent { Line 305  sub numerical_plot_percent {
     #      #
     my $total = $stats->{'submission_count'};      my $total = $stats->{'submission_count'};
     return '' if ($total == 0);      return '' if ($total == 0);
     my $min_bin_size = $stats->{'min_percent'};      my $max_bins = 50;
     my $low_bin  = $stats->{'lowest_ans'}-$stats->{'max_bin_size'};      my $lowest_percent = $stats->{'min_percent'};
     my $high_bin = $stats->{'highest_ans'}+$stats->{'max_bin_size'};      my $highest_percent = $stats->{'max_percent'};
     if ($high_bin > 0 && $low_bin > -$high_bin) {      my $percent_spread = $highest_percent - $lowest_percent;
         $low_bin = -$high_bin;      foreach (qw/20 30 40 50 100 200/) {
     } elsif ($low_bin < 0 && $high_bin < -$low_bin) {          if ($percent_spread < $_) {
         $high_bin = -$low_bin;              $highest_percent =$_/2;
               last;
           }
     }      }
     if (($high_bin -$low_bin)/$min_bin_size > 1000) {      my $bin_size = 1;
         $min_bin_size = abs($high_bin - $low_bin) / 1000;      foreach (qw/0.01 0.05 0.1 0.5 1 2 5 10 20 25 50 100/) {
           if ($lowest_percent/2 < $_){
               $bin_size = $_;
               last;
           }
     }      }
     my @bins;      my @bins;
     for (my $num = $low_bin;$num <= $high_bin;$num+=($min_bin_size/2)) {      for (my $bin = -$highest_percent;$bin<0;$bin+=$bin_size) {
         push(@bins,$num);          push (@bins,$bin);
     }      }
       for (my $bin = 0; $bin<$highest_percent;$bin+=$bin_size) {
           push (@bins,$bin);
       }
       push(@bins,$highest_percent);
     #      #
     my @correct;      my @correct;
     my @incorrect;      my @incorrect;
Line 325  sub numerical_plot_percent { Line 337  sub numerical_plot_percent {
     while (my ($ans,$submissions) = each(%$responses)) {      while (my ($ans,$submissions) = each(%$responses)) {
         while (my ($submission,$counts) = each(%$submissions)) {          while (my ($submission,$counts) = each(%$submissions)) {
             my ($correct_count,$incorrect_count) = @$counts;              my ($correct_count,$incorrect_count) = @$counts;
             my $scaled_value = ($submission-$ans)/$ans;              my $scaled_value = 100*($submission-$ans)/abs($ans);
             my $bin=0;              if ($scaled_value < $bins[0]) {
             for ($bin=0;$bin<$#bins;$bin++) { # not <= for a reason                  $bins[0]=$scaled_value -1;
               }
               my $bin;
               for ($bin=0;$bin<$#bins;$bin++) {
                 last if ($bins[$bin]>$scaled_value);                  last if ($bins[$bin]>$scaled_value);
             }              }
             $correct[$bin]+=$correct_count;              $correct[$bin-1]+=$correct_count;
             $incorrect[$bin]+=$incorrect_count;              $incorrect[$bin-1]+=$incorrect_count;
             $count[$bin]+=$correct_count+$incorrect_count;              $count[$bin-1]+=$correct_count+$incorrect_count;
         }          }
     }      }
     #      #
     # Skip empty bins      my @plot_correct   = @correct;
     my (@plot_correct,@plot_incorrect,@new_bins,@new_count);      my @plot_incorrect = @incorrect;
     my $min_skip = 2;      my $max;
     for (my $i=0;$i<=$#bins;$i++) {      for (my $i=0;$i<$#bins;$i++) {
         my $sum=0;  
         for (my $j=-$min_skip;$j<=$min_skip && $i+$j<=$#bins;$j++) {  
             $sum += $correct[$i+$j] + $incorrect[$i+$j];  
         }  
         if ($sum) {  
             push(@new_bins,$bins[$i]);  
             push(@plot_correct,$correct[$i]);  
             push(@plot_incorrect,$incorrect[$i]);  
             push(@new_count,$correct[$i]+$incorrect[$i]);  
         }  
     }  
     @correct   = @plot_correct;  
     @incorrect = @plot_incorrect;  
     @count     = @new_count;  
     @bins      = @new_bins;  
     for (my $i=0;$i<=$#bins;$i++) {  
         $plot_correct[$i]   *= 100/$total;          $plot_correct[$i]   *= 100/$total;
         $plot_incorrect[$i] *= 100/$total;          $plot_incorrect[$i] *= 100/$total;
           if (! defined($max) || 
               $max < $plot_correct[$i]+$plot_incorrect[$i] ) {
               $max = $plot_correct[$i]+$plot_incorrect[$i];
           }
       }
       foreach (qw/1 5 10 15 20 25 30 40 50 75 100/) {
           if ($max <$_) { $max = $_; last; }
     }      }
     #      #
     my $title = &mt('Percent Difference');      my $title = &mt('Percent Difference');
     my @labels = (1..scalar(@bins));      my @labels = (1..scalar(@bins)-1);
     my $graph = &Apache::loncommon::DrawBarGraph      my $graph = &Apache::loncommon::DrawBarGraph
         ($title,'Percent difference from correct','Number of answers',          ($title,'Percent Difference from Correct','Percent of Answers',
          100,['#33FF00','#FF3300'],\@labels,\@plot_correct,\@plot_incorrect,           $max,['#33FF00','#FF3300'],\@labels,\@plot_correct,\@plot_incorrect,
          {xskip=>1});           {xskip=>1});
     #      #
     my $table = $graph.$/.      my $table = $graph.$/.
Line 377  sub numerical_plot_differences { Line 383  sub numerical_plot_differences {
     #      #
     my $total = $stats->{'submission_count'};      my $total = $stats->{'submission_count'};
     return '' if ($total == 0);      return '' if ($total == 0);
     my $max_bins = 50;      my $max_bins = 21;
     my $min_bin_size = $stats->{'min_abs'};      my $min_bin_size = $stats->{'min_abs'};
     my $low_bin  = $stats->{'lowest_ans'}-$stats->{'max_bin_size'};      my $low_bin  = $stats->{'lowest_ans'}-$stats->{'max_bin_size'};
     my $high_bin = $stats->{'highest_ans'}+$stats->{'max_bin_size'};      my $high_bin = $stats->{'highest_ans'}+$stats->{'max_bin_size'};
Line 401  sub numerical_plot_differences { Line 407  sub numerical_plot_differences {
         while (my ($submission,$counts) = each(%$submissions)) {          while (my ($submission,$counts) = each(%$submissions)) {
             my ($correct_count,$incorrect_count) = @$counts;              my ($correct_count,$incorrect_count) = @$counts;
             my $scaled_value = $submission-$ans;              my $scaled_value = $submission-$ans;
               if ($scaled_value < $bins[0]) {
                   $bins[0]=$scaled_value-1;
               }
             my $bin=0;              my $bin=0;
             for ($bin=0;$bin<$#bins;$bin++) { # not <= for a reason              for ($bin=0;$bin<$#bins;$bin++) {
                 last if ($bins[$bin]>$scaled_value);                  last if ($bins[$bin]>$scaled_value);
             }              }
             $correct[$bin]+=$correct_count;              $correct[$bin-1]+=$correct_count;
             $incorrect[$bin]+=$incorrect_count;              $incorrect[$bin-1]+=$incorrect_count;
             $count[$bin]+=$correct_count+$incorrect_count;              $count[$bin-1]+=$correct_count+$incorrect_count;
         }          }
     }      }
     my @plot_correct   = @correct;      my @plot_correct   = @correct;
     my @plot_incorrect = @incorrect;      my @plot_incorrect = @incorrect;
       my $max;
     for (my $i=0;$i<=$#bins;$i++) {      for (my $i=0;$i<=$#bins;$i++) {
         $plot_correct[$i]   *= 100/$total;          $plot_correct[$i]   *= 100/$total;
         $plot_incorrect[$i] *= 100/$total;          $plot_incorrect[$i] *= 100/$total;
           if (! defined($max) || 
               $max < $plot_correct[$i]+$plot_incorrect[$i] ) {
               $max = $plot_correct[$i]+$plot_incorrect[$i];
           }
       }
       foreach (qw/1 5 10 15 20 25 30 40 50 75 100/) {
           if ($max <$_) { $max = $_; last; }
     }      }
     #      #
     my $title = &mt('Difference between submission and correct');      my $title = &mt('Difference between submission and correct');
     my @labels = (1..scalar(@bins));      my @labels = (1..scalar(@bins)-1);
     my $graph = &Apache::loncommon::DrawBarGraph      my $graph = &Apache::loncommon::DrawBarGraph
         ($title,'Difference from Correct','Number of answers',          ($title,'Difference from Correct','Percent of Answers',
          100,['#33FF00','#FF3300'],\@labels,\@plot_correct,\@plot_incorrect,           $max,['#33FF00','#FF3300'],\@labels,\@plot_correct,\@plot_incorrect,
          {xskip=>1});           {xskip=>1});
     #      #
     my $table = $graph.$/.      my $table = $graph.$/.
Line 498  sub numerical_classify_responses { Line 515  sub numerical_classify_responses {
                 &capa::caparesponse_get_real_response($myunit,                  &capa::caparesponse_get_real_response($myunit,
                                                       $mysub,                                                        $mysub,
                                                       \$scaled);                                                        \$scaled);
             &Apache::lonnet::logthis('scaled = '.$scaled.' result ='.$result);  #            &Apache::lonnet::logthis('scaled = '.$scaled.' result ='.$result);
             next if (! defined($scaled));              next if (! defined($scaled));
 #            next if ($result ne '6');  #            next if ($result ne '6');
             my $submission = $scaled;              my $submission = $scaled;
Line 546  sub numerical_bin_table { Line 563  sub numerical_bin_table {
         '</tr>'.$/;          '</tr>'.$/;
     for (my $i=0;$i<scalar(@{$bins}-1);$i++) {      for (my $i=0;$i<scalar(@{$bins}-1);$i++) {
         my $lownum = $bins->[$i];          my $lownum = $bins->[$i];
           if ($i == 0) { $lownum = '-&infin;'; }
         my $highnum = $bins->[$i+1];          my $highnum = $bins->[$i+1];
           if ($i == scalar(@{$bins})-2) { $highnum = '&infin;'; }
         $table .=           $table .= 
             '<tr>'.              '<tr>'.
             '<td>'.$labels->[$i].'</td>'.              '<td>'.$labels->[$i].'</td>'.
Line 748  sub radio_response_analysis { Line 767  sub radio_response_analysis {
     my $response_data = &Apache::loncoursedata::get_response_data      my $response_data = &Apache::loncoursedata::get_response_data
         (\@Apache::lonstatistics::SelectedSections,          (\@Apache::lonstatistics::SelectedSections,
          $Apache::lonstatistics::enrollment_status,           $Apache::lonstatistics::enrollment_status,
          $resource->{'symb'},$respid);           $resource->symb,$respid);
     my $correct;   # either a hash reference or a scalar      my $correct;   # either a hash reference or a scalar
     if ($problem_analysis->{'answercomputed'} || scalar(@$concepts) > 1) {      if ($problem_analysis->{'answercomputed'} || scalar(@$concepts) > 1) {
         # This takes a while for large classes...          # This takes a while for large classes...
Line 1151  sub OptionResponseAnalysis { Line 1170  sub OptionResponseAnalysis {
     my ($r,$problem,$problem_data,$Students) = @_;      my ($r,$problem,$problem_data,$Students) = @_;
     my ($resource,$respid) = ($problem->{'resource'},      my ($resource,$respid) = ($problem->{'resource'},
                               $problem->{'respid'});                                $problem->{'respid'});
       &Apache::lonnet::logthis('option response analysis on '.$resource->symb);
     # Note: part data is not needed.      # Note: part data is not needed.
     my $PerformanceData = &Apache::loncoursedata::get_response_data      my $PerformanceData = &Apache::loncoursedata::get_response_data
         (\@Apache::lonstatistics::SelectedSections,          (\@Apache::lonstatistics::SelectedSections,
          $Apache::lonstatistics::enrollment_status,           $Apache::lonstatistics::enrollment_status,
          $resource->{'symb'},$respid);           $resource->symb,$respid);
     if (! defined($PerformanceData) ||       if (! defined($PerformanceData) || 
         ref($PerformanceData) ne 'ARRAY' ) {          ref($PerformanceData) ne 'ARRAY' ) {
         $r->print('<h2>'.          $r->print('<h2>'.
Line 1196  sub OR_tries_analysis { Line 1216  sub OR_tries_analysis {
     if (! defined($Concepts)) {      if (! defined($Concepts)) {
         $Concepts = [];          $Concepts = [];
     }      }
       &Apache::lonnet::logthis('got '.scalar(@$PerformanceData).' data points');
     my %response_data = &OR_analyze_by_tries($r,$PerformanceData,      my %response_data = &OR_analyze_by_tries($r,$PerformanceData,
                                                      $mintries,$maxtries);                                                       $mintries,$maxtries);
     my $analysis = '';      my $analysis = '';
Line 1252  sub OR_tries_analysis { Line 1273  sub OR_tries_analysis {
     if (defined($Concepts)) { $num_concepts = scalar(@$Concepts); }      if (defined($Concepts)) { $num_concepts = scalar(@$Concepts); }
     #      #
     for (my $try=$mintries;$try<=$maxtries;$try++) {      for (my $try=$mintries;$try<=$maxtries;$try++) {
           &Apache::lonnet::logthis('try = '.$try);
         if (! defined($response_data{'_total'}->[$try]) ||          if (! defined($response_data{'_total'}->[$try]) ||
             $response_data{'_total'}->[$try] == 0) {               $response_data{'_total'}->[$try] == 0) { 
             if ($try > 1) {              if ($try > 1) {
Line 1285  sub OR_tries_analysis { Line 1307  sub OR_tries_analysis {
                 ('Correct Concepts','Concept Number','Percent Correct',                  ('Correct Concepts','Concept Number','Percent Correct',
                  100,$plotcolors,undef,\@concept_plot_data,{xskip=>1});                   100,$plotcolors,undef,\@concept_plot_data,{xskip=>1});
         }          }
           &Apache::lonnet::logthis('got here a_0');
         #          #
         # Create Foil Plots          # Create Foil Plots
         my $data_count = $response_data{'_total'}->[$try];          my $data_count = $response_data{'_total'}->[$try];
         my $correct = $response_data{'_correct'}->[$try];          my $correct = $response_data{'_correct'}->[$try];
         my @Datasets;          my @Datasets;
         foreach my $option ('_correct',@{$ORdata->{'_Options'}}) {          foreach my $option ('_correct',@{$ORdata->{'_Options'}}) {
               &Apache::lonnet::logthis('checking option '.$option);
             next if (! exists($foil_plot[$try]->{$option}));              next if (! exists($foil_plot[$try]->{$option}));
             push(@Datasets,$foil_plot[$try]->{$option});              push(@Datasets,$foil_plot[$try]->{$option});
         }          }
           &Apache::lonnet::logthis('got here a');
         #          #
         # Put a blank in the data set between concepts          # Put a blank in the data set between concepts
         for (my $set =0;$set<=$#Datasets;$set++) {          for (my $set =0;$set<=$#Datasets;$set++) {
Line 1310  sub OR_tries_analysis { Line 1335  sub OR_tries_analysis {
         }          }
         #          #
         # Set up the labels needed for the bar graph          # Set up the labels needed for the bar graph
           &Apache::lonnet::logthis('got here b');
         my @Labels;          my @Labels;
         my $idx = 1;          my $idx = 1;
         foreach my $concept (@{$Concepts}) {          foreach my $concept (@{$Concepts}) {
Line 1325  sub OR_tries_analysis { Line 1351  sub OR_tries_analysis {
                   
         #          #
         #          #
           &Apache::lonnet::logthis('got here c');
         next if (! defined($Datasets[0]));          next if (! defined($Datasets[0]));
           &Apache::lonnet::logthis('got here d');
         for (my $i=0; $i< scalar(@{$Datasets[0]});$i++) {          for (my $i=0; $i< scalar(@{$Datasets[0]});$i++) {
             $Datasets[0]->[$i]=0;              $Datasets[0]->[$i]=0;
         }          }
Line 1334  sub OR_tries_analysis { Line 1362  sub OR_tries_analysis {
         my $incorrect_graph = &Apache::loncommon::DrawBarGraph          my $incorrect_graph = &Apache::loncommon::DrawBarGraph
             ('Incorrect Statements','Statement','% Chosen Incorrectly',              ('Incorrect Statements','Statement','% Chosen Incorrectly',
              100,$plotcolors,\@Labels,@Datasets,{xskip=>1});               100,$plotcolors,\@Labels,@Datasets,{xskip=>1});
           &Apache::lonnet::logthis('incorrect graph = '.$incorrect_graph);
         $analysis_html.=           $analysis_html.= 
             '<tr><td colspan="4" align="center">'.              '<tr><td colspan="4" align="center">'.
             '<font size="+1">'.              '<font size="+1">'.
Line 1794  sub CreateInterface { Line 1823  sub CreateInterface {
     $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('Enrollment Status').'</b></td>';
 #    $Str .= '<td align="center"><b>'.&mt('Sequences and Folders').'</b></td>';  
     $Str .= '<td align="center">&nbsp;</td>';      $Str .= '<td align="center">&nbsp;</td>';
     $Str .= '</tr>'."\n";      $Str .= '</tr>'."\n";
     ##      ##
Line 1807  sub CreateInterface { Line 1835  sub CreateInterface {
     $Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5);      $Str .= &Apache::lonhtmlcommon::StatusOptions(undef,undef,5);
     $Str .= '</td>';      $Str .= '</td>';
     #      #
 #    $Str .= '<td align="center">';  
     my $only_seq_with_assessments = sub {   
         my $s=shift;  
         if ($s->{'num_assess'} < 1) {   
             return 0;  
         } else {   
             return 1;  
         }  
     };  
     &Apache::lonstatistics::MapSelect('Maps','multiple,all',5,  
                                               $only_seq_with_assessments);  
     ##      ##
     ##      ##
     $Str .= '<td>';      $Str .= '<td>';

Removed from v.1.113  
changed lines
  Added in v.1.116


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