
libjava/ 2008-06-28 Matthias Klose <doko@ubuntu.com> Import GNU Classpath (classpath-0_97_2-release). * Regenerate class and header files. * Regenerate auto* files. * gcj/javaprims.h: Define jobjectRefType. * jni.cc (_Jv_JNI_GetObjectRefType): New (stub only). (_Jv_JNIFunctions): Initialize GetObjectRefType. * gnu/classpath/jdwp/VMVirtualMachine.java, java/security/VMSecureRandom.java: Merge from classpath. * HACKING: Fix typo. * ChangeLog-2007: New file. * configure.ac: Set JAVAC, pass --disable-regen-headers to classpath. libjava/classpath/ 2008-06-28 Matthias Klose <doko@ubuntu.com> * m4/ac_prog_javac.m4: Disable check for JAVAC, when not configured with --enable-java-maintainer-mode. * aclocal.m4, configure: Regenerate. * native/jni/gstreamer-peer/Makefile.am: Do not link with libclasspathnative. * native/jni/gstreamer-peer/Makefile.in: Regenerate. * tools/Makefile.am, lib/Makefile.am: Use JAVAC for setting JCOMPILER, drop flags not understood by gcj. From-SVN: r137223
1349 lines
37 KiB
Java
1349 lines
37 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);
|
|
}
|
|
}
|