mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			1350 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			1350 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			Java
		
	
	
	
/* PostScriptGraphics2D.java -- AWT printer rendering class.
 | 
						|
   Copyright (C) 2006  Free Software Foundation, Inc.
 | 
						|
 | 
						|
This file is part of GNU Classpath.
 | 
						|
 | 
						|
GNU Classpath is free software; you can redistribute it and/or modify
 | 
						|
it under the terms of the GNU General Public License as published by
 | 
						|
the Free Software Foundation; either version 2, or (at your option)
 | 
						|
any later version.
 | 
						|
 | 
						|
GNU Classpath is distributed in the hope that it will be useful, but
 | 
						|
WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | 
						|
General Public License for more details.
 | 
						|
 | 
						|
You should have received a copy of the GNU General Public License
 | 
						|
along with GNU Classpath; see the file COPYING.  If not, write to the
 | 
						|
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | 
						|
02110-1301 USA.
 | 
						|
 | 
						|
Linking this library statically or dynamically with other modules is
 | 
						|
making a combined work based on this library.  Thus, the terms and
 | 
						|
conditions of the GNU General Public License cover the whole
 | 
						|
combination.
 | 
						|
 | 
						|
As a special exception, the copyright holders of this library give you
 | 
						|
permission to link this library with independent modules to produce an
 | 
						|
executable, regardless of the license terms of these independent
 | 
						|
modules, and to copy and distribute the resulting executable under
 | 
						|
terms of your choice, provided that you also meet, for each linked
 | 
						|
independent module, the terms and conditions of the license of that
 | 
						|
module.  An independent module is a module which is not derived from
 | 
						|
or based on this library.  If you modify this library, you may extend
 | 
						|
this exception to your version of the library, but you are not
 | 
						|
obligated to do so.  If you do not wish to do so, delete this
 | 
						|
exception statement from your version. */
 | 
						|
 | 
						|
package gnu.java.awt.print;
 | 
						|
 | 
						|
import java.awt.BasicStroke;
 | 
						|
import java.awt.Color;
 | 
						|
import java.awt.Composite;
 | 
						|
import java.awt.Paint;
 | 
						|
import java.awt.Font;
 | 
						|
import java.awt.FontMetrics;
 | 
						|
import java.awt.GradientPaint;
 | 
						|
import java.awt.Graphics;
 | 
						|
import java.awt.GraphicsConfiguration;
 | 
						|
import java.awt.Graphics2D;
 | 
						|
import java.awt.Image;
 | 
						|
import java.awt.Polygon;
 | 
						|
import java.awt.Rectangle;
 | 
						|
import java.awt.RenderingHints;
 | 
						|
import java.awt.Shape;
 | 
						|
import java.awt.Stroke;
 | 
						|
import java.awt.geom.AffineTransform;
 | 
						|
import java.awt.geom.Arc2D;
 | 
						|
import java.awt.geom.Ellipse2D;
 | 
						|
import java.awt.geom.RoundRectangle2D;
 | 
						|
import java.awt.geom.PathIterator;
 | 
						|
import java.awt.geom.Point2D;
 | 
						|
import java.awt.geom.Rectangle2D;
 | 
						|
import java.awt.font.FontRenderContext;
 | 
						|
import java.awt.font.GlyphVector;
 | 
						|
import java.awt.font.TextLayout;
 | 
						|
import java.awt.image.BufferedImage;
 | 
						|
import java.awt.image.BufferedImageOp;
 | 
						|
import java.awt.image.renderable.RenderableImage;
 | 
						|
import java.awt.image.RenderedImage;
 | 
						|
import java.awt.image.ImageObserver;
 | 
						|
import java.awt.image.PixelGrabber;
 | 
						|
import java.awt.print.PageFormat;
 | 
						|
import java.awt.print.Pageable;
 | 
						|
import java.awt.print.Paper;
 | 
						|
import java.awt.print.Printable;
 | 
						|
import java.awt.print.PrinterException;
 | 
						|
import java.awt.print.PrinterJob;
 | 
						|
import java.io.BufferedWriter;
 | 
						|
import java.io.File;
 | 
						|
import java.io.FileOutputStream;
 | 
						|
import java.io.IOException;
 | 
						|
import java.io.OutputStreamWriter;
 | 
						|
import java.io.PrintWriter;
 | 
						|
import java.text.AttributedCharacterIterator;
 | 
						|
import java.util.Map;
 | 
						|
 | 
						|
/**
 | 
						|
 * Class PostScriptGraphics2D - Class that implements the Graphics2D object,
 | 
						|
 * writing the output to a PostScript or EPS file
 | 
						|
 *
 | 
						|
 * @author Sven de Marothy
 | 
						|
 *
 | 
						|
 */
 | 
						|
