--- loncom/homework/drawimage.pm	2006/05/30 12:45:36	1.9
+++ loncom/homework/drawimage.pm	2024/04/09 20:01:31	1.14
@@ -1,7 +1,7 @@
 # The LearningOnline Network with CAPA
 # programatic image drawing
 #
-# $Id: drawimage.pm,v 1.9 2006/05/30 12:45:36 www Exp $
+# $Id: drawimage.pm,v 1.14 2024/04/09 20:01:31 raeburn Exp $
 #
 # Copyright Michigan State University Board of Trustees
 #
@@ -29,7 +29,11 @@
 package Apache::drawimage;
 use strict;
 use Apache::loncommon;
+use Apache::lonnet;
+use Apache::lonxml;
+use Apache::edit;
 use lib '/home/httpd/lib/perl/';
+use Time::HiRes qw(gettimeofday);
 use LONCAPA;
  
 
@@ -43,40 +47,82 @@ BEGIN {
 sub start_drawimage {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     &Apache::lonxml::register('Apache::drawimage',('text','line','rectangle','arc','fill','polygon','image'));
+    push(@Apache::lonxml::namespace,'drawimage');
+    my $result;
     if ($target eq 'web' || $target eq 'tex') {
 	my $new_id=&Apache::loncommon::get_cgi_id();
 	if ($cgi_id) { push(@cgi_ids,$cgi_id); } else { undef(%args); }
 	$cgi_id=$new_id;
+    } elsif ($target eq 'edit') {
+        $result .= &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('Background color:','bgcolor' ,$token,8).
+            &Apache::edit::text_arg('Width(pixel):'    ,'width'   ,$token,6).
+            &Apache::edit::text_arg('Height(pixel):'   ,'height'  ,$token,6).
+            &Apache::edit::text_arg('TeXWidth(mm):'    ,'texwidth',$token,6).
+            &Apache::edit::end_row().&Apache::edit::start_spanning_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'bgcolor','width',
+                                                     'height','texwidth');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
-    return '';
+    return $result;
 }
 
 sub end_drawimage {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    # need to call rand everytime start_script would evaluate, as the
+    # safe space rand number generator and the global rand generator
+    # are not separate
+    my $randnumber;
+    if ($target eq 'web' || $target eq 'tex' || $target eq 'grade' ||
+        $target eq 'answer') {
+        $randnumber=int(rand(1000));
+    }
     if ($target eq 'web' || $target eq 'tex') {
 	my $width = &Apache::lonxml::get_param('width',$parstack,$safeeval);
 	my $height =&Apache::lonxml::get_param('height',$parstack,$safeeval);
 	my $bgcolor =&Apache::lonxml::get_param('bgcolor',$parstack,$safeeval);
 	if (!$width) { $width=300; }
 	if (!$height) { $height=300; }
-	$result.="<img width='$width' height='$height'
+        $args{"cgi.$cgi_id.BGCOLOR"}=join(':',($bgcolor));
+        if ($target eq 'tex') {
+            my $texwidth=&Apache::lonxml::get_param('texwidth',$parstack,$safeeval,undef,1);
+            if (!$texwidth) { $texwidth='90'; }
+            $args{"cgi.$cgi_id.SIZE"}=join(':',($width,$height,$texwidth));
+            my $tmpdir = LONCAPA::tempdir(); # Where temporary files live:
+            ## Determine filename
+            my ($seconds, $microseconds) = gettimeofday;
+            my $filename = $env{'user.name'}.'_'.$env{'user.domain'}.
+                           '_'.$seconds.'_'.$microseconds.'_'.$$.$randnumber.'_drawimage.eps';
+            $args{"cgi.$cgi_id.EPSFILE"} = $env{'user.name'}.'_'.$env{'user.domain'}.
+                                           '_'.$seconds.'_'.$microseconds.'_'.$$.$randnumber.
+                                           '_drawimage.eps';
+            $result = "%DYNAMICIMAGE:$width:$height:$texwidth\n";
+            $result .= '\graphicspath{{'.$tmpdir.'}}'."\n";
+            $result .= '\includegraphics[width='.$texwidth.' mm]{'.$filename.'}';
+            &Apache::lonxml::register_ssi('/adm/randomlabel.png?token='.$cgi_id);
+        } else {
+	    $args{"cgi.$cgi_id.SIZE"}=join(':',($width,$height));
+	    $result.="<img width='$width' height='$height'
                            src='/adm/randomlabel.png?token=$cgi_id' />\n";
-	$args{"cgi.$cgi_id.SIZE"}=join(':',($width,$height));
-	$args{"cgi.$cgi_id.BGCOLOR"}=join(':',($bgcolor));
-	&Apache::lonnet::appenv(%args);
+        }
+	&Apache::lonnet::appenv(\%args);
 	if (@cgi_ids) {
 	    $cgi_id=pop(@cgi_ids);
 	} else {
 	    undef($cgi_id);
 	}
     } elsif ($target eq 'edit') {
-    } elsif ($target eq 'modified') {
+        $result.=&Apache::edit::end_table;
     }
-    
-    &Apache::lonxml::register('Apache::drawimage',
-			      ('text','line','rectangle','arc','fill',
-			       'polygon'));
+    pop(@Apache::lonxml::namespace);
+    &Apache::lonxml::deregister('Apache::drawimage',
+			        ('text','line','rectangle','arc','fill',
+			         'polygon'));
     return $result;
 }
 
@@ -85,6 +131,31 @@ sub start_text {
     my $result;
     if ($target eq 'web' || $target eq 'tex') {
 	&Apache::lonxml::startredirection();
+    } elsif ($target eq 'edit') {
+        $result =
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x:','x',$token,6).
+            &Apache::edit::text_arg('y:','y',$token,6).
+            &Apache::edit::text_arg('font:','font',$token,12).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::select_arg('direction:','direction',
+                                      ['vertical','horizontal'],$token).
+            &Apache::edit::text_arg('rotation:','rotation',$token,6);
+        my $text=&Apache::lonxml::get_all_text('/text',$parser,$style);
+        $result .=
+            &Apache::edit::editfield($token->[1],$text,'Text',60,2).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x','y',
+                                                     'font','color',
+                                                     'direction','rotation');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        } else {
+            $result = $token->[4];
+        }
+        $result .= &Apache::edit::modifiedfield('/text',$parser);
     }
     return $result;
 }
