version 1.15, 2002/10/22 16:27:39
|
version 1.27, 2003/01/13 01:34:11
|
Line 30
|
Line 30
|
# 11/1,11/4,11/16,12/14,12/16,12/18,12/20,12/31 Behrouz Minaei |
# 11/1,11/4,11/16,12/14,12/16,12/18,12/20,12/31 Behrouz Minaei |
# YEAR=2002 |
# YEAR=2002 |
# 1/22,2/1,2/6,2/25,3/2,3/6,3/17,3/21,3/22,3/26,4/7,5/6 Behrouz Minaei |
# 1/22,2/1,2/6,2/25,3/2,3/6,3/17,3/21,3/22,3/26,4/7,5/6 Behrouz Minaei |
# 5/12,5/14,5/15,5/19,5/26,7/16 Behrouz Minaei |
# 5/12,5/14,5/15,5/19,5/26,7/16,12/24 Behrouz Minaei |
# |
# YEAR=2003 |
|
# 1/11 Behrouz Minaei |
### |
### |
|
|
package Apache::lonstudentassessment; |
package Apache::lonstudentassessment; |
|
|
use strict; |
use strict; |
use Apache::lonhtmlcommon; |
use Apache::lonhtmlcommon; |
use Apache::loncoursedata; |
use Apache::loncoursedata; |
use GDBM_File; |
use GDBM_File; |
|
|
#my $jr; |
|
|
|
sub BuildStudentAssessmentPage { |
sub BuildStudentAssessmentPage { |
my ($cacheDB,$students,$courseID,$formName,$headings,$spacing, |
my ($cacheDB,$students,$courseID,$formName,$headings,$spacing, |
$studentInformation,$r,$c)=@_; |
$studentInformation,$r,$c)=@_; |
# $jr = $r; |
|
my %cache; |
my %cache; |
unless(tie(%cache,'GDBM_File',$cacheDB,&GDBM_READER(),0640)) { |
unless(tie(%cache,'GDBM_File',$cacheDB,&GDBM_READER(),0640)) { |
$r->print('<html><body>Unable to tie database.</body></html>'); |
$r->print('<html><body>Unable to tie database.</body></html>'); |
Line 77 sub BuildStudentAssessmentPage {
|
Line 76 sub BuildStudentAssessmentPage {
|
} |
} |
} |
} |
my ($infoHeadings, $infoKeys, $sequenceHeadings, $sequenceKeys, |
my ($infoHeadings, $infoKeys, $sequenceHeadings, $sequenceKeys, |
$doNotShow) = |
$doNotShow) = |
&ShouldShowColumns(\%cache, $headings, $studentInformation); |
&ShouldShowColumns(\%cache, $headings, $studentInformation); |
|
|
my $selectedName = &FindSelectedStudent(\%cache, |
my $selectedName = &FindSelectedStudent(\%cache, |
$cache{'StudentAssessmentStudent'}, |
$cache{'StudentAssessmentStudent'}, |
$students); |
$students); |
$r->print(&CreateInterface(\%cache, $selectedName, $students, $formName, |
$r->print(&CreateInterface(\%cache, $selectedName, $students, $formName, |
Line 101 sub BuildStudentAssessmentPage {
|
Line 100 sub BuildStudentAssessmentPage {
|
if($c->aborted()) { return $Str; } |
if($c->aborted()) { return $Str; } |
|
|
my $selected=0; |
my $selected=0; |
|
my $Count = 0; |
$r->print('<pre>'."\n"); |
$r->print('<pre>'."\n"); |
foreach (@$students) { |
foreach (@$students) { |
if($c->aborted()) { return $Str; } |
if($c->aborted()) { return $Str; } |
next if ($_ ne $selectedName && |
next if ($_ ne $selectedName && |
$selectedName ne 'All Students'); |
$selectedName ne 'All Students'); |
$selected = 1; |
$selected = 1; |
|
|
my @who = ($_); |
my @who = ($_); |
next if(&Apache::loncoursedata::DownloadStudentCourseData(\@who, 'true', |
next if(&Apache::loncoursedata::DownloadStudentCourseData(\@who, 'true', |
$cacheDB, 'true', |
$cacheDB, 'true', |
'false', $courseID, |
'false', $courseID, |
$r, $c) ne 'OK'); |
$r, $c) ne 'OK'); |
next if($c->aborted()); |
next if($c->aborted()); |
Line 132 sub BuildStudentAssessmentPage {
|
Line 132 sub BuildStudentAssessmentPage {
|
push(@before, $_); |
push(@before, $_); |
} |
} |
} |
} |
my $displayString = 'DISPLAYDATA'.$spacing; |
$Count++; |
$r->print(&Apache::lonhtmlcommon::FormatStudentInformation( |
my $out = ''; |
|
$out .= sprintf("%3d) ", $Count); |
|
if($Count % 2) { |
|
$out .= '<bgcolor="#FFFFFF">'; |
|
} else { |
|
$out .= '<bgcolor="#505050">'; |
|
} |
|
my $displayString = $out.'DISPLAYDATA'.$spacing; |
|
$r->print(&Apache::lonhtmlcommon::FormatStudentInformation( |
\%cache, $_, |
\%cache, $_, |
\@before, |
\@before, |
$displayString, |
$displayString, |
Line 164 sub BuildStudentAssessmentPage {
|
Line 172 sub BuildStudentAssessmentPage {
|
untie(%cache); |
untie(%cache); |
} |
} |
} |
} |
$r->print('</pre>'."\n"); |
$r->print('</pre>'."\n"); |
if($selected == 0) { |
if($selected == 0) { |
$Str .= '<h3><font color=blue>WARNING: '; |
$Str .= '<h3><font color=blue>WARNING: '; |
$Str .= 'Please select a student</font></h3>'; |
$Str .= 'No Students enrolled OR Please select a student</font></h3>'; |
$r->print($Str); |
$r->print($Str); |
|
} else { |
|
if(tie(%cache,'GDBM_File',$cacheDB,&GDBM_READER(),0640)) { |
|
$r->print(&StudentAverageTotal(\%cache, $students, $sequenceKeys)); |
|
untie(%cache); |
|
} |
} |
} |
|
|
return; |
return; |
} |
} |
|
|
|
|
#---- Student Assessment Web Page -------------------------------------------- |
#---- Student Assessment Web Page -------------------------------------------- |
|
|
sub CreateInterface { |
sub CreateInterface { |
Line 185 sub CreateInterface {
|
Line 198 sub CreateInterface {
|
$Str .= '<input type="submit" name="PreviousStudent" '; |
$Str .= '<input type="submit" name="PreviousStudent" '; |
$Str .= 'value="Previous Student" />'."\n"; |
$Str .= 'value="Previous Student" />'."\n"; |
$Str .= '   '."\n"; |
$Str .= '   '."\n"; |
$Str .= &Apache::lonhtmlcommon::StudentOptions($cache, $students, |
$Str .= &Apache::lonhtmlcommon::StudentOptions($cache, $students, |
$selectedName, |
$selectedName, |
'StudentAssessment', |
'StudentAssessment', |
$formName); |
$formName); |
$Str .= "\n".'   '."\n"; |
$Str .= "\n".'   '."\n"; |
$Str .= '<input type="submit" name="NextStudent" '; |
$Str .= '<input type="submit" name="NextStudent" '; |
Line 210 sub CreateInterface {
|
Line 223 sub CreateInterface {
|
$Str .= '</td><td>'."\n"; |
$Str .= '</td><td>'."\n"; |
$Str .= '<input type="submit" name="DefaultColumns" '; |
$Str .= '<input type="submit" name="DefaultColumns" '; |
$Str .= 'value="Default Column Display" />'."\n"; |
$Str .= 'value="Default Column Display" />'."\n"; |
|
$Str .= '</td><td>'."\n"; |
|
$Str .= '<input type="submit" name="displaymode" '; |
|
if (! exists($ENV{'form.displaymode'}) || |
|
lc($ENV{'form.displaymode'}) eq 'display with links') { |
|
$Str .= 'value="Display without links" />'; |
|
# Set the current value, in case it is undefined |
|
$ENV{'form.displaymode'} = 'Display with links'; |
|
} else { |
|
$Str .= 'value="Display with links" />'; |
|
} |
|
$Str .= "\n"; |
$Str .= '</td></tr></table>'."\n"; |
$Str .= '</td></tr></table>'."\n"; |
|
|
return $Str; |
return $Str; |
Line 219 sub CreateTableHeadings {
|
Line 243 sub CreateTableHeadings {
|
my($cache,$spacing,$infoKeys,$infoHeadings,$sequenceKeys, |
my($cache,$spacing,$infoKeys,$infoHeadings,$sequenceKeys, |
$sequenceHeadings)=@_; |
$sequenceHeadings)=@_; |
|
|
|
# my $Str = '     '; |
my $Str = ''; |
my $Str = ''; |
$Str .= '<table border="0" cellpadding="0" cellspacing="0">'."\n"; |
$Str .= '<table border="0" cellpadding="0" cellspacing="0">'."\n"; |
|
|
$Str .= '<tr>'."\n"; |
$Str .= '<tr>'."\n"; |
|
$Str .= '<td><pre> </pre></td>'."\n"; |
$Str .= &CreateColumnSelectors($infoHeadings, $sequenceHeadings, |
$Str .= &CreateColumnSelectors($infoHeadings, $sequenceHeadings, |
$sequenceKeys); |
$sequenceKeys); |
$Str .= '<td></td></tr>'."\n"; |
$Str .= '<td></td></tr>'."\n"; |
|
|
$Str .= '<tr>'."\n"; |
$Str .= '<tr>'."\n"; |
my $displayString = '<td align="left"><pre><a href="/adm/statistics?'; |
my $displayString = ''; |
|
$displayString .= '<td><pre> </pre></td>'."\n"; |
|
$displayString .= '<td align="left"><pre><a href="/adm/statistics?'; |
$displayString .= 'sort=LINKDATA">DISPLAYDATA</a>FORMATTING'; |
$displayString .= 'sort=LINKDATA">DISPLAYDATA</a>FORMATTING'; |
$displayString .= $spacing.'</pre></td>'."\n"; |
$displayString .= $spacing.'</pre></td>'."\n"; |
$Str .= &Apache::lonhtmlcommon::CreateHeadings($cache, |
$Str .= &Apache::lonhtmlcommon::CreateHeadings($cache, |
$infoKeys, |
$infoKeys, |
$infoHeadings, |
$infoHeadings, |
$displayString, |
$displayString, |
Line 256 sub CreateTableHeadings {
|
Line 283 sub CreateTableHeadings {
|
=item &FormatStudentData() |
=item &FormatStudentData() |
|
|
First, FormatStudentInformation is called and prefixes the course information. |
First, FormatStudentInformation is called and prefixes the course information. |
This function produces a formatted string of the student's course information. |
This function produces a formatted string of the student\'s course information. |
Each column of data represents all the problems for a given sequence. For |
Each column of data represents all the problems for a given sequence. For |
valid grade data, a link is created for that problem to a submission record |
valid grade data, a link is created for that problem to a submission record |
for that problem. |
for that problem. |
Line 276 database.
|
Line 303 database.
|
|
|
Output: $Str |
Output: $Str |
|
|
$Str: Formatted string that is an entire row of the chart. It is a |
$Str: Formatted string that is an entire row of the chart. It is a |
concatenation of student information and student course information. |
concatenation of student information and student course information. |
|
|
=back |
=back |
Line 318 sub StudentReport {
|
Line 345 sub StudentReport {
|
} |
} |
$hasVersion = 'true'; |
$hasVersion = 'true'; |
$hasData = 'true'; |
$hasData = 'true'; |
$Str .= '<a href="/adm/grades?symb='; |
if (lc($ENV{'form.displaymode'}) ne 'display without links') { |
$Str .= &Apache::lonnet::escape($problem); |
$Str .= '<a href="/adm/grades?symb='; |
$Str .= '&student='.$username.'&domain='.$domain; |
$Str .= &Apache::lonnet::escape($problem); |
$Str .= '&command=submission">'; |
$Str .= '&student='.$username.'&domain='.$domain; |
|
$Str .= '&command=submission">'; |
|
} |
my $code = $cache->{$name.':'.$problemID.':'.$_.':code'}; |
my $code = $cache->{$name.':'.$problemID.':'.$_.':code'}; |
my $tries = $cache->{$name.':'.$problemID.':'.$_.':tries'}; |
my $tries = $cache->{$name.':'.$problemID.':'.$_.':tries'}; |
if($code eq '*' && $tries < 10 && $tries ne '') { |
if($code eq '*' && $tries < 10 && $tries ne '') { |
$code = $tries; |
$code = $tries; |
} |
} |
$Str .= $code; |
$Str .= $code; |
$Str .= '</a>'; |
if (lc($ENV{'form.displaymode'}) ne 'display without links') { |
|
$Str .= '</a>'; |
|
} |
$characterCount++; |
$characterCount++; |
} |
} |
} |
} |
|
|
# Output the number of correct answers for the current sequence. |
# Output the number of correct answers for the current sequence. |
# This part takes up 6 character slots, but is formated right |
# This part takes up 6 character slots, but is formated right |
# justified. |
# justified. |
my $spacesNeeded=$cache->{$sequence.':columnWidth'}-$characterCount; |
my $spacesNeeded=$cache->{$sequence.':columnWidth'}-$characterCount; |
$spacesNeeded -= 3; |
$spacesNeeded -= 3; |
Line 344 sub StudentReport {
|
Line 375 sub StudentReport {
|
# ':problemsCorrect'}); |
# ':problemsCorrect'}); |
|
|
my $outputProblemsCorrect = sprintf("%2d/%2d", $cache->{$name.':'.$sequence. |
my $outputProblemsCorrect = sprintf("%2d/%2d", $cache->{$name.':'.$sequence. |
':problemsCorrect'}, |
':problemsCorrect'}, |
$characterCount); |
$characterCount); |
if($hasData eq 'true') { |
if($hasData eq 'true') { |
$Str .= '<font color="#007700">'.$outputProblemsCorrect.'</font>'; |
$Str .= '<font color="#007700">'.$outputProblemsCorrect.'</font>'; |
Line 374 sub StudentReport {
|
Line 405 sub StudentReport {
|
return $Str; |
return $Str; |
} |
} |
|
|
|
|
|
sub StudentAverageTotal { |
|
my ($cache, $students, $sequenceKeys)=@_; |
|
my $Str = "\n<b>Summary Tables:</b>\n"; |
|
my %Correct = (); |
|
my $ProblemsSolved = 0; |
|
my $TotalProblems = 0; |
|
my $StudentCount = 0; |
|
|
|
foreach my $name (@$students) { |
|
$StudentCount++; |
|
foreach my $sequence (@$sequenceKeys) { |
|
$Correct{$sequence} += |
|
$cache->{$name.':'.$sequence.':problemsCorrect'}; |
|
} |
|
$ProblemsSolved += $cache->{$name.':problemsSolved'}; |
|
$TotalProblems += $cache->{$name.':totalProblems'}; |
|
} |
|
if ($StudentCount) { |
|
$ProblemsSolved = sprintf( "%.2f", |
|
$ProblemsSolved/$StudentCount); |
|
$TotalProblems /= $StudentCount; |
|
} else { |
|
$ProblemsSolved = 0; |
|
$TotalProblems = 0; |
|
} |
|
|
|
$Str .= '<table border=2 cellspacing="1">'."\n"; |
|
$Str .= '<tr><td><b>Students Count</b></td><td><b>'. |
|
$StudentCount.'</b></td></tr>'."\n"; |
|
$Str .= '<tr><td><b>Total Problems</b></td><td><b>'. |
|
$TotalProblems.'</b></td></tr>'."\n"; |
|
$Str .= '<tr><td><b>Average Correct</b></td><td><b>'. |
|
$ProblemsSolved.'</b></td></tr>'."\n"; |
|
$Str .= '</table>'."\n"; |
|
|
|
$Str .= '<table border=2 cellspacing="1">'."\n"; |
|
$Str .= '<tr><th>Title</th><th>Total Problems</th>'. |
|
'<th>Average Correct</th></tr>'."\n"; |
|
foreach my $S(@$sequenceKeys) { |
|
my $title=$cache->{$S.':title'}; |
|
#$Str .= $cache->{$S.':problems'}; |
|
#my @problems=split(':', $cache->{$S.':problems'}); |
|
#my $pCount=scalar @problems; |
|
my $pCount=MaxSeqPr($cache,@$students[0],$S); |
|
my $crr; |
|
if ($StudentCount) { |
|
$crr=sprintf( "%.2f", $Correct{$S}/$StudentCount ); |
|
} else { |
|
$crr="0.00"; |
|
} |
|
$Str .= '<tr><td>'.$title. |
|
'</td><td align=center>'.$pCount. |
|
'</td><td align=center>'.$crr. |
|
'</td></tr>'."\n"; |
|
} |
|
|
|
$Str .= '</table>'."\n"; |
|
|
|
return $Str; |
|
} |
|
|
|
|
|
|
|
sub MaxSeqPr { |
|
my ($cache, $name, $sequence)=@_; |
|
my $prCount=0; |
|
foreach my $problemID (split(':', $cache->{$sequence.':problems'})) { |
|
my $problem = $cache->{$problemID.':problem'}; |
|
foreach(split(/\:/,$cache->{$sequence.':'.$problemID.':parts'})) { |
|
if($cache->{$name.':'.$problemID.':NoVersion'} eq 'true' || |
|
$cache->{$name.':'.$problemID.':'.$_.':code'} eq ' ' || |
|
$cache->{$name.':'.$problemID.':'.$_.':code'} eq '') { |
|
$prCount++; |
|
next; |
|
} |
|
$prCount++; |
|
} |
|
} |
|
return $prCount; |
|
} |
|
|
|
|
|
|
|
|
|
|
=pod |
=pod |
|
|
=item &CreateLegend() |
=item &CreateLegend() |
Line 389 sub CreateLegend {
|
Line 506 sub CreateLegend {
|
" 1 correct by student in 1 try\n". |
" 1 correct by student in 1 try\n". |
" 7 correct by student in 7 tries\n". |
" 7 correct by student in 7 tries\n". |
" * correct by student in more than 9 tries\n". |
" * correct by student in more than 9 tries\n". |
" + correct by override\n". |
" + correct by hand grading or override\n". |
" - incorrect by override\n". |
" - incorrect by override\n". |
" . incorrect attempted\n". |
" . incorrect attempted\n". |
" # ungraded attempted\n". |
" # ungraded attempted\n". |
" not attempted (blank field)\n". |
" not attempted (blank field)\n". |
" x excused". |
" x excused". |
"</pre><p>"; |
"</pre><p>"; |
return $Str; |
return $Str; |
} |
} |
|
|
Line 404 sub CreateLegend {
|
Line 521 sub CreateLegend {
|
=item &CreateColumnSelectionBox() |
=item &CreateColumnSelectionBox() |
|
|
If there are columns not being displayed then this selection box is created |
If there are columns not being displayed then this selection box is created |
with a list of those columns. When selections are made and the page |
with a list of those columns. When selections are made and the page |
refreshed, the columns will be removed from this box and the column is |
refreshed, the columns will be removed from this box and the column is |
put back in the chart. If there is no columns to select, no row is added |
put back in the chart. If there is no columns to select, no row is added |
to the interface table. |
to the interface table. |
Line 415 Input: $CacheData, $headings
|
Line 532 Input: $CacheData, $headings
|
|
|
$CacheData: A pointer to a hash tied to the cached data |
$CacheData: A pointer to a hash tied to the cached data |
|
|
$headings: An array of the names of the columns for the student information. |
$headings: An array of the names of the columns for the student information. |
They are used for displaying which columns are missing. |
They are used for displaying which columns are missing. |
|
|
Output: $notThere |
Output: $notThere |
|
|
$notThere: The string contains one row of a table. The first column has the |
$notThere: The string contains one row of a table. The first column has the |
name of the selection box. The second contains the selection box |
name of the selection box. The second contains the selection box |
which has a size of four. |
which has a size of four. |
|
|
=back |
=back |
Line 451 sub CreateColumnSelectionBox {
|
Line 568 sub CreateColumnSelectionBox {
|
|
|
=item &CreateColumnSelectors() |
=item &CreateColumnSelectors() |
|
|
This function generates the checkboxes above the column headings. The |
This function generates the checkboxes above the column headings. The |
column will be removed if the checkbox is unchecked. |
column will be removed if the checkbox is unchecked. |
|
|
=over 4 |
=over 4 |
Line 501 sub CreateColumnSelectors {
|
Line 618 sub CreateColumnSelectors {
|
sub FindSelectedStudent { |
sub FindSelectedStudent { |
my($cache, $selectedName, $students)=@_; |
my($cache, $selectedName, $students)=@_; |
|
|
if($selectedName eq 'All Students' || |
if($selectedName eq 'All Students' || |
$selectedName eq 'No Student Selected') { |
$selectedName eq 'No Student Selected') { |
return $selectedName; |
return $selectedName; |
} |
} |
Line 585 sub ShouldShowColumns {
|
Line 702 sub ShouldShowColumns {
|
|
|
foreach my $sequence (split(/\:/,$cache->{'orderedSequences'})) { |
foreach my $sequence (split(/\:/,$cache->{'orderedSequences'})) { |
$check = 'SequenceColumn'.$sequence; |
$check = 'SequenceColumn'.$sequence; |
if($cache->{'SequencesFound'} eq 'All Sequences' || |
if($cache->{'SequencesFound'} eq 'All Sequences' || |
$cache->{'SequencesFound'} =~ /$check/) { |
$cache->{'SequencesFound'} =~ /$check/) { |
push(@sequenceHeadings, $cache->{$sequence.':title'}); |
push(@sequenceHeadings, $cache->{$sequence.':title'}); |
push(@sequenceKeys, $sequence); |
push(@sequenceKeys, $sequence); |
Line 598 sub ShouldShowColumns {
|
Line 715 sub ShouldShowColumns {
|
|
|
$doNotShow{'count'} = $count; |
$doNotShow{'count'} = $count; |
|
|
return (\@infoHeadings, \@infoKeys, \@sequenceHeadings, |
return (\@infoHeadings, \@infoKeys, \@sequenceHeadings, |
\@sequenceKeys, \%doNotShow); |
\@sequenceKeys, \%doNotShow); |
} |
} |
|
|