version 1.75, 2011/11/19 13:45:36
|
version 1.93, 2012/02/28 22:42:31
|
Line 32 use Apache::response();
|
Line 32 use Apache::response();
|
use Apache::lonlocal; |
use Apache::lonlocal; |
use Apache::lonnet; |
use Apache::lonnet; |
use Apache::run; |
use Apache::run; |
|
use LONCAPA; |
|
|
BEGIN { |
BEGIN { |
&Apache::lonxml::register('Apache::functionplotresponse',('functionplotresponse','backgroundplot','spline', |
&Apache::lonxml::register('Apache::functionplotresponse',('functionplotresponse','backgroundplot','spline', |
'plotobject','plotvector','functionplotvectorrule', |
'plotobject','plotvector', |
|
'functionplotvectorrule','functionplotvectorsumrule', |
|
'drawvectorsum', |
|
'functionplotcustomrule', |
'functionplotrule','functionplotruleset', |
'functionplotrule','functionplotruleset', |
'functionplotelements')); |
'functionplotelements')); |
} |
} |
Line 46 BEGIN {
|
Line 50 BEGIN {
|
# |
# |
|
|
sub geogebra_startcode { |
sub geogebra_startcode { |
my ($id)=@_; |
my ($id,$width,$height)=@_; |
|
$width=int(1.*$width); |
|
$height=int(1.*$height); |
|
unless ($width) { $width=700; } |
|
unless ($height) { $height=400; } |
return (<<ENDSTARTCODE); |
return (<<ENDSTARTCODE); |
<applet name="ggbApplet_$id" code="geogebra.GeoGebraApplet" archive="geogebra.jar" |
<applet name="ggbApplet_$id" code="geogebra.GeoGebraApplet" archive="geogebra.jar" |
codebase="/adm/geogebra/" width="722" height="447" MAYSCRIPT> |
codebase="/adm/geogebra/" width="$width" height="$height" MAYSCRIPT> |
<param name="java_arguments" value="-Xmx512m -Djnlp.packEnabled=true"/> |
<param name="java_arguments" value="-Xmx512m -Djnlp.packEnabled=true"/> |
ENDSTARTCODE |
ENDSTARTCODE |
} |
} |
Line 389 sub plotobject_script {
|
Line 397 sub plotobject_script {
|
$Apache::functionplotresponse::counter++; |
$Apache::functionplotresponse::counter++; |
$label='O'.$Apache::functionplotresponse::counter; |
$label='O'.$Apache::functionplotresponse::counter; |
} |
} |
|
&generate_input_field($id,$label,$x,$y); |
return "document.ggbApplet_$id.evalCommand('a=1');\n". |
return "document.ggbApplet_$id.evalCommand('a=1');\n". |
"document.ggbApplet_$id.setVisible('a', false);\n". |
"document.ggbApplet_$id.setVisible('a', false);\n". |
"document.ggbApplet_$id.setLabelVisible('a', false);\n". |
"document.ggbApplet_$id.setLabelVisible('a', false);\n". |
Line 402 sub plotobject_script {
|
Line 411 sub plotobject_script {
|
# |
# |
|
|
sub plotvector_script { |
sub plotvector_script { |
my ($id,$label,$xs,$ys,$xe,$ye,$xmin,$xmax)=@_; |
my ($id,$label,$xs,$ys,$xe,$ye,$xmin,$xmax,$fixed)=@_; |
unless ($label) { |
unless ($label) { |
$Apache::functionplotresponse::counter++; |
$Apache::functionplotresponse::counter++; |
$label='V'.$Apache::functionplotresponse::counter; |
$label='V'.$Apache::functionplotresponse::counter; |
Line 413 sub plotvector_script {
|
Line 422 sub plotvector_script {
|
my $pointx=2.*($xmax-$xmin)+$xmax; |
my $pointx=2.*($xmax-$xmin)+$xmax; |
my $anglelabel=$label.'Angle'; |
my $anglelabel=$label.'Angle'; |
return |
return |
&new_point_coordinate($id,$startlabel,$xs,$ys,0). |
&new_point_coordinate($id,$startlabel,$xs,$ys,$fixed). |
&new_point_coordinate($id,$endlabel,$xe,$ye,0). |
&new_point_coordinate($id,$endlabel,$xe,$ye,$fixed). |
(<<ENDVECTOR); |
(<<ENDVECTOR); |
document.ggbApplet_$id.evalCommand("$label=Vector[$startlabel,$endlabel]"); |
document.ggbApplet_$id.evalCommand("$label=Vector[$startlabel,$endlabel]"); |
document.ggbApplet_$id.setLabelVisible("$label",true); |
document.ggbApplet_$id.setLabelVisible("$label",true); |
Line 509 sub start_plotobject {
|
Line 518 sub start_plotobject {
|
$token,'8'). |
$token,'8'). |
&Apache::edit::end_row(); |
&Apache::edit::end_row(); |
} elsif ($target eq 'modified') { |
} elsif ($target eq 'modified') { |
$env{'form.'.&Apache::edit::html_element_name('label')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('label')}); |
$env{'form.'.&Apache::edit::html_element_name('label')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('label')}); |
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'label','x','y'); |
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'label','x','y'); |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
} |
} |
Line 548 sub start_plotvector {
|
Line 557 sub start_plotvector {
|
unless (defined($taily)) { $taily=$ymin; } |
unless (defined($taily)) { $taily=$ymin; } |
unless (defined($tipx)) { $tipx=$xmin; } |
unless (defined($tipx)) { $tipx=$xmin; } |
unless (defined($tipy)) { $tipy=$ymin; } |
unless (defined($tipy)) { $tipy=$ymin; } |
$result.=&plotvector_script($internalid,$label,$tailx,$taily,$tipx,$tipy,$xmin,$xmax); |
my $fixed=0; |
|
if ((&Apache::response::show_answer()) || (&Apache::response::check_status()>=2)) { $fixed=1; } |
|
$result.=&plotvector_script($internalid,$label,$tailx,$taily,$tipx,$tipy,$xmin,$xmax,$fixed); |
} elsif ($target eq 'edit') { |
} elsif ($target eq 'edit') { |
$result=&Apache::edit::tag_start($target,$token,'Plot Vector'). |
$result=&Apache::edit::tag_start($target,$token,'Plot Vector'). |
&Apache::edit::text_arg('Label on Plot:','label', |
&Apache::edit::text_arg('Label on Plot:','label', |
Line 564 sub start_plotvector {
|
Line 575 sub start_plotvector {
|
|
|
&Apache::edit::end_row(); |
&Apache::edit::end_row(); |
} elsif ($target eq 'modified') { |
} elsif ($target eq 'modified') { |
$env{'form.'.&Apache::edit::html_element_name('label')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('label')}); |
$env{'form.'.&Apache::edit::html_element_name('label')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('label')}); |
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'label','tailx','taily','tipx','tipy'); |
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'label','tailx','taily','tipx','tipy'); |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
} |
} |
Line 581 sub end_plotvector {
|
Line 592 sub end_plotvector {
|
} |
} |
|
|
|
|
|
# |
|
# Vector sum - have GeoGebra draw a sum of specified vectors to help students draw |
|
# |
|
|
|
sub start_drawvectorsum { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
my $internalid = $Apache::inputtags::part.'_'.$Apache::inputtags::response[-1]; |
|
my $tailx=&Apache::lonxml::get_param('tailx',$parstack,$safeeval); |
|
my $taily=&Apache::lonxml::get_param('taily',$parstack,$safeeval); |
|
my $vectorlist=&Apache::lonxml::get_param('vectorlist',$parstack,$safeeval); |
|
my $label=&Apache::lonxml::get_param('label',$parstack,$safeeval); |
|
$label=~s/\W//gs; |
|
$label=ucfirst($label); |
|
unless ($label) { $label="NewVector"; } |
|
if ($target eq 'web') { |
|
my ($xmin,$xmax,$ymin,$ymax)=&boundaries($parstack,$safeeval,-3); |
|
unless (defined($tailx)) { $tailx=$xmin; } |
|
unless (defined($taily)) { $taily=$ymin; } |
|
unless (defined($vectorlist)) { $vectorlist=''; } |
|
my @vectors=split(/\,/,$vectorlist); |
|
if ($#vectors>0) { |
|
my @sumx=(); |
|
my @sumy=(); |
|
foreach my $thisvector (@vectors) { |
|
$thisvector=~s/\W//gs; |
|
$thisvector=ucfirst($thisvector); |
|
unless ($thisvector) { next; } |
|
my $vectorx=$thisvector.'X'; |
|
my $vectory=$thisvector.'Y'; |
|
$result.=(<<ENDADDVEC); |
|
document.ggbApplet_$internalid.evalCommand("$vectorx=x($thisvector)"); |
|
document.ggbApplet_$internalid.evalCommand("$vectory=y($thisvector)"); |
|
document.ggbApplet_$internalid.evalCommand("Include$thisvector$label=Checkbox[]"); |
|
ENDADDVEC |
|
push(@sumx,"If[Include$thisvector$label,$vectorx,0]"); |
|
push(@sumy,"If[Include$thisvector$label,$vectory,0]"); |
|
} |
|
$result.="document.ggbApplet_$internalid.evalCommand(".'"'."xTot$label=".join('+',@sumx).'");'."\n"; |
|
$result.="document.ggbApplet_$internalid.evalCommand(".'"'."yTot$label=".join('+',@sumy).'");'."\n"; |
|
$result.=(<<ENDMAKEVECTOR); |
|
document.ggbApplet_$internalid.evalCommand("$label=Vector[($tailx,$taily),($tailx+xTot$label,$taily+yTot$label)]"); |
|
document.ggbApplet_$internalid.setLabelVisible("$label",true); |
|
document.ggbApplet_$internalid.setLineThickness("$label",8); |
|
document.ggbApplet_$internalid.setColor("$label",255,0,0); |
|
ENDMAKEVECTOR |
|
} |
|
} elsif ($target eq 'edit') { |
|
$result=&Apache::edit::tag_start($target,$token,'Draw Vector Sum'). |
|
&Apache::edit::text_arg('Label on Plot:','label', |
|
$token,'16'). |
|
&Apache::edit::text_arg('Tail x:','tailx', |
|
$token,'8'). |
|
&Apache::edit::text_arg('Tail y:','taily', |
|
$token,'8').'<br />'. |
|
&Apache::edit::text_arg('Vector List:','vectorlist', |
|
$token,'40'). |
|
&Apache::edit::end_row(); |
|
} elsif ($target eq 'modified') { |
|
$env{'form.'.&Apache::edit::html_element_name('label')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('label')}); |
|
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'label','tailx','taily','vectorlist'); |
|
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
|
} |
|
return $result; |
|
} |
|
|
|
|
|
sub end_drawvectorsum { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
if ($target eq 'edit') { |
|
$result=&Apache::edit::end_table(); |
|
} |
|
return $result; |
|
} |
|
|
|
|
|
|
# |
# |
# <backgroundplot function="..." fixed="yes/no" /> |
# <backgroundplot function="..." fixed="yes/no" /> |
Line 653 sub start_functionplotrule {
|
Line 741 sub start_functionplotrule {
|
} else { |
} else { |
$label='R'.$label; |
$label='R'.$label; |
} |
} |
if ($Apache::functionplotresponse::splineorder{$label}) { |
|
&Apache::lonxml::error(&mt('Rule indices must be unique.')); |
|
} |
|
|
|
|
|
if ($target eq 'grade') { |
if ($target eq 'grade') { |
# Simply remember - in order - for later |
# Simply remember - in order - for later |
Line 767 sub start_functionplotvectorrule {
|
Line 851 sub start_functionplotvectorrule {
|
} |
} |
$label=~s/\W//gs; |
$label=~s/\W//gs; |
unless ($label) { |
unless ($label) { |
$label='V'.$Apache::functionplotresponse::counter; |
$label='R'.$Apache::functionplotresponse::counter; |
} else { |
} else { |
$label='V'.$label; |
$label='R'.$label; |
} |
|
if ($Apache::functionplotresponse::splineorder{$label}) { |
|
&Apache::lonxml::error(&mt('Rule indices must be unique.')); |
|
} |
} |
|
|
if ($target eq 'grade') { |
if ($target eq 'grade') { |
# Simply remember - in order - for later |
# Simply remember - in order - for later |
|
|
|
my $id=$Apache::inputtags::response[-1]; |
|
my $partid=$Apache::inputtags::part; |
|
my $internalid = $partid.'_'.$id; |
|
|
my $vector=&Apache::lonxml::get_param('vector',$parstack,$safeeval); |
my $vector=&Apache::lonxml::get_param('vector',$parstack,$safeeval); |
$vector=~s/\W//gs; |
$vector=~s/\W//gs; |
$vector=ucfirst($vector); |
$vector=ucfirst($vector); |
my $tailpoint=&Apache::lonxml::get_param('tailpoint',$parstack,$safeeval); |
|
$tailpoint=~s/\W//gs; |
|
$tailpoint=ucfirst($tailpoint); |
|
my $tippoint=&Apache::lonxml::get_param('tippoint',$parstack,$safeeval); |
|
$tippoint=~s/\W//gs; |
|
$tippoint=ucfirst($tippoint); |
|
|
|
my $nottailpoint=&Apache::lonxml::get_param('nottailpoint',$parstack,$safeeval); |
|
$nottailpoint=~s/\W//gs; |
|
$nottailpoint=ucfirst($nottailpoint); |
|
my $nottippoint=&Apache::lonxml::get_param('nottippoint',$parstack,$safeeval); |
|
$nottippoint=~s/\W//gs; |
|
$nottippoint=ucfirst($nottippoint); |
|
|
|
push(@Apache::functionplotresponse::functionplotvectorrules,join(':',( |
push(@Apache::functionplotresponse::functionplotvectorrules,join(':',( |
$label, |
$label, |
|
'vector', |
|
$internalid, |
$vector, |
$vector, |
$tailpoint, |
&Apache::lonxml::get_param('attachpoint',$parstack,$safeeval), |
$tippoint, |
&Apache::lonxml::get_param('notattachpoint',$parstack,$safeeval), |
$nottailpoint, |
&Apache::lonxml::get_param('tailpoint',$parstack,$safeeval), |
$nottippoint, |
&Apache::lonxml::get_param('tippoint',$parstack,$safeeval), |
|
&Apache::lonxml::get_param('nottailpoint',$parstack,$safeeval), |
|
&Apache::lonxml::get_param('nottippoint',$parstack,$safeeval), |
&Apache::lonxml::get_param('length',$parstack,$safeeval), |
&Apache::lonxml::get_param('length',$parstack,$safeeval), |
&Apache::lonxml::get_param('angle',$parstack,$safeeval), |
&Apache::lonxml::get_param('angle',$parstack,$safeeval), |
&Apache::lonxml::get_param('lengthpercenterror',$parstack,$safeeval), |
&Apache::lonxml::get_param('lengtherror',$parstack,$safeeval), |
&Apache::lonxml::get_param('anglepercenterror',$parstack,$safeeval), |
&Apache::lonxml::get_param('angleerror',$parstack,$safeeval), |
))); |
))); |
} elsif ($target eq 'edit') { |
} elsif ($target eq 'edit') { |
$result=&Apache::edit::tag_start($target,$token,'Function Plot Vector Rule'). |
$result=&Apache::edit::tag_start($target,$token,'Function Plot Vector Rule'). |
Line 812 sub start_functionplotvectorrule {
|
Line 889 sub start_functionplotvectorrule {
|
$token,'10').' '. |
$token,'10').' '. |
&Apache::edit::text_arg('Vector:','vector', |
&Apache::edit::text_arg('Vector:','vector', |
$token,'16').'<br />'. |
$token,'16').'<br />'. |
|
&Apache::edit::text_arg('Attached to object:','attachpoint', |
|
$token,'16'). |
|
&Apache::edit::text_arg('Not attached to object:','notattachpoint', |
|
$token,'16').'<br />'. |
&Apache::edit::text_arg('Tail attached to object:','tailpoint', |
&Apache::edit::text_arg('Tail attached to object:','tailpoint', |
$token,'16'). |
$token,'16'). |
&Apache::edit::text_arg('Tip attached to object:','tippoint', |
&Apache::edit::text_arg('Tip attached to object:','tippoint', |
Line 821 sub start_functionplotvectorrule {
|
Line 902 sub start_functionplotvectorrule {
|
&Apache::edit::text_arg('Tip not attached to object:','nottippoint', |
&Apache::edit::text_arg('Tip not attached to object:','nottippoint', |
$token,'16').'<br />'. |
$token,'16').'<br />'. |
&Apache::edit::text_arg('Length:','length', |
&Apache::edit::text_arg('Length:','length', |
$token,'16'). |
$token,'30'). |
|
&Apache::edit::text_arg('Absolute error length:','lengtherror', |
|
$token,'8').'<br />'. |
&Apache::edit::text_arg('Angle:','angle', |
&Apache::edit::text_arg('Angle:','angle', |
$token,'16'). |
$token,'30'). |
&Apache::edit::text_arg('Percent error length:','lengthpercenterror', |
&Apache::edit::text_arg('Absolute error angle:','angleerror', |
$token,'8'). |
|
&Apache::edit::text_arg('Percent error angle:','anglepercenterror', |
|
$token,'8'). |
$token,'8'). |
&Apache::edit::end_row(); |
&Apache::edit::end_row(); |
} elsif ($target eq 'modified') { |
} elsif ($target eq 'modified') { |
$env{'form.'.&Apache::edit::html_element_name('vector')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('vector')}); |
$env{'form.'.&Apache::edit::html_element_name('vector')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('vector')}); |
$env{'form.'.&Apache::edit::html_element_name('tailpoint')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('tailpoint')}); |
|
$env{'form.'.&Apache::edit::html_element_name('tippoint')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('tippoint')}); |
|
$env{'form.'.&Apache::edit::html_element_name('nottailpoint')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('nottailpoint')}); |
|
$env{'form.'.&Apache::edit::html_element_name('nottippoint')}=lcfirst($env{'form.'.&Apache::edit::html_element_name('nottippoint')}); |
|
my $constructtag=&Apache::edit::get_new_args($token,$parstack, |
my $constructtag=&Apache::edit::get_new_args($token,$parstack, |
$safeeval,'index','vector', |
$safeeval,'index','vector','attachpoint','notattachpoint', |
'tailpoint','tippoint','nottailpoint','nottipoint', |
'tailpoint','tippoint','nottailpoint','nottipoint', |
'length','angle', |
'length','angle', |
'lengthpercenterror','anglepercenterror'); |
'lengtherror','angleerror'); |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
} |
} |
return $result; |
return $result; |
Line 854 sub end_functionplotvectorrule {
|
Line 931 sub end_functionplotvectorrule {
|
return $result; |
return $result; |
} |
} |
|
|
|
# |
|
# <functionplotvectorsumrule ... /> |
|
# |
|
sub start_functionplotvectorsumrule { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
my $label=&Apache::lonxml::get_param('index',$parstack,$safeeval); |
|
$Apache::functionplotresponse::counter++; |
|
if ($label=~/\W/) { |
|
&Apache::lonxml::warning(&mt('Rule indices should only contain alphanumeric characters.')); |
|
} |
|
$label=~s/\W//gs; |
|
unless ($label) { |
|
$label='R'.$Apache::functionplotresponse::counter; |
|
} else { |
|
$label='R'.$label; |
|
} |
|
if ($target eq 'grade') { |
|
# Simply remember - in order - for later |
|
my $id=$Apache::inputtags::response[-1]; |
|
my $partid=$Apache::inputtags::part; |
|
my $internalid = $partid.'_'.$id; |
|
my $vectors=&Apache::lonxml::get_param('vectors',$parstack,$safeeval); |
|
push(@Apache::functionplotresponse::functionplotvectorrules,join(':',( |
|
$label, |
|
'sum', |
|
$internalid, |
|
$vectors, |
|
&Apache::lonxml::get_param('length',$parstack,$safeeval), |
|
&Apache::lonxml::get_param('angle',$parstack,$safeeval), |
|
&Apache::lonxml::get_param('lengtherror',$parstack,$safeeval), |
|
&Apache::lonxml::get_param('angleerror',$parstack,$safeeval), |
|
))); |
|
} elsif ($target eq 'edit') { |
|
$result=&Apache::edit::tag_start($target,$token,'Function Plot Vector Sum Rule'). |
|
&Apache::edit::text_arg('Index/Name:','index', |
|
$token,'10').' '. |
|
&Apache::edit::text_arg('Comma-separated list of vectors:','vectors', |
|
$token,'30').'<br />'. |
|
&Apache::edit::text_arg('Sum vector length:','length', |
|
$token,'30'). |
|
&Apache::edit::text_arg('Absolute error length:','lengtherror', |
|
$token,'8').'<br />'. |
|
&Apache::edit::text_arg('Sum vector angle:','angle', |
|
$token,'30'). |
|
&Apache::edit::text_arg('Absolute error angle:','angleerror', |
|
$token,'8'). |
|
&Apache::edit::end_row(); |
|
} elsif ($target eq 'modified') { |
|
my $constructtag=&Apache::edit::get_new_args($token,$parstack, |
|
$safeeval,'index','vectors', |
|
'length','angle', |
|
'lengtherror','angleerror'); |
|
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
|
} |
|
return $result; |
|
} |
|
|
|
sub end_functionplotvectorsumrule { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
if ($target eq 'edit') { |
|
$result=&Apache::edit::end_table(); |
|
} |
|
return $result; |
|
} |
|
|
|
# |
|
# <functionplotcustom ... /> |
|
# |
|
sub start_functionplotcustomrule { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
my $label=&Apache::lonxml::get_param('index',$parstack,$safeeval); |
|
$Apache::functionplotresponse::counter++; |
|
if ($label=~/\W/) { |
|
&Apache::lonxml::warning(&mt('Rule indices should only contain alphanumeric characters.')); |
|
} |
|
$label=~s/\W//gs; |
|
unless ($label) { |
|
$label='R'.$Apache::functionplotresponse::counter; |
|
} else { |
|
$label='R'.$label; |
|
} |
|
&Apache::lonxml::register('Apache::response',('answer')); |
|
if ($target eq 'edit') { |
|
$result=&Apache::edit::tag_start($target,$token,'Function Plot Custom Rule'). |
|
&Apache::edit::text_arg('Index/Name:','index',$token,'10'). |
|
&Apache::edit::end_row(); |
|
} elsif ($target eq 'modified') { |
|
my $constructtag=&Apache::edit::get_new_args($token,$parstack,$safeeval,'index'); |
|
if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); } |
|
} |
|
return $result; |
|
} |
|
|
|
sub end_functionplotcustomrule { |
|
my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_; |
|
my $result=''; |
|
if ($target eq 'edit') { |
|
$result=&Apache::edit::end_table(); |
|
} elsif ($target eq 'grade') { |
|
# Simply remember - in order - for later |
|
my $label=&Apache::lonxml::get_param('index',$parstack,$safeeval); |
|
$Apache::functionplotresponse::counter++; |
|
if ($label=~/\W/) { |
|
&Apache::lonxml::warning(&mt('Rule indices should only contain alphanumeric characters.')); |
|
} |
|
$label=~s/\W//gs; |
|
unless ($label) { |
|
$label='R'.$Apache::functionplotresponse::counter; |
|
} else { |
|
$label='R'.$label; |
|
} |
|
push(@Apache::functionplotresponse::functionplotvectorrules,join(':',( |
|
$label, |
|
'custom', |
|
&escape($Apache::response::custom_answer[-1]) |
|
))); |
|
} |
|
&Apache::lonxml::deregister('Apache::response',('answer')); |
|
return $result; |
|
} |
|
|
|
|
|
|
# |
# |
Line 1131 sub start_functionplotresponse {
|
Line 1332 sub start_functionplotresponse {
|
.&Apache::edit::end_row() |
.&Apache::edit::end_row() |
.&Apache::edit::start_spanning_row() |
.&Apache::edit::start_spanning_row() |
."\n"; |
."\n"; |
$result.=&Apache::edit::text_arg('Label x-axis:','xlabel', |
$result.=&Apache::edit::text_arg('Width (pixels):','width', |
|
$token,'6').' '. |
|
&Apache::edit::text_arg('Height (pixels):','height', |
|
$token,'6').'<br />'. |
|
&Apache::edit::text_arg('Label x-axis:','xlabel', |
$token,'6').' '. |
$token,'6').' '. |
&Apache::edit::text_arg('Minimum x-value:','xmin', |
&Apache::edit::text_arg('Minimum x-value:','xmin', |
$token,'4').' '. |
$token,'4').' '. |
Line 1154 sub start_functionplotresponse {
|
Line 1359 sub start_functionplotresponse {
|
&Apache::edit::end_row().&Apache::edit::start_spanning_row(); |
&Apache::edit::end_row().&Apache::edit::start_spanning_row(); |
} elsif ($target eq 'modified') { |
} elsif ($target eq 'modified') { |
my $constructtag=&Apache::edit::get_new_args($token,$parstack, |
my $constructtag=&Apache::edit::get_new_args($token,$parstack, |
$safeeval,'xlabel','xmin','xmax','ylabel','ymin','ymax', |
$safeeval,'width','height','xlabel','xmin','xmax','ylabel','ymin','ymax', |
'xaxisvisible','yaxisvisible','gridvisible','answerdisplay'); |
'xaxisvisible','yaxisvisible','gridvisible','answerdisplay'); |
if ($constructtag) { $result = &Apache::edit::rebuild_tag($token); } |
if ($constructtag) { $result = &Apache::edit::rebuild_tag($token); } |
|
|
Line 1164 sub start_functionplotresponse {
|
Line 1369 sub start_functionplotresponse {
|
($env{'form.answer_output_mode'} ne 'tex') && |
($env{'form.answer_output_mode'} ne 'tex') && |
($Apache::lonhomework::viewgrades == 'F')) { |
($Apache::lonhomework::viewgrades == 'F')) { |
my (undef,undef,$udom,$uname)=&Apache::lonnet::whichuser(); |
my (undef,undef,$udom,$uname)=&Apache::lonnet::whichuser(); |
my $windowopen=&Apache::lonhtmlcommon::javascript_docopen(); |
|
my $start_page = &Apache::loncommon::start_page('Rules Log', undef, |
|
{'only_body' => 1, |
|
'bgcolor' => '#FFFFFF', |
|
'js_ready' => 1,}); |
|
my $end_page = &Apache::loncommon::end_page({'js_ready' => 1,}); |
|
$uname =~s/\W//g; |
$uname =~s/\W//g; |
$udom =~s/\W//g; |
$udom =~s/\W//g; |
my $function_name = |
my $function_name = |
join('_','LONCAPA_scriptvars',$uname,$udom, |
join('_','LONCAPA_scriptvars',$uname,$udom, |
$env{'form.counter'},$Apache::lonxml::curdepth); |
$env{'form.counter'},$Apache::lonxml::curdepth); |
my $rules_var ="<script type=\"text/javascript\"> |
&Apache::lonxml::add_script_result( |
// <![CDATA[ |
&Apache::loncommon::modal_adhoc_window($function_name,700,500, |
function $function_name() {newWindow=open('','new_W','width=500,height=500,scrollbars=1,resizable=yes');newWindow.$windowopen;newWindow.document.writeln('$start_page<pre>". |
'<pre>'.$Apache::functionplotresponse::ruleslog.'</pre>', |
$Apache::functionplotresponse::ruleslog. |
&mt('Rules Log'))."<br />"); |
"<\\/pre>$end_page');newWindow.document.close();newWindow.focus()} |
} |
// ]]> |
|
</script><a href=\"javascript:$function_name();void(0);\">".&mt('Rules Log')."</a><br />"; |
|
&Apache::lonxml::add_script_result($rules_var); |
|
} |
|
|
|
return $result; |
return $result; |
} |
} |
|
|
Line 1271 sub fpr_d2fdx2 {
|
Line 1465 sub fpr_d2fdx2 {
|
$Apache::functionplotresponse::fpr_xmax, |
$Apache::functionplotresponse::fpr_xmax, |
$arg)]; |
$arg)]; |
} |
} |
|
|
|
sub fpr_vectorcoords { |
|
my ($arg)=@_; |
|
$arg=~s/\W//gs; |
|
$arg=ucfirst($arg); |
|
my $id=$Apache::inputtags::response[-1]; |
|
my $partid=$Apache::inputtags::part; |
|
my $internalid = $partid.'_'.$id; |
|
return ($env{'form.HWVAL_'.$internalid.'_'.$arg.'Start_x'}, |
|
$env{'form.HWVAL_'.$internalid.'_'.$arg.'End_x'}, |
|
$env{'form.HWVAL_'.$internalid.'_'.$arg.'Start_y'}, |
|
$env{'form.HWVAL_'.$internalid.'_'.$arg.'End_y'}); |
|
} |
|
|
|
sub fpr_objectcoords { |
|
my ($arg)=@_; |
|
$arg=~s/\W//gs; |
|
$arg=ucfirst($arg); |
|
my $id=$Apache::inputtags::response[-1]; |
|
my $partid=$Apache::inputtags::part; |
|
my $internalid = $partid.'_'.$id; |
|
return ($env{'form.HWVAL_'.$internalid.'_'.$arg.'_x'}, |
|
$env{'form.HWVAL_'.$internalid.'_'.$arg.'_y'}); |
|
} |
|
|
|
sub fpr_vectorlength { |
|
my ($arg)=@_; |
|
my ($xs,$xe,$ys,$ye)=&fpr_vectorcoords($arg); |
|
return sqrt(($xe-$xs)*($xe-$xs)+($ye-$ys)*($ye-$ys)); |
|
} |
|
|
|
sub fpr_vectorangle { |
|
my ($arg)=@_; |
|
my ($xs,$xe,$ys,$ye)=&fpr_vectorcoords($arg); |
|
my $angle=57.2957795*atan2(($ye-$ys),($xe-$xs)); |
|
if ($angle<0) { $angle=360+$angle; } |
|
return $angle; |
|
} |
|
|
|
sub vectorcoords { |
|
my ($id,$label)=@_; |
|
return ($env{'form.HWVAL_'.$id.'_'.$label.'Start_x'}, |
|
$env{'form.HWVAL_'.$id.'_'.$label.'End_x'}, |
|
$env{'form.HWVAL_'.$id.'_'.$label.'Start_y'}, |
|
$env{'form.HWVAL_'.$id.'_'.$label.'End_y'}); |
|
} |
|
|
|
sub objectcoords { |
|
my ($id,$label)=@_; |
|
return ($env{'form.HWVAL_'.$id.'_'.$label.'_x'}, |
|
$env{'form.HWVAL_'.$id.'_'.$label.'_y'}); |
|
} |
|
|
|
sub attached { |
|
my ($id,$vector,$objects,$xmin,$xmax,$ymin,$ymax)=@_; |
|
my ($xs,$xe,$ys,$ye)=&vectorcoords($id,$vector); |
|
my $tolx=($xmax-$xmin)/100.; |
|
my $toly=($ymax-$ymin)/100.; |
|
my $tail=0; |
|
my $tip=0; |
|
foreach my $obj (split(/\s*\,\s*/,$objects)) { |
|
$obj=~s/\W//g; |
|
unless ($obj) { next; } |
|
$obj=ucfirst($obj); |
|
my ($xo,$yo)=&objectcoords($id,$obj); |
|
&addlog("Proximity $vector ($xs,$ys)-($xe,$ye) to $obj ($xo,$yo)"); |
|
if ((abs($xs-$xo)<$tolx) && (abs($ys-$yo)<$toly)) { |
|
$tail=1; |
|
&addlog("Attached tail: $obj"); |
|
} |
|
if ((abs($xe-$xo)<$tolx) && (abs($ye-$yo)<$toly)) { |
|
$tip=1; |
|
&addlog("Attached tip: $obj"); |
|
} |
|
} |
|
&addlog("Result tail:$tail tip:$tip"); |
|
return($tail,$tip); |
|
} |
|
|
|
|
|
sub vectorangle { |
|
my ($x,$y)=@_; |
|
my $angle=57.2957795*atan2($y,$x); |
|
if ($angle<0) { $angle=360+$angle; } |
|
return $angle; |
|
} |
|
|
|
sub vectorlength { |
|
my ($x,$y)=@_; |
|
return sqrt($x*$x+$y*$y); |
|
} |
|
|
|
sub relvector { |
|
my ($xs,$xe,$ys,$ye)=@_; |
|
return ($xe-$xs,$ye-$ys); |
|
} |
|
|
|
sub plotvectorlength { |
|
return &vectorlength(&relvector(&vectorcoords(@_))); |
|
} |
|
|
|
sub plotvectorangle { |
|
return &vectorangle(&relvector(&vectorcoords(@_))); |
|
} |
|
|
|
|
|
# |
|
# Evaluate a functionplotvectorrule |
|
# |
|
|
|
sub functionplotvectorrulecheck { |
|
my ($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)=@_; |
|
&addlog("================="); |
|
my ($label,$type)=split(/\:/,$rule); |
|
if ($type eq 'vector') { |
|
return &vectorcheck($rule,$xmin,$xmax,$ymin,$ymax,$safeeval); |
|
} elsif ($type eq 'sum') { |
|
return &sumcheck($rule,$xmin,$xmax,$ymin,$ymax,$safeeval); |
|
} elsif ($type eq 'custom') { |
|
return &customcheck($rule,$safeeval); |
|
} |
|
} |
|
|
|
sub vectorcheck { |
|
my ($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)=@_; |
|
my ($label,$type,$id,$vector, |
|
$attachpoint,$notattachpoint, |
|
$tailpoint,$tippoint,$nottailpoint,$nottippoint, |
|
$length,$angle,$lengtherror,$angleerror)=split(/\:/,$rule); |
|
&addlog("Vector Rule $label for vector ".$vector); |
|
if ($length ne '') { |
|
&addlog("Checking for length $length with error $lengtherror"); |
|
$length=&Apache::run::run($length,$safeeval); |
|
&addlog("Length evaluated to $length"); |
|
my $thislength=&plotvectorlength($id,$vector); |
|
&addlog("Found length $thislength"); |
|
if (abs($thislength-$length)>$lengtherror) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($angle ne '') { |
|
&addlog("Checking for angle $angle with error $angleerror"); |
|
$angle=&Apache::run::run($angle,$safeeval); |
|
&addlog("Angle evaluated to $angle"); |
|
my $thisangle=&plotvectorangle($id,$vector); |
|
&addlog("Found angle $thisangle"); |
|
my $anglediff=abs($thisangle-$angle); |
|
&addlog("Angle difference: $anglediff"); |
|
if ($anglediff>360.-$anglediff) { |
|
$anglediff=360.-$anglediff; |
|
} |
|
&addlog("Smallest angle difference: $anglediff"); |
|
if ($anglediff>$angleerror) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($attachpoint ne '') { |
|
&addlog("Checking attached: ".$attachpoint); |
|
my ($tail,$tip)=&attached($id,$vector,$attachpoint,$xmin,$xmax,$ymin,$ymax); |
|
unless ($tail || $tip) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($notattachpoint ne '') { |
|
&addlog("Checking not attached: ".$notattachpoint); |
|
my ($tail,$tip)=&attached($id,$vector,$notattachpoint,$xmin,$xmax,$ymin,$ymax); |
|
if ($tail || $tip) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($tailpoint ne '') { |
|
&addlog("Checking tail: ".$tailpoint); |
|
my ($tail,$tip)=&attached($id,$vector,$tailpoint,$xmin,$xmax,$ymin,$ymax); |
|
unless ($tail) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($nottailpoint ne '') { |
|
&addlog("Checking not tail: ".$nottailpoint); |
|
my ($tail,$tip)=&attached($id,$vector,$nottailpoint,$xmin,$xmax,$ymin,$ymax); |
|
if ($tail) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($tippoint ne '') { |
|
&addlog("Checking tip: ".$tippoint); |
|
my ($tail,$tip)=&attached($id,$vector,$tippoint,$xmin,$xmax,$ymin,$ymax); |
|
unless ($tip) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($nottippoint ne '') { |
|
&addlog("Checking not tip: ".$nottippoint); |
|
my ($tail,$tip)=&attached($id,$vector,$nottippoint,$xmin,$xmax,$ymin,$ymax); |
|
if ($tip) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
|
|
&addlog("Rule $label passed."); |
|
return 1; |
|
} |
|
|
|
sub sumcheck { |
|
my ($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)=@_; |
|
my ($label,$type,$id,$vectors,$length,$angle,$lengtherror,$angleerror)=split(/\:/,$rule); |
|
&addlog("Vector Sum Rule $label for vectors ".$vectors); |
|
my $sumx=0; |
|
my $sumy=0; |
|
foreach my $sv (split(/\s*\,\s*/,$vectors)) { |
|
my ($rx,$ry)=&relvector(&vectorcoords($id,$sv)); |
|
$sumx+=$rx; |
|
$sumy+=$ry; |
|
} |
|
&addlog("Sum vector ($sumx,$sumy)"); |
|
if ($length ne '') { |
|
&addlog("Checking length $length with error $lengtherror"); |
|
$length=&Apache::run::run($length,$safeeval); |
|
&addlog("Evaluated to $length"); |
|
my $thislength=&vectorlength($sumx,$sumy); |
|
&addlog("Actual length $thislength"); |
|
if (abs($length-$thislength)>$lengtherror) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
if ($angle ne '') { |
|
&addlog("Checking angle $angle with error $angleerror"); |
|
$angle=&Apache::run::run($angle,$safeeval); |
|
&addlog("Evaluated to $angle"); |
|
my $thisangle=&vectorangle($sumx,$sumy); |
|
&addlog("Actual angle $thisangle"); |
|
my $anglediff=abs($thisangle-$angle); |
|
&addlog("Angle difference: $anglediff"); |
|
if ($anglediff>360.-$anglediff) { |
|
$anglediff=360.-$anglediff; |
|
} |
|
&addlog("Smallest angle difference: $anglediff"); |
|
if ($anglediff>$angleerror) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
} |
|
&addlog("Rule $label passed."); |
|
return 1; |
|
} |
|
|
|
sub customcheck { |
|
my ($rule,$safeeval)=@_; |
|
my ($label,$type,$prg)=split(/\:/,$rule); |
|
&addlog("Custom Rule ".$label); |
|
my $result=&Apache::run::run(&unescape($prg),$safeeval); |
|
&addlog("Algorithm returned $result"); |
|
unless ($result) { |
|
&setfailed($label); |
|
return 0; |
|
} |
|
&addlog("Rule $label passed."); |
|
return 1; |
|
} |
|
|
|
# |
|
# Evaluate a functionplotrule |
|
# |
|
|
sub functionplotrulecheck { |
sub functionplotrulecheck { |
my ($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)=@_; |
my ($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)=@_; |
Line 1283 sub functionplotrulecheck {
|
Line 1749 sub functionplotrulecheck {
|
# |
# |
# Evaluate the value |
# Evaluate the value |
# |
# |
if ($value=~/\D/) { |
if (($value=~/\D/) && ($value ne 'undef')) { |
$Apache::functionplotresponse::fpr_xmin=$xmin; |
$Apache::functionplotresponse::fpr_xmin=$xmin; |
$Apache::functionplotresponse::fpr_xmax=$xmax; |
$Apache::functionplotresponse::fpr_xmax=$xmax; |
$value=&Apache::run::run($value,$safeeval); |
$value=&Apache::run::run($value,$safeeval); |
Line 1367 sub functionplotrulecheck {
|
Line 1833 sub functionplotrulecheck {
|
unless (&compare_rel($relationship,$value,$integral,$tol)) { |
unless (&compare_rel($relationship,$value,$integral,$tol)) { |
&addlog("Actual integral ".(defined($integral)?$integral:'undef').", expected $value, tolerance $tol"); |
&addlog("Actual integral ".(defined($integral)?$integral:'undef').", expected $value, tolerance $tol"); |
&addlog("Rule $label failed."); |
&addlog("Rule $label failed."); |
my $hintlabel=$label; |
&setfailed($label); |
$hintlabel=~s/^R//; |
|
push(@Apache::functionplotresponse::failedrules,$hintlabel); |
|
&addlog("Set hint condition $hintlabel"); |
|
return 0; |
return 0; |
} |
} |
} else { |
} else { |
Line 1436 sub checklength {
|
Line 1899 sub checklength {
|
} |
} |
|
|
sub setfailed { |
sub setfailed { |
my ($label)=@_; |
my ($hintlabel)=@_; |
my $hintlabel=$label; |
|
$hintlabel=~s/^R//; |
$hintlabel=~s/^R//; |
push(@Apache::functionplotresponse::failedrules,$hintlabel); |
push(@Apache::functionplotresponse::failedrules,$hintlabel); |
&addlog("Set hint condition $hintlabel"); |
&addlog("Set hint condition $hintlabel"); |
Line 1496 sub end_functionplotruleset {
|
Line 1958 sub end_functionplotruleset {
|
last; |
last; |
} |
} |
} |
} |
|
# And now go through the vector rules |
|
foreach my $rule (@Apache::functionplotresponse::functionplotvectorrules) { |
|
unless (&functionplotvectorrulecheck($rule,$xmin,$xmax,$ymin,$ymax,$safeeval)) { |
|
$ad='INCORRECT'; |
|
last; |
|
} |
|
} |
# If it's not wrong, it's correct |
# If it's not wrong, it's correct |
unless ($ad) { $ad='EXACT_ANS' }; |
unless ($ad) { $ad='EXACT_ANS' }; |
} |
} |
Line 1594 sub end_functionplotelements {
|
Line 2063 sub end_functionplotelements {
|
# generate the input fields |
# generate the input fields |
$result.=$Apache::functionplotresponse::inputfields; |
$result.=$Apache::functionplotresponse::inputfields; |
# actually start the <applet>-tag |
# actually start the <applet>-tag |
$result.=&geogebra_startcode($internalid); |
$result.=&geogebra_startcode($internalid, |
|
&Apache::lonxml::get_param('width',$parstack,$safeeval,-2), |
|
&Apache::lonxml::get_param('height',$parstack,$safeeval,-2)); |
# load the spline bytecode |
# load the spline bytecode |
$result.=&geogebra_spline_program(); |
$result.=&geogebra_spline_program(); |
# set default parameters |
# set default parameters |