Diff for /loncom/homework/functionplotresponse.pm between versions 1.76 and 1.90

version 1.76, 2011/11/19 20:03:58 version 1.90, 2011/12/10 18:10:56
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','functionplotvectorsumrule',                                                              'plotobject','plotvector',
                                                               'functionplotvectorrule','functionplotvectorsumrule',
                                                               'functionplotcustomrule',
                                                             'functionplotrule','functionplotruleset',                                                              'functionplotrule','functionplotruleset',
                                                             'functionplotelements'));                                                              'functionplotelements'));
 }  }
Line 46  BEGIN { Line 49  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 396  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 653  sub start_functionplotrule { Line 661  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 771  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 $attachpoint=&Apache::lonxml::get_param('attachpoint',$parstack,$safeeval);  
       $attachpoint=~s/\W//gs;  
       $attachpoint=ucfirst($attachpoint);  
   
       my $notattachpoint=&Apache::lonxml::get_param('notattachpoint',$parstack,$safeeval);  
       $notattachpoint=~s/\W//gs;  
       $notattachpoint=ucfirst($notattachpoint);  
   
       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',             'vector',
              $internalid,
            $vector,             $vector,
            $attachpoint,             &Apache::lonxml::get_param('attachpoint',$parstack,$safeeval),
            $notattachpoint,             &Apache::lonxml::get_param('notattachpoint',$parstack,$safeeval),
            $tailpoint,             &Apache::lonxml::get_param('tailpoint',$parstack,$safeeval),
            $tippoint,             &Apache::lonxml::get_param('tippoint',$parstack,$safeeval),
            $nottailpoint,             &Apache::lonxml::get_param('nottailpoint',$parstack,$safeeval),
            $nottippoint,             &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 837  sub start_functionplotvectorrule { Line 822  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')}=ucfirst($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('attachpoint')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('attachpoint')});  
     $env{'form.'.&Apache::edit::html_element_name('notattachpoint')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('notattachpoint')});  
     $env{'form.'.&Apache::edit::html_element_name('tailpoint')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('tailpoint')});  
     $env{'form.'.&Apache::edit::html_element_name('tippoint')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('tippoint')});  
     $env{'form.'.&Apache::edit::html_element_name('nottailpoint')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('nottailpoint')});  
     $env{'form.'.&Apache::edit::html_element_name('nottippoint')}=ucfirst($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','attachpoint','notattachpoint',                                                   $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 885  sub start_functionplotvectorsumrule { Line 864  sub start_functionplotvectorsumrule {
    }     }
    $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 $object=&Apache::lonxml::get_param('object',$parstack,$safeeval);        my $id=$Apache::inputtags::response[-1];
       $object=~s/\W//gs;        my $partid=$Apache::inputtags::part;
       $object=ucfirst($object);        my $internalid = $partid.'_'.$id;
         my $vectors=&Apache::lonxml::get_param('vectors',$parstack,$safeeval);
       push(@Apache::functionplotresponse::functionplotvectorrules,join(':',(        push(@Apache::functionplotresponse::functionplotvectorrules,join(':',(
            $label,             $label,
            'sum',             'sum',
            $object,             $internalid,
              $vectors,
            &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('lengthabserror',$parstack,$safeeval),             &Apache::lonxml::get_param('angleerror',$parstack,$safeeval),
            &Apache::lonxml::get_param('anglepercenterror',$parstack,$safeeval),  
           )));            )));
    } elsif ($target eq 'edit') {     } elsif ($target eq 'edit') {
         $result=&Apache::edit::tag_start($target,$token,'Function Plot Vector Sum Rule').          $result=&Apache::edit::tag_start($target,$token,'Function Plot Vector Sum Rule').
              &Apache::edit::text_arg('Index/Name:','index',               &Apache::edit::text_arg('Index/Name:','index',
                                      $token,'10').'&nbsp;'.                                       $token,'10').'&nbsp;'.
              &Apache::edit::text_arg('Vectors attached to object:','object',               &Apache::edit::text_arg('Comma-separated list of vectors:','vectors',
                                       $token,'16').'<br />'.                                        $token,'30').'<br />'.
              &Apache::edit::text_arg('Sum vector length:','length',               &Apache::edit::text_arg('Sum vector length:','length',
                                      $token,'16').                                       $token,'30').
                &Apache::edit::text_arg('Absolute error length:','lengtherror',
                                        $token,'8').'<br />'.
              &Apache::edit::text_arg('Sum vector angle:','angle',               &Apache::edit::text_arg('Sum vector 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('Absolute error length:','lengthabserror',  
                                      $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('object')}=ucfirst($env{'form.'.&Apache::edit::html_element_name('object')});        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
     my $constructtag=&Apache::edit::get_new_args($token,$parstack,                                                     $safeeval,'index','vectors',
                                                  $safeeval,'index','object',                                                               'length','angle',
                                                            'length','angle',                                                               'lengtherror','angleerror');
                                                            'lengthpercenterror','lengthabserror','anglepercenterror');        if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); }
     if ($constructtag) { $result=&Apache::edit::rebuild_tag($token); }  
    }     }
    return $result;     return $result;
 }  }
Line 945  sub end_functionplotvectorsumrule { Line 919  sub end_functionplotvectorsumrule {
 }  }
   
 #  #
   # <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;
   }
   
   
   
   #
 # <spline index="..." order="1,2,3,4" initx="..." inity="..." scalex="..." scaley="..." />  # <spline index="..." order="1,2,3,4" initx="..." inity="..." scalex="..." scaley="..." />
 #  #
 # Unfortunately, GeoGebra seems to want all splines after everything else, so we need to store them  # Unfortunately, GeoGebra seems to want all splines after everything else, so we need to store them
Line 1219  sub start_functionplotresponse { Line 1252  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').'&nbsp;'.
                &Apache::edit::text_arg('Height (pixels):','height',
                                        $token,'6').'<br />'.
                &Apache::edit::text_arg('Label x-axis:','xlabel',
                                      $token,'6').'&nbsp;'.                                       $token,'6').'&nbsp;'.
              &Apache::edit::text_arg('Minimum x-value:','xmin',               &Apache::edit::text_arg('Minimum x-value:','xmin',
                                      $token,'4').'&nbsp;'.                                       $token,'4').'&nbsp;'.
Line 1242  sub start_functionplotresponse { Line 1279  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 1252  sub start_functionplotresponse { Line 1289  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 1359  sub fpr_d2fdx2 { Line 1385  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 1455  sub functionplotrulecheck { Line 1753  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 1524  sub checklength { Line 1819  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 1584  sub end_functionplotruleset { Line 1878  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 1682  sub end_functionplotelements { Line 1983  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

Removed from v.1.76  
changed lines
  Added in v.1.90


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