@@ -98,11 +169,14 @@ sub end_text {
         my $font  = &Apache::lonxml::get_param('font',$parstack,$safeeval);
         my $color = &Apache::lonxml::get_param('color',$parstack,$safeeval);
         my $direction = &Apache::lonxml::get_param('direction',$parstack,$safeeval);
+        my $rotation = &Apache::lonxml::get_param('rotation',$parstack,$safeeval);
 	my $text  = &Apache::lonxml::endredirection();
         $text = &escape($text);
         $args{"cgi.$cgi_id.OBJTYPE"}.='LABEL:';
 	my $i=$args{"cgi.$cgi_id.OBJCOUNT"}++;
-	$args{"cgi.$cgi_id.OBJ$i"}=join(':',($x,$y,$text,$font,$color,$direction));
+	$args{"cgi.$cgi_id.OBJ$i"}=join(':',($x,$y,$text,$font,$color,$direction,$rotation));
+    } elsif ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
     }
     return $result;
 }
@@ -120,6 +194,23 @@ sub start_line {
 	my $i=$args{"cgi.$cgi_id.OBJCOUNT"}++;
 	$args{"cgi.$cgi_id.OBJ$i"}=join(':',($x1,$y1,$x2,$y2,$color,$thickness));
 	$args{"cgi.$cgi_id.OBJTYPE"}.='LINE:';
+    } elsif ($target eq 'edit') {
+        $result .=
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x1:','x1' ,$token,6).
+            &Apache::edit::text_arg('y1:','y1' ,$token,6).
+            &Apache::edit::text_arg('x2:','x2' ,$token,6).
+            &Apache::edit::text_arg('y2:','y2' ,$token,6).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::text_arg('thickness:','thickness',$token,6).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x1','y1',
+                                                     'x2','y2','color','thickness');
+        if ($constructtag) { 
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -127,6 +218,9 @@ sub start_line {
 sub end_line {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    if ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
+    }
     return $result;
 }
 
@@ -147,6 +241,26 @@ sub start_rectangle {
 	$args{"cgi.$cgi_id.OBJ$i"}=
 	    join(':',($x1,$y1,$x2,$y2,$color,$thickness,$filled));
 	$args{"cgi.$cgi_id.OBJTYPE"}.='RECTANGLE:';
+    } elsif ($target eq 'edit') {
+        $result .=
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x1:','x1' ,$token,6).
+            &Apache::edit::text_arg('y1:','y1' ,$token,6).
+            &Apache::edit::text_arg('x2:','x2' ,$token,6).
+            &Apache::edit::text_arg('y2:','y2' ,$token,6).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::text_arg('thickness:','thickness',$token,6).
+            &Apache::edit::select_arg('filled:','filled',
+                                      ['no','yes'],$token).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x1','y1',
+                                                     'x2','y2','color','thickness',
+                                                     'filled');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -154,6 +268,9 @@ sub start_rectangle {
 sub end_rectangle {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    if ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
+    }
     return $result;
 }
 
@@ -175,6 +292,30 @@ sub start_arc {
 	    join(':',($x,$y,$width,$height,$start,$end,$color,$thickness,
 		      $filled));
 	$args{"cgi.$cgi_id.OBJTYPE"}.='ARC:';
+    } elsif ($target eq 'edit') {
+        $result .=
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x:','x',$token,6).
+            &Apache::edit::text_arg('y:','y',$token,6).
+            &Apache::edit::text_arg('width:','width',$token,6).
+            &Apache::edit::text_arg('height:','height',$token,6).
+            &Apache::edit::text_arg('start:','start',$token,6).
+            &Apache::edit::text_arg('end:','end',$token,6).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::text_arg('thickness:','thickness',$token,6).
+            &Apache::edit::select_arg('filled:','filled',
+                                      ['no','yes'],$token).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x','y',
+                                                     'width','height',
+                                                     'start','end',
+                                                     'color','thickness',
+                                                     'filled');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -182,6 +323,9 @@ sub start_arc {
 sub end_arc {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    if ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
+    }
     return $result;
 }
 
@@ -195,6 +339,20 @@ sub start_fill {
 	my $i=$args{"cgi.$cgi_id.OBJCOUNT"}++;
 	$args{"cgi.$cgi_id.OBJ$i"}=join(':',($x,$y,$color));
 	$args{"cgi.$cgi_id.OBJTYPE"}.='FILL:';
+    } elsif ($target eq 'edit') {
+        $result .=
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x:','x' ,$token,6).
+            &Apache::edit::text_arg('y:','y' ,$token,6).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x','y',
+                                                     'color');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -202,6 +360,9 @@ sub start_fill {
 sub end_fill {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    if ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
+    }
     return $result;
 }
 
@@ -209,9 +370,27 @@ my @polygon;
 sub start_polygon {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    push(@Apache::lonxml::namespace,'polygon');
     &Apache::lonxml::register('Apache::drawimage',('point'));
-    if ($target eq 'web') {
+    if ($target eq 'web' || $target eq 'tex') {
 	undef(@polygon);
+    } elsif ($target eq 'edit') {
+        $result = 
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('color:','color',$token,8).
+            &Apache::edit::select_arg('Open:','open',
+                                           ['no','yes'],$token).
+            &Apache::edit::select_arg('Filled:','filled',
+                                           ['no','yes'],$token).
+            &Apache::edit::text_arg('thickness:','thickness',$token,6).
+            &Apache::edit::end_row().
+            &Apache::edit::start_spanning_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'open','filled');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -219,7 +398,7 @@ sub start_polygon {
 sub end_polygon {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;    
-    if ($target eq 'web') {
+    if ($target eq 'web' || $target eq 'tex') {
 	my $color=&Apache::lonxml::get_param('color',$parstack,$safeeval);
 	my $filled=&Apache::lonxml::get_param('filled',$parstack,$safeeval);
 	my $open=&Apache::lonxml::get_param('open',$parstack,$safeeval);
@@ -229,7 +408,10 @@ sub end_polygon {
 	$args{"cgi.$cgi_id.OBJTYPE"}.='POLYGON:';
 	$args{"cgi.$cgi_id.OBJ$i"}=join(':',($color,$thickness,$open,$filled));
 	$args{"cgi.$cgi_id.OBJEXTRA$i"}=join('-',@polygon);
+    } elsif ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
     }
+    pop(@Apache::lonxml::namespace);
     &Apache::lonxml::deregister('Apache::drawimage',('point'));
     return $result;
 }
@@ -237,10 +419,22 @@ sub end_polygon {
 sub start_point {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
-    if ($target eq 'web') {
+    if ($target eq 'web' || $target eq 'tex') {
 	my $x = &Apache::lonxml::get_param('x',$parstack,$safeeval);
         my $y = &Apache::lonxml::get_param('y',$parstack,$safeeval);
 	push (@polygon,"($x,$y)");
+    } elsif ($target eq 'edit') {
+        $result.= 
+            &Apache::edit::tag_start($target,$token).
+            &Apache::edit::text_arg('x:','x' ,$token,6).
+            &Apache::edit::text_arg('y:','y' ,$token,6).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x','y');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        }
     }
     return $result;
 }
@@ -248,6 +442,9 @@ sub start_point {
 sub end_point {
     my ($target,$token,$tagstack,$parstack,$parser,$safeeval,$style)=@_;
     my $result;
+    if ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
+    }
     return $result;
 }
 
@@ -256,6 +453,36 @@ sub start_image {
     my $result;
     if ($target eq 'web' || $target eq 'tex') {
 	&Apache::lonxml::startredirection();
+    } elsif ($target eq 'edit') {
+        my $bgimg=&Apache::lonxml::get_all_text('/image',$parser,$style);
+        $Apache::edit::bgimgsrc=$bgimg;
+        $Apache::edit::bgimgsrcdepth=$Apache::lonxml::curdepth;
+        my $only = join(',',&Apache::loncommon::filecategorytypes('Pictures'));
+        $result=&Apache::edit::tag_start($target,$token,'Background Image').
+            &Apache::edit::editline($token->[1],$bgimg,'Image Source File',40).
+            &Apache::edit::browse_or_search(undef,'textnode',undef,$only,undef,1).
+            '<br />'.
+            &Apache::edit::text_arg('x:','x',$token,6).
+            &Apache::edit::text_arg('y:','y',$token,6).
+            &Apache::edit::text_arg('clipx:','clipx',$token,6).
+            &Apache::edit::text_arg('clipy:','clipy',$token,6).
+            &Apache::edit::text_arg('Clip width(pixel):','clipwidth',$token,6).
+            &Apache::edit::text_arg('Clip height(pixel):','clipheight',$token,6).
+            &Apache::edit::text_arg('Scaled width:','scaledwidth',$token,6).
+            &Apache::edit::text_arg('Scaled height:','scaledheight',$token,6).
+            &Apache::edit::text_arg('Transparent:','transparent',$token,6).
+            &Apache::edit::end_row();
+    } elsif ($target eq 'modified') {
+        my $constructtag=&Apache::edit::get_new_args($token,$parstack,
+                                                     $safeeval,'x','y',
+                                                     'clipx','clipy','clipwidth','clipheight',
+                                                     'scaledwidth','scaledheight','transparent');
+        if ($constructtag) {
+            $result = &Apache::edit::rebuild_tag($token);
+        } else {
+            $result = $token->[4];
+        }
+        $result .= &Apache::edit::modifiedfield('/image',$parser);
     }
     return $result;
 }
@@ -285,6 +512,8 @@ sub end_image {
 	$args{"cgi.$cgi_id.OBJ$i"} = 
 	    join(':',($x,$y,&escape($bgimg),$transparent,
 		      $clipx,$clipy,$scaledwidth,$scaledheight,$clipwidth,$clipheight));
+    } elsif ($target eq 'edit') {
+        $result=&Apache::edit::end_table();
     }
     return $result;
 }