version 1.13, 2002/05/19 15:47:25
|
version 1.26, 2003/10/16 19:54:42
|
Line 25
|
Line 25
|
# http://www.lon-capa.org/ |
# http://www.lon-capa.org/ |
# |
# |
# The LearningOnline Network with CAPA |
# The LearningOnline Network with CAPA |
# Behrouz Minaei |
# |
# YEAR=2001 |
|
# 9/13/01, 9/25/01, 10/6/01, 10/9/01, 12/25/01 |
|
# YEAR=2002 |
|
# 2/1/, 5/13, 5/15 |
|
# A CGI script that dynamically outputs a graphical chart for lonstatistics. |
# A CGI script that dynamically outputs a graphical chart for lonstatistics. |
# |
# |
#### |
#### |
|
|
|
=pod |
|
|
|
=head1 NAME |
|
|
|
graph.png |
|
|
|
=head1 SYNOPSIS |
|
|
|
produces plots based on input |
|
|
|
=head1 DESCRIPTION |
|
|
|
graph.png is a cgi-bin script which produces plots based on input data. |
|
|
|
The query string is expected to be as follows (without whitespace): |
|
|
|
escape(Plot title) & escape(X label)& escape(Y label) & Maximum Y value & |
|
Number of bars & $data1 & $data2 |
|
|
|
$data1 and $data2 are expected to be comma seperated lists of numbers. |
|
escape( value ) means the values must be run through lonnet::escape. |
|
|
|
=cut |
|
|
use strict; |
use strict; |
|
use lib '/home/httpd/lib/perl'; |
use GD::Graph::bars; |
use GD::Graph::bars; |
use GD::Graph::colour; |
use GD::Graph::colour; |
use GD::Graph::Data; |
use GD::Graph::Data; |
|
use LONCAPA::loncgi(); |
|
|
$|=1; # Autoflush after each print/write |
sub unescape { |
my ($Titr,$xlab,$ylab,$Max,$PNo,$data1,$data2)=split(/&/,$ENV{'QUERY_STRING'}); |
my $str=shift; |
|
$str =~ s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg; |
|
return $str; |
|
} |
|
|
|
if (! &LONCAPA::loncgi::check_cookie_and_load_env()) { |
|
print <<END; |
|
Content-type: text/html |
|
|
|
<html> |
|
<head><title>Bad Cookie</title></head> |
|
<body> |
|
Your cookie information is incorrect. What\'s up with that? |
|
</body> |
|
</html> |
|
END |
|
exit; |
|
} |
|
|
my @data11=split(/\,/,$data1); |
$|=1; # Autoflush after each print/write |
my @data12=split(/\,/,$data2); |
my $identifier = $ENV{'QUERY_STRING'}; |
|
my $Title = &unescape($ENV{$identifier.'.title'}); |
|
my $xlabel = &unescape($ENV{$identifier.'.xlabel'}); |
|
my $ylabel = &unescape($ENV{$identifier.'.ylabel'}); |
|
my $Max = $ENV{$identifier.'.Max'}; |
|
my $NumBars = $ENV{$identifier.'.NumBars'}; |
|
my $NumSets = $ENV{$identifier.'.NumSets'}; |
|
my @Colors = split(',',$ENV{$identifier.'.Colors'}); |
|
|
|
# |
|
# Labels are always digits |
my @xlabels; |
my @xlabels; |
for (my $nIdx=0; $nIdx<$PNo; $nIdx++ ) { |
for (my $nIdx=0; $nIdx<$NumBars; $nIdx++ ) { |
$xlabels[$nIdx]=$nIdx+1; |
$xlabels[$nIdx]=$nIdx+1; |
} |
} |
|
my @data; # stores the data for the graph |
|
push(@data,\@xlabels); |
|
for (my $i=1;$i<=$NumSets;$i++) { |
|
push(@data,[split(',',$ENV{$identifier.'.data.'.$i})]); |
|
} |
|
|
|
my $skip_x = 1; |
|
my $bar_width=10; |
|
|
my @data =(\@xlabels,\@data11,\@data12); |
# |
|
# Customize graph based on the |
|
my $width; |
|
my $height = 200; |
|
|
|
if ($NumBars < 10) { |
|
$width = 120+$NumBars*15; |
|
$skip_x = 1; |
|
$bar_width = 15; |
|
} elsif ($NumBars <= 25) { |
|
$width = 120+$NumBars*11; |
|
$skip_x = 5; |
|
$bar_width = 8; |
|
} elsif ($NumBars <= 50) { |
|
$width = 120+$NumBars*8; |
|
$skip_x = 5; |
|
$bar_width = 4; |
|
} else { |
|
$width = 120+$NumBars*8; |
|
$skip_x = 5; |
|
$bar_width = 4; |
|
} |
|
|
my $Range1; |
my $x_tick_offset = 0; |
my $Range2; |
if ($skip_x > 1) { |
|
$x_tick_offset = $skip_x - 1; |
|
} |
|
|
if ($xlab=~/^Concepts$/){ |
my $MyGraph = GD::Graph::bars->new($width,$height); |
$Range1=270; |
my $error = ''; |
$Range2=200; |
if (! $MyGraph->set( x_label => $xlabel, |
} else { |
y_label => $ylabel, |
if ( $PNo > 10 ) { |
x_label_position => 0.5, |
$Range1 = 20*$PNo; |
long_ticks => 1, |
} else { |
tick_length => 0, |
$Range1 = 250+30*$PNo; |
x_ticks => 0, |
} |
title => $Title, |
$Range2=200; |
y_max_value => $Max, |
|
x_label_skip => $skip_x, |
|
x_tick_offset => $x_tick_offset, |
|
# |
|
dclrs => \@Colors, |
|
bar_width => $bar_width, |
|
cumulate => 2, |
|
zero_axis => 1, |
|
fgclr => 'black', |
|
boxclr => 'white', |
|
accentclr => 'dblue', |
|
valuesclr => '#ffff77', |
|
l_margin => 10, |
|
b_margin => 10, |
|
r_margin => 10, |
|
t_margin => 10, |
|
# |
|
transparent => 0, |
|
)) { |
|
$error = $MyGraph->error; |
|
print <<"END"; |
|
Content-type: text/html |
|
|
|
<html> |
|
<head><title>Bad Graph</title></head> |
|
<body> |
|
<p> |
|
There was an error producing the graph you requested. |
|
</p><p> |
|
$error |
|
</p> |
|
</body> |
|
</html> |
|
END |
|
return; |
} |
} |
|
|
|
my $plot = $MyGraph->plot(\@data); |
|
if (! defined($plot)) { |
|
print <<"END"; |
|
Content-type: text/html |
|
|
|
<html> |
|
<head><title>Bad Graph</title></head> |
|
<body> |
|
The system was unable to create the graph you requested. |
|
</body> |
|
</html> |
|
END |
|
return; |
|
} |
|
|
my $MyGraph = GD::Graph::bars->new($Range1,$Range2); |
my $BinaryData=$plot->png; |
|
undef($MyGraph); |
$MyGraph->set( |
undef($plot); |
# x_label => $xlab, |
|
y_label => $ylab, |
if (! defined($BinaryData)) { |
long_ticks => 1, |
print <<"END"; |
tick_length => 0, |
Content-type: text/html |
x_ticks => 0, |
|
# title => 'LON-CAPA Option Response Problem:'.$cid, |
<html> |
title => $Titr, |
<head><title>Bad Graph</title></head> |
y_max_value => $Max, |
<body> |
# y_tick_number => $ytic, |
The system was unable to produce a png image of the graph you requested. |
y_label_skip => 1, |
</body> |
|
</html> |
dclrs => [ qw( lgreen dgreen lyellow lpurple cyan lorange)], |
END |
|
return; |
bar_spacing => 10, |
} |
cumulate => 2, |
|
zero_axis => 1, |
|
|
|
# legend_placement => 'RT', |
|
|
|
fgclr => 'black', |
|
boxclr => 'white', |
|
accentclr => 'dblue', |
|
valuesclr => '#ffff77', |
|
l_margin => 10, |
|
b_margin => 10, |
|
r_margin => 10, |
|
t_margin => 10, |
|
|
|
transparent => 0, |
|
) or warn $MyGraph->error; |
|
#if ($xlab=~/^Concepts$/){ |
|
# $MyGraph->set_legend( 'Correct Answers', 'Incorrect Answers'); |
|
#} |
|
|
|
|
|
# Tell the server we are sending a gif graphic |
# Tell the server we are sending a png graphic |
print <<END; |
print <<END; |
Content-type: image/gif |
Content-type: image/png |
|
|
END |
END |
|
|
#$MyGraph->set_y_label_font('/home/httpd/cgi-bin/cetus.ttf', 16); |
|
#$MyGraph->set_x_label_font('/home/httpd/cgi-bin/cetus.ttf', 16); |
|
#$MyGraph->set_y_axis_font('/home/httpd/cgi-bin/cetus.ttf', 12); |
|
#$MyGraph->set_x_axis_font('/home/httpd/cgi-bin/cetus.ttf', 12); |
|
#$MyGraph->set_title_font('/home/httpd/cgi-bin/cetus.ttf', 18); |
|
#$MyGraph->set_legend_font('/home/httpd/cgi-bin/cetus.ttf', 10); |
|
#$MyGraph->set_values_font('/home/httpd/cgi-bin/cetus.ttf', 10); |
|
|
|
|
|
my $BinaryData=$MyGraph->plot(\@data)->png; |
|
undef $MyGraph; |
|
binmode(STDOUT); |
binmode(STDOUT); |
open IMG,"|pngtopnm|ppmtogif 2>/dev/null"; # convert into a gif image |
#open IMG,"|pngtopnm|ppmtogif 2>/dev/null"; # convert into a gif image |
print IMG $BinaryData; # output image |
#print IMG $BinaryData; # output image |
$|=1; # be sure to flush before closing |
#$|=1; # be sure to flush before closing |
close IMG; |
#close IMG; |
|
print $BinaryData; |