version 1.43, 2002/06/05 05:05:38
|
version 1.44, 2002/06/28 18:06:14
|
Line 55 use Apache::loncommon();
|
Line 55 use Apache::loncommon();
|
use HTML::TokeParser; |
use HTML::TokeParser; |
use GDBM_File; |
use GDBM_File; |
|
|
# -------------------------------------------------------------- Module Globals |
#my $jr; |
my %hash; |
# ----- FORMAT PRINT DATA ---------------------------------------------- |
my %CachData; |
|
my @cols; |
|
my $r; |
|
my $c; |
|
|
|
# ------------------------------------------------------------- Find out status |
|
|
|
sub ExtractStudentData { |
sub FormatStudentInformation { |
my ($name,$coid)=@_; |
my ($cache,$name,$studentInformation,$spacePadding)=@_; |
|
my $Str='<pre>'; |
|
|
|
foreach (@$studentInformation) { |
|
my $data=$cache->{$name.':'.$_}; |
|
$Str .= $data; |
|
|
|
my @dataLength=split(//,$data); |
|
my $length=scalar @dataLength; |
|
$Str .= (' 'x($cache->{$_.'Length'}-$length)); |
|
$Str .= $spacePadding; |
|
} |
|
|
|
return $Str; |
|
} |
|
|
|
sub FormatStudentData { |
|
my ($name,$coid,$studentInformation,$spacePadding,$ChartDB)=@_; |
my ($sname,$sdom) = split(/\:/,$name); |
my ($sname,$sdom) = split(/\:/,$name); |
my $ResId; |
|
my $Code; |
|
my $Tries; |
|
my $Wrongs; |
|
my %TempHash; |
|
my $Version; |
|
my $problemsCorrect; |
|
my $problemsSolved; |
|
my $totalProblems; |
|
my $LatestVersion; |
|
my $Str; |
my $Str; |
|
my %CacheData; |
|
|
|
unless(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_READER,0640)) { |
|
return ''; |
|
} |
# Handle Student information ------------------------------------------ |
# Handle Student information ------------------------------------------ |
|
# Handle user data |
|
$Str=&FormatStudentInformation(\%CacheData, $name, $studentInformation, |
|
$spacePadding); |
|
|
# Handle errors |
# Handle errors |
# if($CachData{$name.':error'} =~ /environment/) { |
if($CacheData{$name.':error'} =~ /environment/) { |
# my $errorMessage = $CachData{$name.':error'}; |
untie(%CacheData); |
|
$Str .= '</pre>'; |
|
return $Str; |
|
# my $errorMessage = $CacheData{$name.':error'}; |
# return '<td>'.$sname.'</td><td>'.$sdom. |
# return '<td>'.$sname.'</td><td>'.$sdom. |
# '</td><td><font color="#000088">'.$errorMessage.'</font></td>'; |
# '</td><td><font color="#000088">'.$errorMessage.'</font></td>'; |
# } |
} |
|
|
# Handle user data |
|
$Str = '<td><pre>'.$sname.'</pre></td><td><pre>'.$sdom; |
|
$Str .= '</pre></td><td><pre>'.$CachData{$name.':section'}; |
|
$Str .= '</pre></td><td><pre>'.$CachData{$name.':id'}; |
|
$Str .= '</pre></td><td><pre>'.$CachData{$name.':fullname'}; |
|
$Str .= '</pre></td>'; |
|
|
|
if($CachData{$name.':error'} =~ /course/) { |
if($CacheData{$name.':error'} =~ /course/) { |
|
untie(%CacheData); |
|
$Str .= '</pre>'; |
return $Str; |
return $Str; |
# my $errorMessage = 'May have no course data or '. |
# my $errorMessage = 'May have no course data or '. |
# $CachData{$name.':error'}; |
# $CacheData{$name.':error'}; |
# return '<td>'.$sname.'</td><td>'.$sdom. |
# return '<td>'.$sname.'</td><td>'.$sdom. |
# '</td><td><font color="#000088">'.$errorMessage.'</font></td>'; |
# '</td><td><font color="#000088">'.$errorMessage.'</font></td>'; |
} |
} |
|
|
# Handle problem data ------------------------------------------------ |
# Handle problem data ------------------------------------------------ |
$Str .= '<td><pre>'; |
my $Version; |
$problemsCorrect = 0; |
my $problemsCorrect = 0; |
$totalProblems = 0; |
my $totalProblems = 0; |
$problemsSolved = 0; |
my $problemsSolved = 0; |
my $IterationNo = 0; |
my $numberOfParts = 0; |
foreach $ResId (@cols) { |
foreach my $sequence (split(/\:/,$CacheData{'orderedSequences'})) { |
if ($IterationNo == 0) { |
my $characterCount=0; |
# Looks to be skipping start resource |
foreach my $problemID (split(/\:/,$CacheData{$sequence.':problems'})) { |
$IterationNo++; |
my $problem = $CacheData{$problemID.':problem'}; |
next; |
my $LatestVersion = $CacheData{$name.":version:$problem"}; |
} |
|
|
if(!$LatestVersion) { |
# ResId is 0 for sequences and pages, |
foreach my $part (split(/\:/,$CacheData{$sequence.':'. |
# please check tracetable for changes |
$problemID. |
if (!$ResId) { |
':parts'})) { |
my $outputProblemsCorrect = sprintf( "%3d", $problemsCorrect ); |
$Str .= ' '; |
$Str .= '<font color="#007700">'.$outputProblemsCorrect. |
$totalProblems++; |
'</font></pre></td>'; |
$characterCount++; |
$Str .= '<td><pre>'; |
} |
$problemsSolved += $problemsCorrect; |
next; |
$problemsCorrect=0; |
} |
next; |
|
} |
my %partData=undef; |
|
#initialize data, displays skips correctly |
# Set $1 and $2 |
foreach my $part (split(/\:/,$CacheData{$sequence.':'. |
$ResId=~/(\d+)\.(\d+)/; |
$problemID. |
my $meta=$hash{'src_'.$ResId}; |
':parts'})) { |
my $numberOfParts = 0; |
$partData{$part.':tries'}=0; |
undef %TempHash; |
$partData{$part.':code'}=' '; |
foreach (split(/\,/,&Apache::lonnet::metadata($meta,'keys'))) { |
} |
#----------- Overwrite $1 in next statement --------------------------------- |
for(my $Version=1; $Version<=$LatestVersion; $Version++) { |
if ($_=~/^stores\_(\d+)\_tries$/) { |
foreach my $part (split(/\:/,$CacheData{$sequence.':'. |
my $Part=&Apache::lonnet::metadata($meta,$_.'.part'); |
$problemID. |
if ( $TempHash{"$Part"} eq '' ) { |
':parts'})) { |
$TempHash{"$Part"} = $Part; |
|
$TempHash{$numberOfParts}=$Part; |
if(!defined($CacheData{$name.":$Version:$problem". |
$TempHash{"$Part.Code"} = ' '; |
":resource.$part.solved"})) { |
$numberOfParts++; |
next; |
} |
} |
} |
|
} |
my $tries=0; |
|
my $code=' '; |
#----------- Using $1 and $2 ----------------------------------------------- |
|
my $Prob = &Apache::lonnet::symbclean( |
$tries = $CacheData{$name.":$Version:$problem". |
&Apache::lonnet::declutter($hash{'map_id_'.$1} ). |
":resource.$part.tries"}; |
'___'.$2.'___'. |
$partData{$part.':tries'}=($tries) ? $tries : 0; |
&Apache::lonnet::declutter( $hash{'src_'.$ResId} )); |
|
$Code=' '; |
my $val = $CacheData{$name.":$Version:$problem". |
$Tries = 0; |
":resource.$part.solved"}; |
$LatestVersion = $CachData{$name.":version:$Prob"}; |
if ($val eq 'correct_by_student') {$code = '*';} |
|
elsif ($val eq 'correct_by_override') {$code = '+';} |
if ( $LatestVersion ) { |
elsif ($val eq 'incorrect_attempted') {$code = '.';} |
for ( my $Version=1; $Version<=$LatestVersion; $Version++ ) { |
elsif ($val eq 'incorrect_by_override'){$code = '-';} |
my $vkeys = $CachData{$name.":$Version:keys:$Prob"}; |
elsif ($val eq 'excused') {$code = 'x';} |
my @keys = split(/\:/,$vkeys); |
elsif ($val eq 'ungraded_attempted') {$code = '#';} |
|
else {$code = ' ';} |
foreach my $Key (@keys) { |
$partData{$part.':code'}=$code; |
#---------------------- Changing $1 ------------------------------------------- |
} |
if (($Key=~/\.(\w+)\.solved$/) && ($Key!~/^\d+\:/)) { |
} |
#---------------------- Using $1 ----------------------------------------------- |
|
my $Part = $1; |
|
$Tries = $CachData{$name.":$Version:$Prob". |
|
":resource.$Part.tries"}; |
|
$TempHash{"$Part.Tries"}=($Tries) ? $Tries : 0; |
|
my $Val = $CachData{$name.":$Version:$Prob". |
|
":resource.$Part.solved"}; |
|
if ($Val eq 'correct_by_student') {$Code = '*';} |
|
elsif ($Val eq 'correct_by_override') {$Code = '+';} |
|
elsif ($Val eq 'incorrect_attempted') {$Code = '.';} |
|
elsif ($Val eq 'incorrect_by_override'){$Code = '-';} |
|
elsif ($Val eq 'excused') {$Code = 'x';} |
|
elsif ($Val eq 'ungraded_attempted') {$Code = '#';} |
|
else {$Code = ' ';} |
|
|
|
$TempHash{"$Part.Code"} = $Code; |
$Str.='<a href="/adm/grades?symb='. |
} |
&Apache::lonnet::escape($problem). |
} |
|
} |
|
# Actually append problem to output (all parts) |
|
$Str.='<a href="/adm/grades?symb='. |
|
&Apache::lonnet::escape($Prob). |
|
'&student='.$sname.'&domain='.$sdom.'&command=submission">'; |
'&student='.$sname.'&domain='.$sdom.'&command=submission">'; |
for(my $n = 0; $n < $numberOfParts; $n++) { |
foreach(split(/\:/,$CacheData{$sequence.':'.$problemID. |
my $part = $TempHash{$n}; |
':parts'})) { |
my $code2 = $TempHash{"$part.Code"}; |
if($partData{$_.':code'} eq '*') { |
if($code2 eq '*') { |
$problemsCorrect++; |
$problemsCorrect++; |
if (($partData{$_.':tries'}<10) && |
# !!!!!!!!!!!------------------------- Should 10 not be maxtries? ------------ |
($partData{$_.':tries'} ne '')) { |
if (($TempHash{"$part.Tries"}<10) || |
$partData{$_.':code'}=$partData{$_.':tries'}; |
($TempHash{"$part.Tries"} eq '')) { |
} |
$TempHash{"$part.Code"}=$TempHash{"$part.Tries"}; |
} elsif($partData{$_.':code'} eq '+') { |
} |
$problemsCorrect++; |
} elsif($code2 eq '+') { |
} |
$problemsCorrect++; |
|
} |
$Str .= $partData{$_.':code'}; |
|
$characterCount++; |
|
|
|
if($partData{$_.':code'} ne 'x') { |
|
$totalProblems++; |
|
} |
|
} |
|
$Str.='</a>'; |
|
} |
|
|
|
my $spacesNeeded=$CacheData{$sequence.':columnWidth'}-$characterCount; |
|
$spacesNeeded -= 3; |
|
$Str .= (' 'x$spacesNeeded); |
|
|
|
my $outputProblemsCorrect = sprintf( "%3d", $problemsCorrect ); |
|
$Str .= '<font color="#007700">'.$outputProblemsCorrect.'</font>'; |
|
$problemsSolved += $problemsCorrect; |
|
$problemsCorrect=0; |
|
|
|
$Str .= $spacePadding; |
|
} |
|
|
$Str .= $TempHash{"$part.Code"}; |
$Str .= '<font color="#000088">'.$problemsSolved. |
|
' / '.$totalProblems.'</font></pre>'; |
|
|
if($code2 ne 'x') { |
untie(%CacheData); |
$totalProblems++; |
return $Str; |
} |
} |
} |
|
$Str.='</a>'; |
sub CreateTableHeadings { |
} else { |
my ($CacheData,$studentInformation,$headings,$spacePadding)=@_; |
for(my $n=0; $n<$numberOfParts; $n++) { |
my $Str='<pre>'; |
$Str.=' '; |
|
$totalProblems++; |
for(my $index=0; $index<(scalar @$headings); $index++) { |
} |
my $data=$$headings[$index]; |
} |
$Str .= $data; |
|
|
|
my @dataLength=split(//,$data); |
|
my $length=scalar @dataLength; |
|
$Str .= (' 'x($CacheData->{$$studentInformation[$index].'Length'}- |
|
$length)); |
|
$Str .= $spacePadding; |
|
} |
|
|
|
foreach my $sequence (split(/\:/,$CacheData->{'orderedSequences'})) { |
|
$Str .= $CacheData->{$sequence.':title'}; |
|
my @titleLength=split(//,$CacheData->{$sequence.':title'}); |
|
my $leftover=$CacheData->{$sequence.':columnWidth'}- |
|
(scalar @titleLength); |
|
$Str .= (' 'x$leftover); |
|
$Str .= $spacePadding; |
} |
} |
|
|
$Str .= '<td><pre><font color="#000088">'.$problemsSolved. |
$Str .= 'Total Solved/Total Problems'; |
' / '.$totalProblems.'</font></pre></td>'; |
$Str .= '</pre>'; |
|
|
return $Str; |
return $Str; |
} |
} |
Line 229 sub CreateForm {
|
Line 253 sub CreateForm {
|
elsif ($Status eq 'Expired' ) { $OpSel2 = 'selected'; } |
elsif ($Status eq 'Expired' ) { $OpSel2 = 'selected'; } |
else { $OpSel1 = 'selected'; } |
else { $OpSel1 = 'selected'; } |
|
|
my $Ptr = '<form name=stat method=post action="/adm/chart" >'."\n"; |
my $Ptr = '<form name="stat" method="post" action="/adm/chart" >'."\n"; |
$Ptr .= '<b> Sort by: </b>'."\n"; |
$Ptr .= '<b> Sort by: </b>'."\n"; |
$Ptr .= ' '; |
$Ptr .= ' '; |
$Ptr .= '<input type=submit name=sort value="User Name" />'."\n"; |
$Ptr .= '<input type="submit" name="sort" value="User Name" />'."\n"; |
$Ptr .= ' '; |
$Ptr .= ' '; |
$Ptr .= '<input type=submit name=sort value="Last Name" />'."\n"; |
$Ptr .= '<input type="submit" name="sort" value="Last Name" />'."\n"; |
$Ptr .= ' '; |
$Ptr .= ' '; |
$Ptr .= '<input type=submit name=sort value="Section"/>'."\n"; |
$Ptr .= '<input type="submit" name="sort" value="Section"/>'."\n"; |
$Ptr .= '<br><br>'; |
$Ptr .= '<br><br>'; |
$Ptr .= '<b> Student Status: </b>'."\n". |
$Ptr .= '<b> Student Status: </b>'."\n". |
'<select name="status">'. |
'<select name="status">'. |
Line 244 sub CreateForm {
|
Line 268 sub CreateForm {
|
'<option '.$OpSel2.' >Expired</option>'."\n". |
'<option '.$OpSel2.' >Expired</option>'."\n". |
'<option '.$OpSel3.' >Any</option> </select> '."\n"; |
'<option '.$OpSel3.' >Any</option> </select> '."\n"; |
$Ptr .= ' '; |
$Ptr .= ' '; |
$Ptr .= '<input type=submit name=sort value="Recalculate Chart"/>'."\n"; |
$Ptr .= '<input type="submit" name="sort" value="Recalculate Chart"/>'; |
|
$Ptr .= "\n"; |
$Ptr .= '</form>'."\n"; |
$Ptr .= '</form>'."\n"; |
$r->print( $Ptr ); |
|
|
return $Ptr; |
} |
} |
|
|
sub CreateTableHeadings { |
sub CreateLegend { |
$r->print('<tr>'); |
my $Str = '<h1>'.$ENV{'course.'.$ENV{'request.course.id'}.'.description'}. |
$r->print('<td>User Name</td>'); |
'</h1><h3>'.localtime(). |
$r->print('<td>Domain</td>'); |
"</h3><p><pre>1..9: correct by student in 1..9 tries\n". |
$r->print('<td>Section</td>'); |
" *: correct by student in more than 9 tries\n". |
$r->print('<td>PID</td>'); |
" +: correct by override\n". |
$r->print('<td>Full Name</td>'); |
" -: incorrect by override\n". |
|
" .: incorrect attempted\n". |
my $ResId; |
" #: ungraded attempted\n". |
my $IterationNo = 0; |
" : not attempted\n". |
foreach $ResId (@cols) { |
" x: excused</pre><p>"; |
if ($IterationNo == 0) {$IterationNo++; next;} |
return $Str; |
if (!$ResId) { |
} |
# my $PrNo = sprintf( "%3d", $ProbNo ); |
|
# $Str .= '<td><font color="#007700">Chapter '.$PrNo.'</font></td>'; |
|
$r->print('<td><font color="#007700">Chapter '.'0'.'</font></td>'); |
|
} |
|
} |
|
|
|
$r->print('</tr>'); |
sub StartDocument { |
$r->rflush(); |
my $Str = ''; |
|
$Str .= '<html>'; |
|
$Str .= '<head><title>'; |
|
$Str .= 'LON-CAPA Assessment Chart</title></head>'; |
|
$Str .= '<body bgcolor="#FFFFFF">'; |
|
$Str .= '<script>window.focus();</script>'; |
|
$Str .= '<img align=right src=/adm/lonIcons/lonlogos.gif>'; |
|
$Str .= '<h1>Assessment Chart</h1>'; |
|
|
return; |
return $Str; |
} |
} |
|
|
# ------------------------------------------------------------ Build page table |
# ----- END FORMAT PRINT DATA ------------------------------------------ |
|
|
|
# ----- DOWNLOAD INFORMATION ------------------------------------------- |
|
|
|
sub DownloadPrerequisiteData { |
|
my ($courseID, $c)=@_; |
|
my ($courseDomain,$courseNumber)=split(/\_/,$courseID); |
|
|
|
my %classlist=&Apache::lonnet::dump('classlist',$courseDomain, |
|
$courseNumber); |
|
my ($checkForError)=keys (%classlist); |
|
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
|
return \%classlist; |
|
} |
|
|
|
foreach my $name (keys(%classlist)) { |
|
if($c->aborted()) { |
|
$classlist{'error'}='aborted'; |
|
return \%classlist; |
|
} |
|
|
|
my ($studentName,$studentDomain) = split(/\:/,$name); |
|
# Download student environment data, specifically the full name and id. |
|
my %studentInformation=&Apache::lonnet::get('environment', |
|
['lastname','generation', |
|
'firstname','middlename', |
|
'id'], |
|
$studentDomain, |
|
$studentName); |
|
$classlist{$name.':studentInformation'}=\%studentInformation; |
|
|
sub tracetable { |
if($c->aborted()) { |
my ($rid,$beenhere)=@_; |
$classlist{'error'}='aborted'; |
unless ($beenhere=~/\&$rid\&/) { |
return \%classlist; |
$beenhere.=$rid.'&'; |
} |
# new ... updating the map according to sequence and page |
|
if (defined($hash{'is_map_'.$rid})) { |
#Section |
my $cmap=$hash{'map_type_'.$hash{'map_pc_'.$hash{'src_'.$rid}}}; |
my %section=&Apache::lonnet::dump('roles',$studentDomain,$studentName); |
if ( $cmap eq 'sequence' || $cmap eq 'page' ) { |
$classlist{$name.':section'}=\%section; |
$cols[$#cols+1]=0; |
|
} |
|
if ((defined($hash{'map_start_'.$hash{'src_'.$rid}})) && |
|
(defined($hash{'map_finish_'.$hash{'src_'.$rid}}))) { |
|
my $frid=$hash{'map_finish_'.$hash{'src_'.$rid}}; |
|
|
|
&tracetable($hash{'map_start_'.$hash{'src_'.$rid}}, |
|
'&'.$frid.'&'); |
|
|
|
if ($hash{'src_'.$frid}) { |
|
if ($hash{'src_'.$frid}=~ |
|
/\.(problem|exam|quiz|assess|survey|form)$/) { |
|
$cols[$#cols+1]=$frid; |
|
} |
|
} |
|
|
|
} |
|
} else { |
|
if ($hash{'src_'.$rid}) { |
|
if ($hash{'src_'.$rid}=~ |
|
/\.(problem|exam|quiz|assess|survey|form)$/) { |
|
$cols[$#cols+1]=$rid; |
|
} |
|
} |
|
} |
|
if (defined($hash{'to_'.$rid})) { |
|
foreach (split(/\,/,$hash{'to_'.$rid})){ |
|
&tracetable($hash{'goesto_'.$_},$beenhere); |
|
} |
|
} |
|
} |
} |
|
|
|
return \%classlist; |
} |
} |
|
|
sub usection { |
sub DownloadStudentCourseInformation { |
my ($udom,$unam,$courseid,$ActiveFlag)=@_; |
my ($name,$courseID)=@_; |
$courseid=~s/\_/\//g; |
my ($studentName,$studentDomain) = split(/\:/,$name); |
$courseid=~s/^(\w)/\/$1/; |
|
|
|
my %result=&Apache::lonnet::dump('roles',$udom,$unam); |
# Download student course data |
|
my %courseData=&Apache::lonnet::dump($courseID,$studentDomain, |
|
$studentName); |
|
return \%courseData; |
|
} |
|
|
my($checkForError)=keys (%result); |
# ----- END DOWNLOAD INFORMATION --------------------------------------- |
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
|
return -1; |
# ----- END PROCESSING FUNCTIONS --------------------------------------- |
|
|
|
sub ProcessTopResourceMap { |
|
my ($ChartDB,$c)=@_; |
|
my %hash; |
|
my $fn=$ENV{'request.course.fn'}; |
|
if(-e "$fn.db") { |
|
my $tieTries=0; |
|
while($tieTries < 3) { |
|
if(tie(%hash,'GDBM_File',"$fn.db",&GDBM_READER,0640)) { |
|
last; |
|
} |
|
$tieTries++; |
|
sleep 1; |
|
} |
|
if($tieTries >= 3) { |
|
return 'Coursemap undefined.'; |
|
} |
|
} else { |
|
return 'Can not open Coursemap.'; |
} |
} |
|
|
|
my %CacheData; |
|
unless(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_WRCREAT,0640)) { |
|
untie(%hash); |
|
return 'Could not tie cache hash.'; |
|
} |
|
|
|
my (@sequences, @currentResource, @finishResource); |
|
my ($currentSequence, $currentResourceID, $lastResourceID); |
|
|
|
$currentResourceID=$hash{'ids_/res/'.$ENV{'request.course.uri'}}; |
|
$lastResourceID=-1; |
|
$currentSequence=-1; |
|
my $topLevelSequenceNumber = $currentSequence; |
|
|
|
while(1) { |
|
if($c->aborted()) { |
|
last; |
|
} |
|
# HANDLE NEW SEQUENCE! |
|
#if page || sequence |
|
if(defined($hash{'map_pc_'.$hash{'src_'.$currentResourceID}})) { |
|
push(@sequences, $currentSequence); |
|
push(@currentResource, $currentResourceID); |
|
push(@finishResource, $lastResourceID); |
|
|
|
$currentSequence=$hash{'map_pc_'.$hash{'src_'.$currentResourceID}}; |
|
$lastResourceID=$hash{'map_finish_'. |
|
$hash{'src_'.$currentResourceID}}; |
|
$currentResourceID=$hash{'map_start_'. |
|
$hash{'src_'.$currentResourceID}}; |
|
|
|
if(!($currentResourceID) || !($lastResourceID)) { |
|
$currentSequence=pop(@sequences); |
|
$currentResourceID=pop(@currentResource); |
|
$lastResourceID=pop(@finishResource); |
|
if($currentSequence eq $topLevelSequenceNumber) { |
|
last; |
|
} |
|
} |
|
} |
|
|
|
# Handle gradable resources: exams, problems, etc |
|
$currentResourceID=~/(\d+)\.(\d+)/; |
|
my $partA=$1; |
|
my $partB=$2; |
|
if($hash{'src_'.$currentResourceID}=~ |
|
/\.(problem|exam|quiz|assess|survey|form)$/ && |
|
$partA eq $currentSequence) { |
|
my $Problem = &Apache::lonnet::symbclean( |
|
&Apache::lonnet::declutter($hash{'map_id_'.$partA}). |
|
'___'.$partB.'___'. |
|
&Apache::lonnet::declutter($hash{'src_'. |
|
$currentResourceID})); |
|
|
|
$CacheData{$currentResourceID.':problem'}=$Problem; |
|
if(!defined($CacheData{$currentSequence.':problems'})) { |
|
$CacheData{$currentSequence.':problems'}=$currentResourceID; |
|
} else { |
|
$CacheData{$currentSequence.':problems'}.= |
|
':'.$currentResourceID; |
|
} |
|
|
|
#Get Parts for problem |
|
my $meta=$hash{'src_'.$currentResourceID}; |
|
foreach (split(/\,/,&Apache::lonnet::metadata($meta,'keys'))) { |
|
if($_=~/^stores\_(\d+)\_tries$/) { |
|
my $Part=&Apache::lonnet::metadata($meta,$_.'.part'); |
|
if(!defined($CacheData{$currentSequence.':'. |
|
$currentResourceID.':parts'})) { |
|
$CacheData{$currentSequence.':'.$currentResourceID. |
|
':parts'}=$Part; |
|
} else { |
|
$CacheData{$currentSequence.':'.$currentResourceID. |
|
':parts'}.=':'.$Part; |
|
} |
|
} |
|
} |
|
} |
|
|
|
#if resource == finish resource |
|
if($currentResourceID eq $lastResourceID) { |
|
#pop off last resource of sequence |
|
$currentResourceID=pop(@currentResource); |
|
#pop to get last resource in previous sequence |
|
$currentResourceID=pop(@currentResource); |
|
$lastResourceID=pop(@finishResource); |
|
|
|
if(defined($CacheData{$currentSequence.':problems'})) { |
|
# Capture sequence information here |
|
if(!defined($CacheData{'orderedSequences'})) { |
|
$CacheData{'orderedSequences'}=$currentSequence; |
|
} else { |
|
$CacheData{'orderedSequences'}.=':'.$currentSequence; |
|
} |
|
|
|
$CacheData{$currentSequence.':title'}= |
|
$hash{'title_'.$currentResourceID}; |
|
|
|
my $totalProblems=0; |
|
foreach (split(/\:/,$CacheData{$currentSequence. |
|
':problems'})) { |
|
foreach ($CacheData{$currentSequence.':'.$_.':parts'}) { |
|
$totalProblems++; |
|
} |
|
} |
|
my @titleLength=split(//,$CacheData{$currentSequence. |
|
':title'}); |
|
# $extra is 3 for problems correct and 3 for space |
|
# between problems correct and problem output |
|
my $extra = 6; |
|
if(($totalProblems + $extra) > (scalar @titleLength)) { |
|
$CacheData{$currentSequence.':columnWidth'}= |
|
$totalProblems + $extra; |
|
} else { |
|
$CacheData{$currentSequence.':columnWidth'}= |
|
(scalar @titleLength); |
|
} |
|
} |
|
|
|
$currentSequence=pop(@sequences); |
|
if($currentSequence eq $topLevelSequenceNumber) { |
|
last; |
|
} |
|
#else |
|
} |
|
|
|
# MOVE!!! |
|
#move to next resource |
|
unless(defined($hash{'to_'.$currentResourceID})) { |
|
# big problem, need to handle. Next is probably wrong |
|
last; |
|
} |
|
my @nextResources=(); |
|
foreach (split(/\,/,$hash{'to_'.$currentResourceID})) { |
|
push(@nextResources, $hash{'goesto_'.$_}); |
|
} |
|
pop(@currentResource); |
|
push(@currentResource, @nextResources); |
|
# Set the next resource to be popped(processed) |
|
$currentResourceID=$currentResource[-1]; |
|
} |
|
|
|
unless (untie(%hash)) { |
|
&Apache::lonnet::logthis("<font color=blue>WARNING: ". |
|
"Could not untie coursemap $fn (browse)". |
|
".</font>"); |
|
} |
|
|
|
unless (untie(%CacheData)) { |
|
&Apache::lonnet::logthis("<font color=blue>WARNING: ". |
|
"Could not untie Cache Hash (browse)". |
|
".</font>"); |
|
} |
|
|
|
return 'OK'; |
|
} |
|
|
|
sub ProcessSection { |
|
my ($sectionData, $courseid,$ActiveFlag)=@_; |
|
$courseid=~s/\_/\//g; |
|
$courseid=~s/^(\w)/\/$1/; |
|
|
my $cursection='-1'; |
my $cursection='-1'; |
my $oldsection='-1'; |
my $oldsection='-1'; |
my $status='Expired'; |
my $status='Expired'; |
foreach my $key (keys (%result)) { |
my $section=''; |
my $value = $result{$key}; |
foreach my $key (keys (%$sectionData)) { |
|
my $value = $sectionData->{$key}; |
if ($key=~/^$courseid(?:\/)*(\w+)*\_st$/) { |
if ($key=~/^$courseid(?:\/)*(\w+)*\_st$/) { |
my $section=$1; |
$section=$1; |
if ($key eq $courseid.'_st') { $section=''; } |
if($key eq $courseid.'_st') { |
|
$section=''; |
|
} |
my ($dummy,$end,$start)=split(/\_/,$value); |
my ($dummy,$end,$start)=split(/\_/,$value); |
my $now=time; |
my $now=time; |
my $notactive=0; |
my $notactive=0; |
Line 353 sub usection {
|
Line 569 sub usection {
|
if($notactive == 0) { |
if($notactive == 0) { |
$status='Active'; |
$status='Active'; |
$cursection=$section; |
$cursection=$section; |
|
last; |
} |
} |
if($notactive == 1) { |
if($notactive == 1) { |
$oldsection=$section; |
$oldsection=$section; |
Line 374 sub usection {
|
Line 591 sub usection {
|
return '-1'; |
return '-1'; |
} |
} |
|
|
|
sub ProcessStudentInformation { |
|
my ($CacheData,$studentInformation,$section,$date,$name,$courseID,$c)=@_; |
|
my ($studentName,$studentDomain) = split(/\:/,$name); |
|
|
|
$CacheData->{$name.':username'}=$studentName; |
|
$CacheData->{$name.':domain'}=$studentDomain; |
|
$CacheData->{$name.':date'}=$date; |
|
|
|
my ($checkForError)=keys(%$studentInformation); |
|
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
|
$CacheData->{$name.':error'}= |
|
'Could not download student environment data.'; |
|
$CacheData->{$name.':fullname'}=''; |
|
$CacheData->{$name.':id'}=''; |
|
} else { |
|
$CacheData->{$name.':fullname'}=&ProcessFullName( |
|
$studentInformation->{'lastname'}, |
|
$studentInformation->{'generation'}, |
|
$studentInformation->{'firstname'}, |
|
$studentInformation->{'middlename'}); |
|
$CacheData->{$name.':id'}=$studentInformation->{'id'}; |
|
} |
|
|
|
# Get student's section number |
|
my $sec=&ProcessSection($section, $courseID, $ENV{'form.status'}); |
|
if($sec != -1) { |
|
$CacheData->{$name.':section'}=$sec; |
|
} else { |
|
$CacheData->{$name.':section'}=''; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
sub ProcessClassList { |
|
my ($classlist,$courseID,$ChartDB,$c)=@_; |
|
my @names=(); |
|
|
|
my %CacheData; |
|
if(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_WRCREAT,0640)) { |
|
foreach my $name (keys(%$classlist)) { |
|
if($name =~ /\:section/ || $name =~ /\:studentInformation/) { |
|
next; |
|
} |
|
if($c->aborted()) { |
|
last; |
|
} |
|
push(@names,$name); |
|
&ProcessStudentInformation( |
|
\%CacheData, |
|
$classlist->{$name.':studentInformation'}, |
|
$classlist->{$name.':section'}, |
|
$classlist->{$name}, |
|
$name,$courseID,$c); |
|
} |
|
|
|
$CacheData{'NamesOfStudents'}=join(":::",@names); |
|
# $CacheData{'NamesOfStudents'}=&Apache::lonnet::arrayref2str(\@names); |
|
untie(%CacheData); |
|
} |
|
|
|
return @names; |
|
} |
|
|
|
# ----- END PROCESSING FUNCTIONS --------------------------------------- |
|
|
|
# ----- HELPER FUNCTIONS ----------------------------------------------- |
|
|
|
sub SpaceColumns { |
|
my ($students,$studentInformation,$headings,$ChartDB)=@_; |
|
|
|
my %CacheData; |
|
if(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_WRCREAT,0640)) { |
|
# Initialize Lengths |
|
for(my $index=0; $index<(scalar @$headings); $index++) { |
|
my @titleLength=split(//,$$headings[$index]); |
|
$CacheData{$$studentInformation[$index].'Length'}= |
|
scalar @titleLength; |
|
} |
|
|
|
foreach my $name (@$students) { |
|
foreach (@$studentInformation) { |
|
my @dataLength=split(//,$CacheData{$name.':'.$_}); |
|
my $length=scalar @dataLength; |
|
if($length > $CacheData{$_.'Length'}) { |
|
$CacheData{$_.'Length'}=$length; |
|
} |
|
} |
|
} |
|
untie(%CacheData); |
|
} |
|
|
|
return; |
|
} |
|
|
sub ProcessFullName { |
sub ProcessFullName { |
my ($name)=@_; |
my ($lastname, $generation, $firstname, $middlename)=@_; |
my $Str = ''; |
my $Str = ''; |
|
|
if($CachData{$name.':lastname'} ne '') { |
if($lastname ne '') { |
$Str .= $CachData{$name.':lastname'}.' '; |
$Str .= $lastname.' '; |
if($CachData{$name.':generation'} ne '') { |
if($generation ne '') { |
$Str .= $CachData{$name.':generation'}; |
$Str .= $generation; |
} else { |
} else { |
chop($Str); |
chop($Str); |
} |
} |
$Str .= ', '; |
$Str .= ', '; |
if($CachData{$name.':firstname'} ne '') { |
if($firstname ne '') { |
$Str .= $CachData{$name.':firstname'}.' '; |
$Str .= $firstname.' '; |
} |
} |
if($CachData{$name.':middlename'} ne '') { |
if($middlename ne '') { |
$Str .= $CachData{$name.':middlename'}; |
$Str .= $middlename; |
} else { |
} else { |
chop($Str); |
chop($Str); |
if($CachData{$name.'firstname'} eq '') { |
if($firstname eq '') { |
chop($Str); |
chop($Str); |
} |
} |
} |
} |
} else { |
} else { |
if($CachData{$name.':firstname'} ne '') { |
if($firstname ne '') { |
$Str .= $CachData{$name.':firstname'}.' '; |
$Str .= $firstname.' '; |
} |
} |
if($CachData{$name.':middlename'} ne '') { |
if($middlename ne '') { |
$Str .= $CachData{$name.':middlename'}.' '; |
$Str .= $middlename.' '; |
} |
} |
if($CachData{$name.':generation'} ne '') { |
if($generation ne '') { |
$Str .= $CachData{$name.':generation'}; |
$Str .= $generation; |
} else { |
} else { |
chop($Str); |
chop($Str); |
} |
} |
Line 414 sub ProcessFullName {
|
Line 726 sub ProcessFullName {
|
return $Str; |
return $Str; |
} |
} |
|
|
sub DownloadStudentInformation { |
sub SortStudents { |
my ($name,$courseID)=@_; |
my ($CacheData)=@_; |
my ($studentName,$studentDomain) = split(/\:/,$name); |
my @students = split(/:::/,$CacheData->{'NamesOfStudents'}); |
my $checkForError; |
# my @students=&Apache::lonnet::str2array($CacheData->{'NamesOfStudents'}); |
my $key; |
|
my $Status=$CachData{$name.':Status'}; |
my @sorted1Students=(); |
|
foreach (@students) { |
#----------------------------------------------------------------- |
my ($end,$start)=split(/\:/,$CacheData->{$_.':date'}); |
# Download student environment data, specifically the full name and id. |
my $active=1; |
my %studentInformation=&Apache::lonnet::get('environment', |
my $now=time; |
['lastname','generation', |
my $Status=$ENV{'form.status'}; |
'firstname','middlename', |
$Status = ($Status) ? $Status : 'Active'; |
'id'], |
if((($end) && $now > $end) && (($Status eq 'Active'))) { |
$studentDomain,$studentName); |
$active=0; |
if($c->aborted()) { |
} |
return; |
if(($Status eq 'Expired') && ($end == 0 || $now < $end)) { |
} |
$active=0; |
($checkForError)=keys (%studentInformation); |
} |
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
if($active) { |
$CachData{$name.':error'}= |
push(@sorted1Students, $_); |
'Could not download student environment data.'; |
} |
# return; |
|
$CachData{$name.':lastname'}=''; |
|
$CachData{$name.':generation'}=''; |
|
$CachData{$name.':firstname'}=''; |
|
$CachData{$name.':middlename'}=''; |
|
$CachData{$name.':fullname'}=''; |
|
$CachData{$name.':id'}=''; |
|
} else { |
|
$CachData{$name.':lastname'}=$studentInformation{'lastname'}; |
|
$CachData{$name.':generation'}=$studentInformation{'generation'}; |
|
$CachData{$name.':firstname'}=$studentInformation{'firstname'}; |
|
$CachData{$name.':middlename'}=$studentInformation{'middlename'}; |
|
$CachData{$name.':fullname'}=&ProcessFullName($name); |
|
$CachData{$name.':id'}=$studentInformation{'id'}; |
|
} |
|
|
|
# Download student course data |
|
my %courseData=&Apache::lonnet::dump($courseID,$studentDomain, |
|
$studentName); |
|
if($c->aborted()) { |
|
return; |
|
} |
|
($checkForError)=keys (%courseData); |
|
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
|
$CachData{$name.':error'}='Could not download course data.'; |
|
# return; |
|
} else { |
|
foreach $key (keys (%courseData)) { |
|
$CachData{$name.':'.$key}=$courseData{$key}; |
|
} |
|
} |
|
|
|
# Get student's section number |
|
my $sec=&usection($studentDomain, $studentName, $courseID, $Status); |
|
if($sec != -1) { |
|
$CachData{$name.':section'}=sprintf('%3s',$sec); |
|
} else { |
|
$CachData{$name.':section'}=''; |
|
} |
} |
|
|
return; |
|
} |
|
|
|
sub SortStudents { |
|
# --------------------------------------------------------------- Sort Students |
|
my $Pos = $ENV{'form.sort'}; |
my $Pos = $ENV{'form.sort'}; |
my @students = split(/:::/,$CachData{'NamesOfStudents'}); |
|
my %sortData; |
my %sortData; |
|
|
if($Pos eq 'Last Name') { |
if($Pos eq 'Last Name') { |
for(my $index=0; $index<$#students+1; $index++) { |
for(my $index=0; $index<scalar @sorted1Students; $index++) { |
$sortData{$CachData{$students[$index].':fullname'}}= |
$sortData{$CacheData->{$sorted1Students[$index].':fullname'}}= |
$students[$index]; |
$sorted1Students[$index]; |
} |
} |
} elsif($Pos eq 'Section') { |
} elsif($Pos eq 'Section') { |
for(my $index=0; $index<$#students+1; $index++) { |
for(my $index=0; $index<scalar @sorted1Students; $index++) { |
$sortData{$CachData{$students[$index].':section'}. |
$sortData{$CacheData->{$sorted1Students[$index].':section'}. |
$students[$index]}=$students[$index]; |
$sorted1Students[$index]}=$sorted1Students[$index]; |
} |
} |
} else { |
} else { |
# Sort by user name |
# Sort by user name |
for(my $index=0; $index<$#students+1; $index++) { |
for(my $index=0; $index<scalar @sorted1Students; $index++) { |
$sortData{$students[$index]}=$students[$index]; |
$sortData{$sorted1Students[$index]}=$sorted1Students[$index]; |
} |
} |
} |
} |
|
|
Line 509 sub SortStudents {
|
Line 776 sub SortStudents {
|
return @order; |
return @order; |
} |
} |
|
|
sub CollectClasslist { |
sub TestCacheData { |
# -------------------------------------------------------------- Get class list |
my ($ChartDB)=@_; |
my $cid=$ENV{'request.course.id'}; |
my $isCached=-1; |
my $chome=$ENV{'course.'.$cid.'.home'}; |
my %testData; |
my ($cdom,$cnum)=split(/\_/,$cid); |
my $tieTries=0; |
my %classlist=&Apache::lonnet::dump('classlist',$cdom,$cnum); |
|
my @names = (); |
|
|
|
my($checkForError)=keys (%classlist); |
if ((-e "$ChartDB") && ($ENV{'form.sort'} ne 'Recalculate Chart')) { |
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
$isCached = 1; |
$r->print('<h1>Could not access course data</h1>'); |
} else { |
push (@names, 'error'); |
$isCached = 0; |
return @names; |
|
} |
} |
|
|
# ------------------------------------- Calculate Status and number of students |
while($tieTries < 3) { |
my $now=time; |
my $result=0; |
foreach my $name (sort(keys(%classlist))) { |
if($isCached) { |
my $value=$classlist{$name}; |
$result=tie(%testData,'GDBM_File',$ChartDB,&GDBM_READER,0640); |
my ($end,$start)=split(/\:/,$value); |
} else { |
my $active=1; |
$result=tie(%testData,'GDBM_File',$ChartDB,&GDBM_NEWDB,0640); |
my $Status=$ENV{'form.status'}; |
} |
$Status = ($Status) ? $Status : 'Active'; |
if($result) { |
if((($end) && $now > $end) && (($Status eq 'Active'))) { |
last; |
$active=0; |
} |
} |
$tieTries++; |
if(($Status eq 'Expired') && ($end == 0 || $now < $end)) { |
sleep 1; |
$active=0; |
} |
} |
if($tieTries >= 3) { |
if($active) { |
return -1; |
push(@names,$name); |
|
$CachData{$name.':Status'}=$Status; |
|
} |
|
} |
} |
|
|
$CachData{'NamesOfStudents'}=join(":::",@names); |
untie(%testData); |
|
|
return @names; |
return $isCached; |
} |
} |
|
|
sub BuildChart { |
sub ExtractStudentData { |
# ----------------------- Get first and last resource, see if there is anything |
my ($courseData, $name, $ChartDB)=@_; |
my $firstres=$hash{'map_start_/res/'.$ENV{'request.course.uri'}}; |
|
my $lastres=$hash{'map_finish_/res/'.$ENV{'request.course.uri'}}; |
my %CacheData; |
if (!($firstres) || !($lastres)) { |
if(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_WRCREAT,0640)) { |
$r->print('<h3>Undefined course sequence</h3>'); |
my ($checkForError) = keys(%$courseData); |
return; |
if($checkForError =~ /^(con_lost|error|no_such_host)/i) { |
|
$CacheData{$name.':error'}='Could not download course data.'; |
|
} else { |
|
foreach my $key (keys (%$courseData)) { |
|
$CacheData{$name.':'.$key}=$courseData->{$key}; |
|
} |
|
} |
|
untie(%CacheData); |
} |
} |
|
|
# --------------- Find all assessments and put them into some linear-like order |
return; |
&tracetable($firstres,'&'.$lastres.'&'); |
} |
|
|
|
# ----- END HELPER FUNCTIONS -------------------------------------------- |
|
|
|
sub BuildChart { |
|
my ($r)=@_; |
|
my $c = $r->connection; |
|
|
# ----------------------------------------------------------------- Render page |
# Start the lonchart document |
&CreateForm(); |
$r->content_type('text/html'); |
|
$r->send_http_header; |
|
$r->print(&StartDocument()); |
|
$r->rflush(); |
|
|
|
# Test for access to the CacheData |
|
my $isCached=0; |
my $cid=$ENV{'request.course.id'}; |
my $cid=$ENV{'request.course.id'}; |
my $ChartDB = "/home/httpd/perl/tmp/$ENV{'user.name'}". |
my $ChartDB = "/home/httpd/perl/tmp/$ENV{'user.name'}". |
"_$ENV{'user.domain'}_$cid\_chart.db"; |
"_$ENV{'user.domain'}_$cid\_chart.db"; |
my $isCached = 0; |
|
my @students; |
$isCached=&TestCacheData($ChartDB); |
if ((-e "$ChartDB") && ($ENV{'form.sort'} ne 'Recalculate Chart')) { |
if($isCached < 0) { |
if (tie(%CachData,'GDBM_File',"$ChartDB",&GDBM_READER,0640)) { |
$r->print("Unable to tie hash to db file"); |
$isCached = 1; |
$r->rflush(); |
@students=&SortStudents(); |
return; |
} else { |
} |
$r->print("Unable to tie hash to db file"); |
|
$r->rflush(); |
# Download class list information if not using cached data |
return; |
my @students=(); |
} |
my @studentInformation=('username','domain','section','id','fullname'); |
|
my @headings=('User Name','Domain','Section','PID','Full Name'); |
|
my $spacePadding=' '; |
|
if(!$isCached) { |
|
my $processTopResourceMapReturn=&ProcessTopResourceMap($ChartDB,$c); |
|
if($processTopResourceMapReturn ne 'OK') { |
|
$r->print($processTopResourceMapReturn); |
|
return; |
|
} |
|
if($c->aborted()) { return; } |
|
my $classlist=&DownloadPrerequisiteData($cid, $c); |
|
my ($checkForError)=keys(%$classlist); |
|
if($checkForError =~ /^(con_lost|error|no_such_host)/i || |
|
defined($classlist->{'error'})) { |
|
return; |
|
} |
|
if($c->aborted()) { return; } |
|
@students=&ProcessClassList($classlist,$cid,$ChartDB,$c); |
|
if($c->aborted()) { return; } |
|
&SpaceColumns(\@students,\@studentInformation,\@headings, |
|
$ChartDB); |
|
if($c->aborted()) { return; } |
|
} |
|
|
|
# Sort students and print out table desciptive data |
|
my %CacheData; |
|
if(tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_READER,0640)) { |
|
if(!$c->aborted()) { @students=&SortStudents(\%CacheData); } |
|
if(!$c->aborted()) { $r->print(&CreateLegend()); } |
|
if(!$c->aborted()) { $r->print(&CreateForm()); } |
|
if(!$c->aborted()) { $r->print('<h3>'.(scalar @students). |
|
' students</h3>'); } |
|
if(!$c->aborted()) { $r->rflush(); } |
|
if(!$c->aborted()) { $r->print(&CreateTableHeadings( |
|
\%CacheData, |
|
\@studentInformation, |
|
\@headings, |
|
$spacePadding)); } |
|
untie(%CacheData); |
} else { |
} else { |
if (tie(%CachData,'GDBM_File',$ChartDB,&GDBM_NEWDB,0640)) { |
$r->print("Init2: Unable to tie hash to db file"); |
$isCached = 0; |
return; |
@students=&CollectClasslist(); |
|
if($students[0] eq 'error') { |
|
return; |
|
} |
|
} else { |
|
$r->print("Unable to tie hash to db file"); |
|
return; |
|
} |
|
} |
} |
|
|
$r->print('<h3>'.($#students+1).' students</h3>'); |
|
$r->rflush(); |
|
|
|
# ----------------------------------------------------------------- Start table |
|
$r->print('<table><tbody>'); |
|
# &CreateTableHeadings(); |
|
my @updateStudentList = (); |
my @updateStudentList = (); |
foreach my $student (@students) { |
my $courseData; |
if($c->aborted()) { |
foreach (@students) { |
if($isCached == 0) { |
if($c->aborted()) { |
$CachData{'NamesOfStudents'}=join(":::",@updateStudentList); |
if(!$isCached && |
} |
tie(%CacheData,'GDBM_File',$ChartDB,&GDBM_WRCREAT,0640)) { |
last; |
$CacheData{'NamesOfStudents'}=join(":::", @updateStudentList); |
} |
# $CacheData{'NamesOfStudents'}= |
if($isCached == 0) { |
# &Apache::lonnet::arrayref2str(\@updateStudentList); |
&DownloadStudentInformation($student,$cid); |
untie(%CacheData); |
push (@updateStudentList, $student); |
} |
} |
last; |
my $Str=&ExtractStudentData($student,$cid); |
} |
$r->print('<tr>'.$Str.'</tr>'); |
|
|
if(!$isCached) { |
|
$courseData=&DownloadStudentCourseInformation($_, $cid); |
|
if($c->aborted()) { next; } |
|
push(@updateStudentList, $_); |
|
&ExtractStudentData($courseData, $_, $ChartDB); |
|
} |
|
$r->print(&FormatStudentData($_, $cid, \@studentInformation, |
|
$spacePadding, $ChartDB)); |
|
$r->rflush(); |
} |
} |
$r->print('</tbody></table>'); |
|
|
|
untie(%CachData); |
|
|
|
return; |
$r->print('</body></html>'); |
} |
|
|
|
sub Start { |
|
$r->print('<head><title>'. |
|
'LON-CAPA Assessment Chart</title></head>'); |
|
$r->print('<body bgcolor="#FFFFFF">'. |
|
'<script>window.focus();</script>'. |
|
'<img align=right src=/adm/lonIcons/lonlogos.gif>'. |
|
'<h1>Assessment Chart</h1>'); |
|
# ---------------------------------------------------------------- Course title |
|
$r->print('<h1>'.$ENV{'course.'.$ENV{'request.course.id'}. |
|
'.description'}.'</h1><h3>'.localtime(). |
|
"</h3><p><pre>1..9: correct by student in 1..9 tries\n". |
|
" *: correct by student in more than 9 tries\n". |
|
" +: correct by override\n". |
|
" -: incorrect by override\n". |
|
" .: incorrect attempted\n". |
|
" #: ungraded attempted\n". |
|
" : not attempted\n". |
|
" x: excused</pre><p>"); |
|
# ------------------------------- This is going to take a while, produce output |
|
$r->rflush(); |
$r->rflush(); |
|
|
&BuildChart(); |
|
|
|
$r->print('</body>'); |
|
|
|
return; |
return; |
} |
} |
|
|
# ================================================================ Main Handler |
# ================================================================ Main Handler |
|
|
sub handler { |
sub handler { |
undef %hash; |
my $r=shift; |
undef %CachData; |
# $jr=$r; |
undef @cols; |
unless(&Apache::lonnet::allowed('vgr',$ENV{'request.course.id'})) { |
|
|
$r=shift; |
|
$c = $r->connection; |
|
if (&Apache::lonnet::allowed('vgr',$ENV{'request.course.id'})) { |
|
# ------------------------------------------- Set document type for header only |
|
if ($r->header_only) { |
|
if ($ENV{'browser.mathml'}) { |
|
$r->content_type('text/xml'); |
|
} else { |
|
$r->content_type('text/html'); |
|
} |
|
&Apache::loncommon::no_cache($r); |
|
$r->send_http_header; |
|
return OK; |
|
} |
|
|
|
my $requrl=$r->uri; |
|
# ----------------------------------------------------------------- Tie db file |
|
if ($ENV{'request.course.fn'}) { |
|
my $fn=$ENV{'request.course.fn'}; |
|
if (-e "$fn.db") { |
|
if (tie(%hash,'GDBM_File',"$fn.db",&GDBM_READER,0640)) { |
|
# ------------------------------------------------------------------- Hash tied |
|
# ---------------------------------------------------------------- Send headers |
|
$r->content_type('text/html'); |
|
$r->send_http_header; |
|
$r->print('<html>'); |
|
&Start(); |
|
$r->print('</html>'); |
|
$r->rflush(); |
|
# ------------------------------------------------------------- End render page |
|
} else { |
|
$r->content_type('text/html'); |
|
$r->send_http_header; |
|
$r->print('<html><body>Coursemap undefined.</body></html>'); |
|
} |
|
# ------------------------------------------------------------------ Untie hash |
|
unless (untie(%hash)) { |
|
&Apache::lonnet::logthis("<font color=blue>WARNING: ". |
|
"Could not untie coursemap $fn (browse).</font>"); |
|
} |
|
|
|
# -------------------------------------------------------------------- All done |
|
return OK; |
|
# ----------------------------------------------- Errors, hash could no be tied |
|
} |
|
} else { |
|
$ENV{'user.error.msg'}="$requrl:bre:0:0:Course not initialized"; |
|
return HTTP_NOT_ACCEPTABLE; |
|
} |
|
} else { |
|
$ENV{'user.error.msg'}= |
$ENV{'user.error.msg'}= |
$r->uri.":vgr:0:0:Cannot view grades for complete course"; |
$r->uri.":vgr:0:0:Cannot view grades for complete course"; |
return HTTP_NOT_ACCEPTABLE; |
return HTTP_NOT_ACCEPTABLE; |
} |
} |
|
|
|
# Set document type for header only |
|
if ($r->header_only) { |
|
if($ENV{'browser.mathml'}) { |
|
$r->content_type('text/xml'); |
|
} else { |
|
$r->content_type('text/html'); |
|
} |
|
&Apache::loncommon::no_cache($r); |
|
$r->send_http_header; |
|
return OK; |
|
} |
|
|
|
unless($ENV{'request.course.fn'}) { |
|
my $requrl=$r->uri; |
|
$ENV{'user.error.msg'}="$requrl:bre:0:0:Course not initialized"; |
|
return HTTP_NOT_ACCEPTABLE; |
|
} |
|
|
|
&BuildChart($r); |
|
|
|
return OK; |
} |
} |
1; |
1; |
__END__ |
__END__ |