class PostScriptGraphics2D extends Graphics2D
 | 
						|
{
 | 
						|
  /**
 | 
						|
   * The associated printer job.
 | 
						|
   */
 | 
						|
  private PrinterJob printerJob;
 | 
						|
 | 
						|
  /**
 | 
						|
   * Output file.
 | 
						|
   */
 | 
						|
  private PrintWriter out;
 | 
						|
 | 
						|
  // Graphics data
 | 
						|
  private AffineTransform currentTransform = new AffineTransform();
 | 
						|
  private AffineTransform pageTransform;
 | 
						|
  private RenderingHints renderingHints;
 | 
						|
  private Paint currentPaint = null;
 | 
						|
  private Shape clipShape = null;
 | 
						|
  private Font currentFont = null;
 | 
						|
  private Color currentColor = Color.black;
 | 
						|
  private Color backgroundColor = Color.white;
 | 
						|
  private Stroke currentStroke = null;
 | 
						|
  private static Stroke ordinaryStroke = new BasicStroke(0.0f,
 | 
						|
                                                         BasicStroke.CAP_BUTT,
 | 
						|
                                                         BasicStroke.JOIN_MITER);
 | 
						|
  private float cx; // current drawing position
 | 
						|
  private float cy; // current drawing position
 | 
						|
  private boolean currentFontIsPS; // set if currentFont is one of the above
 | 
						|
 | 
						|
  // settings
 | 
						|
  private double pageX = 595;
 | 
						|
  private double pageY = 842;
 | 
						|
  private double Y = pageY;
 | 
						|
  private boolean gradientOn = false;
 | 
						|
 | 
						|
  /**
 | 
						|
   * Constructor
 | 
						|
   *
 | 
						|
   */
 | 
						|
  public PostScriptGraphics2D( PrinterJob pg )
 | 
						|
  {
 | 
						|
    printerJob = pg;
 | 
						|
    // create transform objects
 | 
						|
    pageTransform = new AffineTransform();
 | 
						|
    currentTransform = new AffineTransform();
 | 
						|
 | 
						|
    /*
 | 
						|
      Create Rendering hints
 | 
						|
      No text aliasing
 | 
						|
      Quality color and rendering
 | 
						|
      Bicubic interpolation
 | 
						|
      Fractional metrics supported
 | 
						|
    */
 | 
						|
    renderingHints = new RenderingHints(null);
 | 
						|
    renderingHints.put(RenderingHints.KEY_RENDERING,
 | 
						|
                       RenderingHints.VALUE_RENDER_QUALITY);
 | 
						|
    renderingHints.put(RenderingHints.KEY_TEXT_ANTIALIASING,
 | 
						|
                       RenderingHints.VALUE_TEXT_ANTIALIAS_OFF);
 | 
						|
    renderingHints.put(RenderingHints.KEY_INTERPOLATION,
 | 
						|
                       RenderingHints.VALUE_INTERPOLATION_BICUBIC);
 | 
						|
    renderingHints.put(RenderingHints.KEY_FRACTIONALMETRICS,
 | 
						|
                       RenderingHints.VALUE_FRACTIONALMETRICS_ON);
 | 
						|
    renderingHints.put(RenderingHints.KEY_COLOR_RENDERING,
 | 
						|
                       RenderingHints.VALUE_COLOR_RENDER_QUALITY);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Spool a document to PostScript.
 | 
						|
   * If Pageable is non-null, it will print that, otherwise it will use
 | 
						|
   * the supplied printable and pageFormat.
 | 
						|
   */
 | 
						|
  public SpooledDocument spoolPostScript(Printable printable,
 | 
						|
                                         PageFormat pageFormat,
 | 
						|
                                         Pageable pageable)
 | 
						|
    throws PrinterException
 | 
						|
  {
 | 
						|
    try
 | 
						|
      {
 | 
						|
        // spool to a temporary file
 | 
						|
        File temp = File.createTempFile("cpspool", ".ps");
 | 
						|
        temp.deleteOnExit();
 | 
						|
 | 
						|
        out = new PrintWriter(new BufferedWriter
 | 
						|
                              (new OutputStreamWriter
 | 
						|
                               (new FileOutputStream(temp),
 | 
						|
                                "ISO8859_1"), 1000000));
 | 
						|
 | 
						|
        writePSHeader();
 | 
						|
 | 
						|
        if(pageable != null)
 | 
						|
          {
 | 
						|
            for(int index = 0; index < pageable.getNumberOfPages(); index++)
 | 
						|
              spoolPage(out, pageable.getPrintable(index),
 | 
						|
                        pageable.getPageFormat(index), index);
 | 
						|
          }
 | 
						|
        else
 | 
						|
          {
 | 
						|
            int index = 0;
 | 
						|
            while(spoolPage(out, printable, pageFormat, index++) ==
 | 
						|
                  Printable.PAGE_EXISTS)
 | 
						|
              ;
 | 
						|
          }
 | 
						|
        out.println("%%Trailer");
 | 
						|
        out.println("%%EOF");
 | 
						|
        out.close();
 | 
						|
        return new SpooledDocument( temp );
 | 
						|
      }
 | 
						|
    catch (IOException e)
 | 
						|
      {
 | 
						|
        PrinterException pe = new PrinterException();
 | 
						|
        pe.initCause(e);
 | 
						|
        throw pe;
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  //--------------------------------------------------------------------------
 | 
						|
 | 
						|
  /**
 | 
						|
   * Write the postscript file header,
 | 
						|
   * setup the page format and transforms.
 | 
						|
   */
 | 
						|
  private void writePSHeader()
 | 
						|
  {
 | 
						|
    out.println("%!PS-Adobe-3.0");
 | 
						|
    out.println("%%Title: "+printerJob.getJobName());
 | 
						|
    out.println("%%Creator: GNU Classpath ");
 | 
						|
    out.println("%%DocumentData: Clean8Bit");
 | 
						|
 | 
						|
    out.println("%%DocumentNeededResources: font Times-Roman Helvetica Courier");
 | 
						|
    out.println("%%EndComments");
 | 
						|
 | 
						|
    out.println("%%BeginProlog");
 | 
						|
    out.println("%%EndProlog");
 | 
						|
    out.println("%%BeginSetup");
 | 
						|
 | 
						|
    out.println("%%EndFeature");
 | 
						|
    setupFonts();
 | 
						|
    out.println("%%EndSetup");
 | 
						|
 | 
						|
    // set default fonts and colors
 | 
						|
    setFont( new Font("Dialog", Font.PLAIN, 12) );
 | 
						|
    currentColor = Color.white;
 | 
						|
    currentStroke = new BasicStroke();
 | 
						|
    setPaint(currentColor);
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * setupFonts - set up the font dictionaries for
 | 
						|
   * helvetica, times and courier
 | 
						|
   */
 | 
						|
  private void setupFonts()
 | 
						|
  {
 | 
						|
    out.println("/helveticaISO");
 | 
						|
    out.println("/Helvetica findfont dup length dict begin");
 | 
						|
    out.println("{ 1 index /FID eq { pop pop } { def } ifelse } forall");
 | 
						|
    out.println("/Encoding ISOLatin1Encoding def");
 | 
						|
    out.println("currentdict end definefont pop");
 | 
						|
 | 
						|
    out.println("/timesISO");
 | 
						|
    out.println("/Times-Roman findfont dup length dict begin");
 | 
						|
    out.println("{ 1 index /FID eq { pop pop } { def } ifelse } forall");
 | 
						|
    out.println("/Encoding ISOLatin1Encoding def");
 | 
						|
    out.println("currentdict end definefont pop");
 | 
						|
 | 
						|
    out.println("/courierISO");
 | 
						|
    out.println("/Courier findfont dup length dict begin");
 | 
						|
    out.println("{ 1 index /FID eq { pop pop } { def } ifelse } forall");
 | 
						|
    out.println("/Encoding ISOLatin1Encoding def");
 | 
						|
    out.println("currentdict end definefont pop");
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Spools a single page, returns NO_SUCH_PAGE unsuccessful,
 | 
						|
   * PAGE_EXISTS if it was.
 | 
						|
   */
 | 
						|
  public int spoolPage(PrintWriter out,
 | 
						|
                       Printable printable,
 | 
						|
                       PageFormat pageFormat,
 | 
						|
                       int index) throws IOException, PrinterException
 | 
						|
  {
 | 
						|
    out.println("%%BeginPageSetup");
 | 
						|
 | 
						|
    Paper p = pageFormat.getPaper();
 | 
						|
    pageX = p.getWidth();
 | 
						|
    pageY = p.getHeight();
 | 
						|
 | 
						|
    if( pageFormat.getOrientation() == PageFormat.PORTRAIT )
 | 
						|
      out.println( "%%Orientation: Portrait" );
 | 
						|
    else
 | 
						|
      {
 | 
						|
        out.println( "%%Orientation: Landscape" );
 | 
						|
        double t = pageX;
 | 
						|
        pageX = pageY;
 | 
						|
        pageY = t;
 | 
						|
      }
 | 
						|
 | 
						|
    setClip(0, 0, (int)pageX, (int)pageY);
 | 
						|
 | 
						|
    out.println("gsave % first save");
 | 
						|
 | 
						|
    // 595x842; 612x792 respectively
 | 
						|
    out.println("<< /PageSize [" +pageX + " "+pageY+ "] >> setpagedevice");
 | 
						|
 | 
						|
    if( pageFormat.getOrientation() != PageFormat.LANDSCAPE )
 | 
						|
      {
 | 
						|
        pageTransform.translate(pageX, 0);
 | 
						|
        pageTransform.scale(-1.0, 1.0);
 | 
						|
      }
 | 
						|
 | 
						|
    // save the original CTM
 | 
						|
    pushCTM();
 | 
						|
    concatCTM(pageTransform);
 | 
						|
    setTransform(new AffineTransform());
 | 
						|
 | 
						|
    out.println("%%EndPageSetup");
 | 
						|
 | 
						|
    out.println("gsave");
 | 
						|
 | 
						|
    if( printable.print(this, pageFormat, index) == Printable.NO_SUCH_PAGE )
 | 
						|
      return Printable.NO_SUCH_PAGE;
 | 
						|
 | 
						|
    out.println("grestore");
 | 
						|
    out.println("showpage");
 | 
						|
 | 
						|
    return Printable.PAGE_EXISTS;
 | 
						|
  }
 | 
						|
 | 
						|
  /** push the Current Transformation Matrix onto the PS stack */
 | 
						|
  private void pushCTM()
 | 
						|
  {
 | 
						|
    out.println("matrix currentmatrix   % pushCTM()");
 | 
						|
  }
 | 
						|
 | 
						|
  /** pop the Current Transformation Matrix from the PS stack */
 | 
						|
  private void popCTM()
 | 
						|
  {
 | 
						|
    out.println("setmatrix % restore CTM");
 | 
						|
  }
 | 
						|
 | 
						|
  ///////////////////////////////////////////////////////////////////////////
 | 
						|
 | 
						|
  public Graphics create()
 | 
						|
  {
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawOval(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    out.println("% drawOval()");
 | 
						|
    setStroke(ordinaryStroke);
 | 
						|
    draw(new Ellipse2D.Double(x, y, width, height));
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawPolyline(int[] xPoints, int[] yPoints, int nPoints)
 | 
						|
  {
 | 
						|
    if (nPoints <= 0 || xPoints.length < nPoints || yPoints.length < nPoints)
 | 
						|
      return;
 | 
						|
    out.println("newpath % drawPolyLine()");
 | 
						|
    out.println(xPoints[0] + " " + yPoints[0] + " moveto");
 | 
						|
    for (int i = 1; i < nPoints; i++)
 | 
						|
      out.println(xPoints[i] + " " + yPoints[i] + " lineto");
 | 
						|
    out.println("closepath");
 | 
						|
    out.println("stroke");
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawRoundRect(int x, int y, int width, int height, int arcWidth,
 | 
						|
                            int arcHeight)
 | 
						|
  {
 | 
						|
    out.println("% drawRoundRect()");
 | 
						|
    RoundRectangle2D.Double rr = new RoundRectangle2D.Double(x, y, width,
 | 
						|
                                                             height, arcWidth,
 | 
						|
                                                             arcHeight);
 | 
						|
    setStroke(ordinaryStroke);
 | 
						|
    draw(rr);
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  public void fillRoundRect(int x, int y, int width, int height, int arcWidth,
 | 
						|
                            int arcHeight)
 | 
						|
  {
 | 
						|
    out.println("% fillRoundRect()");
 | 
						|
    RoundRectangle2D.Double rr = new RoundRectangle2D.Double(x, y, width,
 | 
						|
                                                             height, arcWidth,
 | 
						|
                                                             arcHeight);
 | 
						|
    fill(rr);
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawArc(int x, int y, int width, int height, int startAngle,
 | 
						|
                      int arcAngle)
 | 
						|
  {
 | 
						|
    setStroke(ordinaryStroke);
 | 
						|
    draw(new Arc2D.Double(x, y, width, height, startAngle, arcAngle, Arc2D.OPEN));
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  public void fillArc(int x, int y, int width, int height, int startAngle,
 | 
						|
                      int arcAngle)
 | 
						|
  {
 | 
						|
    fill(new Arc2D.Double(x, y, width, height, startAngle, arcAngle, Arc2D.PIE));
 | 
						|
  }
 | 
						|
 | 
						|
  public void fillOval(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    out.println("% fillOval()");
 | 
						|
    fill( new Ellipse2D.Double(x, y, width, height) );
 | 
						|
  }
 | 
						|
 | 
						|
  public void fillPolygon(int[] x, int[] y, int nPoints)
 | 
						|
  {
 | 
						|
    out.println("% fillPolygon()");
 | 
						|
    fill( new Polygon(x, y, nPoints) );
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawLine(int x1, int y1, int x2, int y2)
 | 
						|
  {
 | 
						|
    out.println("% drawLine()");
 | 
						|
    setStroke(ordinaryStroke);
 | 
						|
    out.println("newpath");
 | 
						|
    out.println(x1 + " " + (y1) + " moveto");
 | 
						|
    out.println(x2 + " " + (y2) + " lineto");
 | 
						|
    out.println("stroke");
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  //--------------- Image drawing ------------------------------------------
 | 
						|
  public boolean drawImage(Image img, int x, int y, Color bgcolor,
 | 
						|
                           ImageObserver observer)
 | 
						|
  {
 | 
						|
    int w = img.getWidth(null);
 | 
						|
    int h = img.getHeight(null);
 | 
						|
 | 
						|
    return drawImage(img, x, y, x + w, y + h, 0, 0, w - 1, h - 1, bgcolor,
 | 
						|
                     observer);
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean drawImage(Image img, int dx1, int dy1, int dx2, int dy2,
 | 
						|
                           int sx1, int sy1, int sx2, int sy2, Color bgcolor,
 | 
						|
                           ImageObserver observer)
 | 
						|
  {
 | 
						|
    int n = 0;
 | 
						|
    boolean flipx = false;
 | 
						|
    boolean flipy = false;
 | 
						|
 | 
						|
    // swap X and Y's
 | 
						|
    if (sx1 > sx2)
 | 
						|
      {
 | 
						|
        n = sx1;
 | 
						|
        sx1 = sx2;
 | 
						|
        sx2 = n;
 | 
						|
        flipx = ! flipx;
 | 
						|
      }
 | 
						|
    if (sy1 > sy2)
 | 
						|
      {
 | 
						|
        n = sy1;
 | 
						|
        sy1 = sy2;
 | 
						|
        sy2 = n;
 | 
						|
        flipy = ! flipy;
 | 
						|
      }
 | 
						|
    if (dx1 > dx2)
 | 
						|
      {
 | 
						|
        n = dx1;
 | 
						|
        dx1 = dx2;
 | 
						|
        dx2 = n;
 | 
						|
        flipx = ! flipx;
 | 
						|
      }
 | 
						|
    if (dy1 > dy2)
 | 
						|
      {
 | 
						|
        n = dy1;
 | 
						|
        dy1 = dy2;
 | 
						|
        dy2 = n;
 | 
						|
        flipy = ! flipy;
 | 
						|
      }
 | 
						|
    n = 0;
 | 
						|
    int sw = sx2 - sx1; // source width
 | 
						|
    int sh = sy2 - sy1; // source height
 | 
						|
    int[] pixels = new int[sw * sh]; // pixel buffer
 | 
						|
    int dw = dx2 - dx1; // destination width
 | 
						|
    int dh = dy2 - dy1; // destination height
 | 
						|
    double x_scale = ((double) dw) / ((double) sw);
 | 
						|
    double y_scale = ((double) dh) / ((double) sh);
 | 
						|
 | 
						|
    out.println("% drawImage() 2");
 | 
						|
    out.println("gsave");
 | 
						|
    out.println(dx1 + " " + dy1 + " translate");
 | 
						|
    out.println(dw + " " + dh + " scale");
 | 
						|
    out.println(sw + " " + sh + " 8 [" + (flipx ? -sw : sw) + " 0 0 "
 | 
						|
                + (flipy ? -sh : sh) + " " + (flipx ? sw : 0) + " "
 | 
						|
                + (flipy ? sh : 0) + " ]");
 | 
						|
    out.println("{currentfile 3 string readhexstring pop} bind");
 | 
						|
    out.println("false 3 colorimage");
 | 
						|
 | 
						|
    PixelGrabber pg = new PixelGrabber(img, sx1, sy1, sw, sh, pixels, 0, sw);
 | 
						|
    try
 | 
						|
      {
 | 
						|
        pg.grabPixels();
 | 
						|
      }
 | 
						|
    catch (InterruptedException e)
 | 
						|
      {
 | 
						|
        System.err.println("interrupted waiting for pixels!");
 | 
						|
        return (false);
 | 
						|
      }
 | 
						|
 | 
						|
    if ((pg.getStatus() & ImageObserver.ABORT) != 0)
 | 
						|
      {
 | 
						|
        System.err.println("image fetch aborted or errored");
 | 
						|
        return (false);
 | 
						|
      }
 | 
						|
 | 
						|
    for (int j = 0; j < sh; j++)
 | 
						|
      {
 | 
						|
        for (int i = 0; i < sw; i++)
 | 
						|
          {
 | 
						|
            out.print(colorTripleHex(new Color(pixels[j * sw + i])));
 | 
						|
            if (((++n) % 11) == 0)
 | 
						|
              out.println();
 | 
						|
          }
 | 
						|
      }
 | 
						|
 | 
						|
    out.println();
 | 
						|
    out.println("%%EOF");
 | 
						|
    out.println("grestore");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean drawImage(Image img, int dx1, int dy1, int dx2, int dy2,
 | 
						|
                           int sx1, int sy1, int sx2, int sy2,
 | 
						|
                           ImageObserver observer)
 | 
						|
  {
 | 
						|
    return drawImage(img, dx1, dy1, dx2, dy2, sx1, sy1, sx2, sy2, null,
 | 
						|
                     observer);
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean drawImage(Image img, int x, int y, ImageObserver observer)
 | 
						|
  {
 | 
						|
    return drawImage(img, x, y, null, observer);
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean drawImage(Image img, int x, int y, int width, int height,
 | 
						|
                           Color bgcolor, ImageObserver observer)
 | 
						|
  {
 | 
						|
    int sw = img.getWidth(null);
 | 
						|
    int sh = img.getHeight(null);
 | 
						|
    return drawImage(img, x, y, x + width, y + height, /* destination */
 | 
						|
                     0, 0, sw - 1, sh - 1, /* source */
 | 
						|
                     bgcolor, observer);
 | 
						|
    // correct?
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean drawImage(Image img, int x, int y, int width, int height,
 | 
						|
                           ImageObserver observer)
 | 
						|
  {
 | 
						|
    return drawImage(img, x, y, width, height, null, observer);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders a BufferedImage that is filtered with a BufferedImageOp. */
 | 
						|
  public void drawImage(BufferedImage img, BufferedImageOp op, int x, int y)
 | 
						|
  {
 | 
						|
    BufferedImage result = op.filter(img, null);
 | 
						|
    drawImage(result, x, y, null);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders an image, applying a transform from image space
 | 
						|
      into user space before drawing. */
 | 
						|
  public boolean drawImage(Image img, AffineTransform xform, ImageObserver obs)
 | 
						|
  {
 | 
						|
    AffineTransform oldTransform = new AffineTransform(currentTransform);
 | 
						|
    boolean ret;
 | 
						|
 | 
						|
    transform(xform);
 | 
						|
    ret = drawImage(img, 0, 0, null, obs);
 | 
						|
    setTransform(oldTransform);
 | 
						|
 | 
						|
    return ret;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders a RenderableImage, applying a transform from image
 | 
						|
      space into user space before drawing. */
 | 
						|
  public void drawRenderableImage(RenderableImage img, AffineTransform xform)
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders a RenderedImage, applying a transform from
 | 
						|
      image space into user space before drawing. */
 | 
						|
  public void drawRenderedImage(RenderedImage img, AffineTransform xform)
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
  }
 | 
						|
 | 
						|
  //-------------------------------------------------------------------------
 | 
						|
  public void drawPolygon(int[] xPoints, int[] yPoints, int nPoints)
 | 
						|
  {
 | 
						|
    setStroke(ordinaryStroke);
 | 
						|
    draw(new Polygon(xPoints, yPoints, nPoints));
 | 
						|
    setStroke(currentStroke);
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawString(String str, int x, int y)
 | 
						|
  {
 | 
						|
    drawString(str, (float) x, (float) y);
 | 
						|
  }
 | 
						|
 | 
						|
  public void drawString(String str, float x, float y)
 | 
						|
  {
 | 
						|
    if( str.trim().equals("") )
 | 
						|
      return; // don't draw whitespace, silly!
 | 
						|
 | 
						|
    if( currentFontIsPS )
 | 
						|
      {
 | 
						|
        drawStringPSFont(str, x, y);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
    TextLayout text = new TextLayout(str, currentFont, getFontRenderContext());
 | 
						|
    Shape s = text.getOutline(AffineTransform.getTranslateInstance(x, y));
 | 
						|
    drawStringShape(s);
 | 
						|
  }
 | 
						|
 | 
						|
  private void drawStringPSFont(String str, float x, float y)
 | 
						|
  {
 | 
						|
    out.println("% drawString PS font");
 | 
						|
    out.println(x + " " + y + " moveto");
 | 
						|
    saveAndInvertAxis();
 | 
						|
    out.println("(" + str + ") show");
 | 
						|
    restoreAxis();
 | 
						|
  }
 | 
						|
 | 
						|
  private void saveAndInvertAxis()
 | 
						|
  {
 | 
						|
    // Invert the Y axis of the CTM.
 | 
						|
    popCTM();
 | 
						|
    pushCTM();
 | 
						|
 | 
						|
    double[] test =
 | 
						|
      {
 | 
						|
        pageTransform.getScaleX(), pageTransform.getShearY(),
 | 
						|
        pageTransform.getShearX(), pageTransform.getScaleY(),
 | 
						|
        pageTransform.getTranslateX(),
 | 
						|
        -pageTransform.getTranslateY() + pageY
 | 
						|
      };
 | 
						|
 | 
						|
    double[] test2 =
 | 
						|
      {
 | 
						|
        currentTransform.getScaleX(),
 | 
						|
        currentTransform.getShearY(),
 | 
						|
        -currentTransform.getShearX(),
 | 
						|
        -currentTransform.getScaleY(),
 | 
						|
        currentTransform.getTranslateX(),
 | 
						|
        currentTransform.getTranslateY()
 | 
						|
      };
 | 
						|
 | 
						|
    AffineTransform total = new AffineTransform(test);
 | 
						|
    total.concatenate(new AffineTransform(test2));
 | 
						|
    concatCTM(total);
 | 
						|
  }
 | 
						|
 | 
						|
  private void restoreAxis()
 | 
						|
  {
 | 
						|
    // reset the CTM
 | 
						|
    popCTM();
 | 
						|
    pushCTM();
 | 
						|
    AffineTransform total = new AffineTransform(pageTransform);
 | 
						|
    total.concatenate(currentTransform);
 | 
						|
    concatCTM(total);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * special drawing routine for string shapes,
 | 
						|
   * which need to be drawn with the Y axis uninverted.
 | 
						|
   */
 | 
						|
  private void drawStringShape(Shape s)
 | 
						|
  {
 | 
						|
    saveAndInvertAxis();
 | 
						|
 | 
						|
    // draw the shape s with an inverted Y axis.
 | 
						|
    PathIterator pi = s.getPathIterator(null);
 | 
						|
    float[] coords = new float[6];
 | 
						|
 | 
						|
    while (! pi.isDone())
 | 
						|
      {
 | 
						|
        switch (pi.currentSegment(coords))
 | 
						|
          {
 | 
						|
          case PathIterator.SEG_MOVETO:
 | 
						|
            out.println((coords[0]) + " " + (Y - coords[1]) + " moveto");
 | 
						|
            cx = coords[0];
 | 
						|
            cy = coords[1];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_LINETO:
 | 
						|
            out.println((coords[0]) + " " + (Y - coords[1]) + " lineto");
 | 
						|
            cx = coords[0];
 | 
						|
            cy = coords[1];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_QUADTO:
 | 
						|
            // convert to cubic bezier points
 | 
						|
            float x1 = (cx + 2 * coords[0]) / 3;
 | 
						|
            float y1 = (cy + 2 * coords[1]) / 3;
 | 
						|
            float x2 = (2 * coords[2] + coords[0]) / 3;
 | 
						|
            float y2 = (2 * coords[3] + coords[1]) / 3;
 | 
						|
 | 
						|
            out.print((x1) + " " + (Y - y1) + " ");
 | 
						|
            out.print((x2) + " " + (Y - y2) + " ");
 | 
						|
            out.println((coords[2]) + " " + (Y - coords[3]) + " curveto");
 | 
						|
            cx = coords[2];
 | 
						|
            cy = coords[3];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_CUBICTO:
 | 
						|
            out.print((coords[0]) + " " + (Y - coords[1]) + " ");
 | 
						|
            out.print((coords[2]) + " " + (Y - coords[3]) + " ");
 | 
						|
            out.println((coords[4]) + " " + (Y - coords[5]) + " curveto");
 | 
						|
            cx = coords[4];
 | 
						|
            cy = coords[5];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_CLOSE:
 | 
						|
            out.println("closepath");
 | 
						|
            break;
 | 
						|
          }
 | 
						|
        pi.next();
 | 
						|
      }
 | 
						|
    out.println("fill");
 | 
						|
 | 
						|
    restoreAxis();
 | 
						|
  }
 | 
						|
 | 
						|
  public void setColor(Color c)
 | 
						|
  {
 | 
						|
    /* don't set the color if it's already set */
 | 
						|
    if (c.equals(currentColor))
 | 
						|
      return;
 | 
						|
    gradientOn = false;
 | 
						|
    currentColor = c;
 | 
						|
    currentPaint = c; // Graphics2D extends colors to paint
 | 
						|
 | 
						|
    out.println(colorTriple(c) + " setrgbcolor");
 | 
						|
  }
 | 
						|
 | 
						|
  public void clearRect(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    out.println("% clearRect");
 | 
						|
    Color c = currentColor;
 | 
						|
    setColor(backgroundColor);
 | 
						|
    fill(new Rectangle2D.Double(x, y, width, height));
 | 
						|
    setColor(c);
 | 
						|
  }
 | 
						|
 | 
						|
  public void clipRect(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    clip(new Rectangle2D.Double(x, y, width, height));
 | 
						|
  }
 | 
						|
 | 
						|
  public void copyArea(int x, int y, int width, int height, int dx, int dy)
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
  }
 | 
						|
 | 
						|
  public void fillRect(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    fill(new Rectangle2D.Double(x, y, width, height));
 | 
						|
  }
 | 
						|
 | 
						|
  public void dispose()
 | 
						|
  {
 | 
						|
  }
 | 
						|
 | 
						|
  public void setClip(int x, int y, int width, int height)
 | 
						|
  {
 | 
						|
    out.println("% setClip()");
 | 
						|
    setClip(new Rectangle2D.Double(x, y, width, height));
 | 
						|
  }
 | 
						|
 | 
						|
  public void setClip(Shape s)
 | 
						|
  {
 | 
						|
    clip(s);
 | 
						|
  }
 | 
						|
 | 
						|
  public Shape getClip()
 | 
						|
  {
 | 
						|
    return clipShape;
 | 
						|
  }
 | 
						|
 | 
						|
  public Rectangle getClipBounds()
 | 
						|
  {
 | 
						|
    return clipShape.getBounds();
 | 
						|
  }
 | 
						|
 | 
						|
  public Color getColor()
 | 
						|
  {
 | 
						|
    return currentColor;
 | 
						|
  }
 | 
						|
 | 
						|
  public Font getFont()
 | 
						|
  {
 | 
						|
    return currentFont;
 | 
						|
  }
 | 
						|
 | 
						|
  public FontMetrics getFontMetrics()
 | 
						|
  {
 | 
						|
    return getFontMetrics(currentFont);
 | 
						|
  }
 | 
						|
 | 
						|
  public FontMetrics getFontMetrics(Font f)
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
 | 
						|
  public void setFont(Font font)
 | 
						|
  {
 | 
						|
    out.println("% setfont()");
 | 
						|
    if (font == null)
 | 
						|
      // use the default font
 | 
						|
      font = new Font("Dialog", Font.PLAIN, 12);
 | 
						|
    currentFont = font;
 | 
						|
    setPSFont(); // set up the PostScript fonts
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Setup the postscript font if the current font is one
 | 
						|
   */
 | 
						|
  private void setPSFont()
 | 
						|
  {
 | 
						|
    currentFontIsPS = false;
 | 
						|
 | 
						|
    String s = currentFont.getName();
 | 
						|
    out.println("% setPSFont: Fontname: " + s);
 | 
						|
    if (s.equalsIgnoreCase("Helvetica") || s.equalsIgnoreCase("SansSerif"))
 | 
						|
      out.print("/helveticaISO findfont ");
 | 
						|
    else if (s.equalsIgnoreCase("Times New Roman"))
 | 
						|
      out.print("/timesISO findfont ");
 | 
						|
    else if (s.equalsIgnoreCase("Courier"))
 | 
						|
      out.print("/courierISO findfont ");
 | 
						|
    else
 | 
						|
      return;
 | 
						|
 | 
						|
    currentFontIsPS = true;
 | 
						|
 | 
						|
    out.print(currentFont.getSize() + " scalefont ");
 | 
						|
    out.println("setfont");
 | 
						|
  }
 | 
						|
 | 
						|
  /** XOR mode is not supported */
 | 
						|
  public void setPaintMode()
 | 
						|
  {
 | 
						|
  }
 | 
						|
 | 
						|
  /** XOR mode is not supported */
 | 
						|
  public void setXORMode(Color c1)
 | 
						|
  {
 | 
						|
  }
 | 
						|
 | 
						|
  public void close()
 | 
						|
  {
 | 
						|
    out.println("showpage");
 | 
						|
    out.println("%%Trailer");
 | 
						|
    out.println("grestore % restore original stuff");
 | 
						|
    out.println("%%EOF");
 | 
						|
 | 
						|
    try
 | 
						|
      {
 | 
						|
        out.close();
 | 
						|
      }
 | 
						|
    catch (Exception e)
 | 
						|
      {
 | 
						|
      }
 | 
						|
    out = null;
 | 
						|
  }
 | 
						|
 | 
						|
  //----------------------------------------------------------------
 | 
						|
  // Graphics2D stuff ----------------------------------------------
 | 
						|
 | 
						|
  /**  Sets the values of an arbitrary number of
 | 
						|
       preferences for the rendering algorithms. */
 | 
						|
  public void addRenderingHints(Map hints)
 | 
						|
  {
 | 
						|
    /* rendering hint changes are disallowed */
 | 
						|
  }
 | 
						|
 | 
						|
  /** write a shape to the file */
 | 
						|
  private void writeShape(Shape s)
 | 
						|
  {
 | 
						|
    PathIterator pi = s.getPathIterator(null);
 | 
						|
    float[] coords = new float[6];
 | 
						|
 | 
						|
    while (! pi.isDone())
 | 
						|
      {
 | 
						|
        switch (pi.currentSegment(coords))
 | 
						|
          {
 | 
						|
          case PathIterator.SEG_MOVETO:
 | 
						|
            out.println(coords[0] + " " + (coords[1]) + " moveto");
 | 
						|
            cx = coords[0];
 | 
						|
            cy = coords[1];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_LINETO:
 | 
						|
            out.println(coords[0] + " " + (coords[1]) + " lineto");
 | 
						|
            cx = coords[0];
 | 
						|
            cy = coords[1];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_QUADTO:
 | 
						|
            // convert to cubic bezier points
 | 
						|
            float x1 = (cx + 2 * coords[0]) / 3;
 | 
						|
            float y1 = (cy + 2 * coords[1]) / 3;
 | 
						|
            float x2 = (2 * coords[2] + coords[0]) / 3;
 | 
						|
            float y2 = (2 * coords[3] + coords[1]) / 3;
 | 
						|
 | 
						|
            out.print(x1 + " " + (Y - y1) + " ");
 | 
						|
            out.print(x2 + " " + (Y - y2) + " ");
 | 
						|
            out.println(coords[2] + " " + (Y - coords[3]) + " curveto");
 | 
						|
            cx = coords[2];
 | 
						|
            cy = coords[3];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_CUBICTO:
 | 
						|
            out.print(coords[0] + " " + coords[1] + " ");
 | 
						|
            out.print(coords[2] + " " + coords[3] + " ");
 | 
						|
            out.println(coords[4] + " " + coords[5] + " curveto");
 | 
						|
            cx = coords[4];
 | 
						|
            cy = coords[5];
 | 
						|
            break;
 | 
						|
          case PathIterator.SEG_CLOSE:
 | 
						|
            out.println("closepath");
 | 
						|
            break;
 | 
						|
          }
 | 
						|
        pi.next();
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /** Intersects the current Clip with the interior of
 | 
						|
      the specified Shape and sets the Clip to the resulting intersection. */
 | 
						|
  public void clip(Shape s)
 | 
						|
  {
 | 
						|
    clipShape = s;
 | 
						|
    out.println("% clip INACTIVE");
 | 
						|
    //  writeShape(s);
 | 
						|
    //  out.println("clip");
 | 
						|
  }
 | 
						|
 | 
						|
  /** Strokes the outline of a Shape using the
 | 
						|
      settings of the current Graphics2D context.*/
 | 
						|
  public void draw(Shape s)
 | 
						|
  {
 | 
						|
    if(!(currentStroke instanceof BasicStroke))
 | 
						|
      fill(currentStroke.createStrokedShape(s));
 | 
						|
 | 
						|
    out.println("% draw");
 | 
						|
    writeShape(s);
 | 
						|
    out.println("stroke");
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders the text of the specified GlyphVector using the
 | 
						|
      Graphics2D context's rendering attributes. */
 | 
						|
  public void drawGlyphVector(GlyphVector gv, float x, float y)
 | 
						|
  {
 | 
						|
    out.println("% drawGlyphVector");
 | 
						|
    Shape s = gv.getOutline();
 | 
						|
    drawStringShape(AffineTransform.getTranslateInstance(x, y)
 | 
						|
                    .createTransformedShape(s));
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders the text of the specified iterator,
 | 
						|
      using the Graphics2D context's current Paint.*/
 | 
						|
  public void drawString(AttributedCharacterIterator iterator, float x, float y)
 | 
						|
  {
 | 
						|
    TextLayout text = new TextLayout(iterator, getFontRenderContext());
 | 
						|
    Shape s = text.getOutline(AffineTransform.getTranslateInstance(x, y));
 | 
						|
    drawStringShape(s);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Renders the text of the specified iterator,
 | 
						|
      using the Graphics2D context's current Paint. */
 | 
						|
  public void drawString(AttributedCharacterIterator iterator, int x, int y)
 | 
						|
  {
 | 
						|
    drawString(iterator, (float) x, (float) y);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Fills the interior of a Shape using the settings of the Graphics2D context. */
 | 
						|
  public void fill(Shape s)
 | 
						|
  {
 | 
						|
    out.println("% fill");
 | 
						|
    if (! gradientOn)
 | 
						|
      {
 | 
						|
        writeShape(s);
 | 
						|
        out.println("fill");
 | 
						|
      }
 | 
						|
    else
 | 
						|
      {
 | 
						|
        out.println("gsave");
 | 
						|
        writeShape(s);
 | 
						|
        out.println("clip");
 | 
						|
        writeGradient();
 | 
						|
        out.println("shfill");
 | 
						|
        out.println("grestore");
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the background color used for clearing a region. */
 | 
						|
  public Color getBackground()
 | 
						|
  {
 | 
						|
    return backgroundColor;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the current Composite in the Graphics2D context. */
 | 
						|
  public Composite getComposite()
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the device configuration associated with this Graphics2D. */
 | 
						|
  public GraphicsConfiguration getDeviceConfiguration()
 | 
						|
  {
 | 
						|
    // FIXME
 | 
						|
    out.println("% getDeviceConfiguration()");
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Get the rendering context of the Font within this Graphics2D context. */
 | 
						|
  public FontRenderContext getFontRenderContext()
 | 
						|
  {
 | 
						|
    out.println("% getFontRenderContext()");
 | 
						|
 | 
						|
    double[] scaling =
 | 
						|
      {
 | 
						|
        pageTransform.getScaleX(), 0, 0,
 | 
						|
        -pageTransform.getScaleY(), 0, 0
 | 
						|
      };
 | 
						|
 | 
						|
    return (new FontRenderContext(new AffineTransform(scaling), false, true));
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the current Paint of the Graphics2D context. */
 | 
						|
  public Paint getPaint()
 | 
						|
  {
 | 
						|
    return currentPaint;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the value of a single preference for the rendering algorithms. */
 | 
						|
  public Object getRenderingHint(RenderingHints.Key hintKey)
 | 
						|
  {
 | 
						|
    return renderingHints.get(hintKey);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Gets the preferences for the rendering algorithms. */
 | 
						|
  public RenderingHints getRenderingHints()
 | 
						|
  {
 | 
						|
    return renderingHints;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns the current Stroke in the Graphics2D context. */
 | 
						|
  public Stroke getStroke()
 | 
						|
  {
 | 
						|
    return currentStroke;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Returns a copy of the current Transform in the Graphics2D context. */
 | 
						|
  public AffineTransform getTransform()
 | 
						|
  {
 | 
						|
    return currentTransform;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Checks whether or not the specified Shape intersects
 | 
						|
   * the specified Rectangle, which is in device space.
 | 
						|
   */
 | 
						|
  public boolean hit(Rectangle rect, Shape s, boolean onStroke)
 | 
						|
  {
 | 
						|
    Rectangle2D.Double r = new Rectangle2D.Double(rect.getX(), rect.getY(),
 | 
						|
                                                  rect.getWidth(),
 | 
						|
                                                  rect.getHeight());
 | 
						|
    return s.intersects(r);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Sets the background color for the Graphics2D context.*/
 | 
						|
  public void setBackground(Color color)
 | 
						|
  {
 | 
						|
    out.println("% setBackground(" + color + ")");
 | 
						|
    backgroundColor = color;
 | 
						|
  }
 | 
						|
 | 
						|
  /** Sets the Composite for the Graphics2D context.
 | 
						|
      Not supported. */
 | 
						|
  public void setComposite(Composite comp)
 | 
						|
  {
 | 
						|
  }
 | 
						|
 | 
						|
  /** Sets the Paint attribute for the Graphics2D context.*/
 | 
						|
  public void setPaint(Paint paint)
 | 
						|
  {
 | 
						|
    currentPaint = paint;
 | 
						|
    gradientOn = false;
 | 
						|
    if (paint instanceof Color)
 | 
						|
      {
 | 
						|
        setColor((Color) paint);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    if (paint instanceof GradientPaint)
 | 
						|
      {
 | 
						|
        gradientOn = true;
 | 
						|
        return;
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /* get a space seperated 0.0 - 1.0 color RGB triple */
 | 
						|
  private String colorTriple(Color c)
 | 
						|
  {
 | 
						|
    return (((double) c.getRed() / 255.0) + " "
 | 
						|
            + ((double) c.getGreen() / 255.0) + " "
 | 
						|
            + ((double) c.getBlue() / 255.0));
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Get a nonsperated hex RGB triple, eg FFFFFF = white
 | 
						|
   * used by writeGradient and drawImage
 | 
						|
   */
 | 
						|
  private String colorTripleHex(Color c)
 | 
						|
  {
 | 
						|
    String r = "00" + Integer.toHexString(c.getRed());
 | 
						|
    r = r.substring(r.length() - 2);
 | 
						|
    String g = "00" + Integer.toHexString(c.getGreen());
 | 
						|
    g = g.substring(g.length() - 2);
 | 
						|
    String b = "00" + Integer.toHexString(c.getBlue());
 | 
						|
    b = b.substring(b.length() - 2);
 | 
						|
    return r + g + b;
 | 
						|
  }
 | 
						|
 | 
						|
  /* write the current gradient fill */
 | 
						|
  private void writeGradient()
 | 
						|
  {
 | 
						|
    GradientPaint paint = (GradientPaint) currentPaint;
 | 
						|
    out.println("% writeGradient()");
 | 
						|
 | 
						|
    int n = 1;
 | 
						|
    double x;
 | 
						|
    double y;
 | 
						|
    double dx;
 | 
						|
    double dy;
 | 
						|
    Point2D p1 = currentTransform.transform(paint.getPoint1(), null);
 | 
						|
    Point2D p2 = currentTransform.transform(paint.getPoint2(), null);
 | 
						|
    x = p1.getX();
 | 
						|
    y = p1.getY();
 | 
						|
    dx = p2.getX() - x;
 | 
						|
    dy = p2.getY() - y;
 | 
						|
 | 
						|
    // get number of repetitions
 | 
						|
    while (x + n * dx < pageY && y + n * dy < pageX && x + n * dx > 0
 | 
						|
           && y + n * dy > 0)
 | 
						|
      n++;
 | 
						|
 | 
						|
    out.println("<<"); // start
 | 
						|
    out.println("/ShadingType 2"); // gradient fill
 | 
						|
    out.println("/ColorSpace [ /DeviceRGB ]"); // RGB colors
 | 
						|
    out.print("/Coords [");
 | 
						|
    out.print(x + " " + y + " " + (x + n * dx) + " " + (y + n * dy) + " ");
 | 
						|
    out.println("]"); // coordinates defining the axis
 | 
						|
    out.println("/Function <<");
 | 
						|
    out.println("/FunctionType 0");
 | 
						|
    out.println("/Order 1");
 | 
						|
    out.println("/Domain [ 0 1 ]");
 | 
						|
    out.println("/Range [ 0 1  0 1  0 1 ]");
 | 
						|
    out.println("/BitsPerSample 8");
 | 
						|
    out.println("/Size [ " + (1 + n) + " ]");
 | 
						|
    out.print("/DataSource < " + colorTripleHex(paint.getColor1()) + " "
 | 
						|
              + colorTripleHex(paint.getColor2()) + " ");
 | 
						|
    for (; n > 1; n--)
 | 
						|
      if (paint.isCyclic())
 | 
						|
        {
 | 
						|
          if ((n % 2) == 1)
 | 
						|
            out.print(colorTripleHex(paint.getColor1()) + " ");
 | 
						|
          else
 | 
						|
            out.print(colorTripleHex(paint.getColor2()) + " ");
 | 
						|
        }
 | 
						|
      else
 | 
						|
        out.print(colorTripleHex(paint.getColor2()) + " ");
 | 
						|
    out.println(">");
 | 
						|
    out.println(">>");
 | 
						|
    out.println(">>");
 | 
						|
  }
 | 
						|
 | 
						|
  /** Sets the value of a single preference for the rendering algorithms. */
 | 
						|
  public void setRenderingHint(RenderingHints.Key hintKey, Object hintValue)
 | 
						|
  {
 | 
						|
    /* we don't allow the changing of rendering hints. */
 | 
						|
  }
 | 
						|
 | 
						|
  /** Replaces the values of all preferences for the rendering algorithms
 | 
						|
      with the specified hints. */
 | 
						|
  public void setRenderingHints(Map hints)
 | 
						|
  {
 | 
						|
    /* we don't allow the changing of rendering hints. */
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Sets the Stroke for the Graphics2D context. BasicStroke fully implemented.
 | 
						|
   */
 | 
						|
  public void setStroke(Stroke s)
 | 
						|
  {
 | 
						|
    currentStroke = s;
 | 
						|
 | 
						|
    if (! (s instanceof BasicStroke))
 | 
						|
      return;
 | 
						|
 | 
						|
    BasicStroke bs = (BasicStroke) s;
 | 
						|
    out.println("% setStroke()");
 | 
						|
    try
 | 
						|
      {
 | 
						|
        // set the line width
 | 
						|
        out.println(bs.getLineWidth() + " setlinewidth");
 | 
						|
 | 
						|
        // set the line dash
 | 
						|
        float[] dashArray = bs.getDashArray();
 | 
						|
        if (dashArray != null)
 | 
						|
          {
 | 
						|
            out.print("[ ");
 | 
						|
            for (int i = 0; i < dashArray.length; i++)
 | 
						|
              out.print(dashArray[i] + " ");
 | 
						|
            out.println("] " + bs.getDashPhase() + " setdash");
 | 
						|
          }
 | 
						|
        else
 | 
						|
          out.println("[] 0 setdash"); // set solid
 | 
						|
 | 
						|
        // set the line cap
 | 
						|
        switch (bs.getEndCap())
 | 
						|
          {
 | 
						|
          case BasicStroke.CAP_BUTT:
 | 
						|
            out.println("0 setlinecap");
 | 
						|
            break;
 | 
						|
          case BasicStroke.CAP_ROUND:
 | 
						|
            out.println("1 setlinecap");
 | 
						|
            break;
 | 
						|
          case BasicStroke.CAP_SQUARE:
 | 
						|
            out.println("2 setlinecap");
 | 
						|
            break;
 | 
						|
          }
 | 
						|
 | 
						|
        // set the line join
 | 
						|
        switch (bs.getLineJoin())
 | 
						|
          {
 | 
						|
          case BasicStroke.JOIN_BEVEL:
 | 
						|
            out.println("2 setlinejoin");
 | 
						|
            break;
 | 
						|
          case BasicStroke.JOIN_MITER:
 | 
						|
            out.println("0 setlinejoin");
 | 
						|
            out.println(bs.getMiterLimit() + " setmiterlimit");
 | 
						|
            break;
 | 
						|
          case BasicStroke.JOIN_ROUND:
 | 
						|
            out.println("1 setlinejoin");
 | 
						|
            break;
 | 
						|
          }
 | 
						|
      }
 | 
						|
    catch (Exception e)
 | 
						|
      {
 | 
						|
        out.println("% Exception in setStroke()");
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  //////////////////// TRANSFORM SETTING /////////////////////////////////////
 | 
						|
  private void concatCTM(AffineTransform Tx)
 | 
						|
  {
 | 
						|
    double[] matrixElements = new double[6];
 | 
						|
    Tx.getMatrix(matrixElements);
 | 
						|
 | 
						|
    out.print("[ ");
 | 
						|
    for (int i = 0; i < 6; i++)
 | 
						|
      out.print(matrixElements[i] + " ");
 | 
						|
    out.println("] concat");
 | 
						|
  }
 | 
						|
 | 
						|
  /** Sets the Transform in the Graphics2D context. */
 | 
						|
  public void setTransform(AffineTransform Tx)
 | 
						|
  {
 | 
						|
    // set the transformation matrix;
 | 
						|
    currentTransform = Tx;
 | 
						|
 | 
						|
    // concatenate the current transform and the page transform
 | 
						|
    AffineTransform totalTransform = new AffineTransform(pageTransform);
 | 
						|
    totalTransform.concatenate(currentTransform);
 | 
						|
    out.println("% setTransform()");
 | 
						|
    out.println("% pageTransform:" + pageTransform);
 | 
						|
    out.println("% currentTransform:" + currentTransform);
 | 
						|
    out.println("% totalTransform:" + totalTransform);
 | 
						|
 | 
						|
    popCTM();
 | 
						|
    pushCTM(); // set the CTM to it's original state
 | 
						|
    concatCTM(totalTransform); // apply our transforms
 | 
						|
  }
 | 
						|
 | 
						|
  /** Composes an AffineTransform object with the Transform
 | 
						|
      in this Graphics2D according to the rule last-specified-first-applied. */
 | 
						|
  public void transform(AffineTransform Tx)
 | 
						|
  {
 | 
						|
    // concatenate the current transform
 | 
						|
    currentTransform.concatenate(Tx);
 | 
						|
    // and the PS CTM
 | 
						|
    concatCTM(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  ////////////////////////// TRANSFORMS //////////////////////////////////////
 | 
						|
 | 
						|
  /** shear transform */
 | 
						|
  public void shear(double shx, double shy)
 | 
						|
  {
 | 
						|
    out.println("% shear()");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.shear(shx, shy);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Translates the origin of the Graphics2D context
 | 
						|
      to the point (x, y) in the current coordinate system. */
 | 
						|
  public void translate(int x, int y)
 | 
						|
  {
 | 
						|
    out.println("% translate()");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.translate(x, y);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Translates the origin of the Graphics2D context
 | 
						|
      to the point (x, y) in the current coordinate system. */
 | 
						|
  public void translate(double x, double y)
 | 
						|
  {
 | 
						|
    out.println("% translate(" + x + ", " + y + ")");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.translate(x, y);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Concatenates the current Graphics2D Transform with a rotation transform.*/
 | 
						|
  public void rotate(double theta)
 | 
						|
  {
 | 
						|
    out.println("% rotate(" + theta + ")");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.rotate(theta);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Concatenates the current Graphics2D Transform with
 | 
						|
      a translated rotation transform.*/
 | 
						|
  public void rotate(double theta, double x, double y)
 | 
						|
  {
 | 
						|
    out.println("% rotate()");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.rotate(theta, x, y);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
 | 
						|
  /** Concatenates the current Graphics2D Transform with a scaling
 | 
						|
      transformation Subsequent rendering is resized according to the
 | 
						|
      specified scaling factors relative to the previous scaling.*/
 | 
						|
  public void scale(double sx, double sy)
 | 
						|
  {
 | 
						|
    out.println("% scale(" + sx + ", " + sy + ")");
 | 
						|
    AffineTransform Tx = new AffineTransform();
 | 
						|
    Tx.scale(sx, sy);
 | 
						|
    transform(Tx);
 | 
						|
  }
 | 
						|
}
 |