519 lines
13 KiB
Java
519 lines
13 KiB
Java
![]() |
/* JavaPrinterGraphics.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 gnu.java.awt.peer.gtk.CairoSurface;
|
||
|
|
||
|
import java.awt.Color;
|
||
|
import java.awt.Font;
|
||
|
import java.awt.FontMetrics;
|
||
|
import java.awt.Graphics;
|
||
|
import java.awt.Image;
|
||
|
import java.awt.Rectangle;
|
||
|
import java.awt.Shape;
|
||
|
import java.awt.geom.AffineTransform;
|
||
|
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.PrinterGraphics;
|
||
|
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;
|
||
|
|
||
|
/**
|
||
|
* Graphics context to draw to PostScript.
|
||
|
*
|
||
|
* @author Sven de Marothy
|
||
|
*/
|
||
|
public class JavaPrinterGraphics extends Graphics implements PrinterGraphics
|
||
|
{
|
||
|
|
||
|
/**
|
||
|
* The used graphics context.
|
||
|
*/
|
||
|
private Graphics g;
|
||
|
|
||
|
/**
|
||
|
* The associated printer job.
|
||
|
*/
|
||
|
private PrinterJob printerJob;
|
||
|
|
||
|
/**
|
||
|
* Rendering resolution
|
||
|
*/
|
||
|
private static final double DPI = 72.0;
|
||
|
|
||
|
/**
|
||
|
* Rendered image size.
|
||
|
*/
|
||
|
private int xSize, ySize;
|
||
|
|
||
|
/**
|
||
|
* The image to render to.
|
||
|
*/
|
||
|
private Image image;
|
||
|
|
||
|
public JavaPrinterGraphics( PrinterJob printerJob )
|
||
|
{
|
||
|
this.printerJob = printerJob;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 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();
|
||
|
|
||
|
PrintWriter out = new PrintWriter
|
||
|
(new BufferedWriter
|
||
|
(new OutputStreamWriter
|
||
|
(new FileOutputStream(temp), "ISO8859_1"), 1000000));
|
||
|
|
||
|
writePSHeader(out);
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 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
|
||
|
{
|
||
|
initImage( pageFormat );
|
||
|
if(printable.print(this, pageFormat, index) == Printable.NO_SUCH_PAGE)
|
||
|
return Printable.NO_SUCH_PAGE;
|
||
|
g.dispose();
|
||
|
g = null;
|
||
|
writePage( out, pageFormat );
|
||
|
return Printable.PAGE_EXISTS;
|
||
|
}
|
||
|
|
||
|
private void initImage(PageFormat pageFormat)
|
||
|
{
|
||
|
// Create a really big image and draw to that.
|
||
|
xSize = (int)(DPI*pageFormat.getWidth()/72.0);
|
||
|
ySize = (int)(DPI*pageFormat.getHeight()/72.0);
|
||
|
|
||
|
// Swap X and Y sizes if it's a Landscape page.
|
||
|
if( pageFormat.getOrientation() != PageFormat.PORTRAIT )
|
||
|
{
|
||
|
int t = xSize;
|
||
|
xSize = ySize;
|
||
|
ySize = t;
|
||
|
}
|
||
|
|
||
|
// FIXME: This should at least be BufferedImage.
|
||
|
// Fix once we have a working B.I.
|
||
|
// Graphics2D should also be supported of course.
|
||
|
image = CairoSurface.getBufferedImage(xSize, ySize);
|
||
|
|
||
|
g = image.getGraphics();
|
||
|
setColor(Color.white);
|
||
|
fillRect(0, 0, xSize, ySize);
|
||
|
setColor(Color.black);
|
||
|
}
|
||
|
|
||
|
private void writePSHeader(PrintWriter out)
|
||
|
{
|
||
|
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("%%Pages: "+); // FIXME # pages.
|
||
|
out.println("%%EndComments");
|
||
|
|
||
|
out.println("%%BeginProlog");
|
||
|
out.println("%%EndProlog");
|
||
|
out.println("%%BeginSetup");
|
||
|
|
||
|
// FIXME: Paper name
|
||
|
// E.g. "A4" "Letter"
|
||
|
// out.println("%%BeginFeature: *PageSize A4");
|
||
|
|
||
|
out.println("%%EndFeature");
|
||
|
|
||
|
out.println("%%EndSetup");
|
||
|
|
||
|
// out.println("%%Page: 1 1");
|
||
|
}
|
||
|
|
||
|
private void writePage(PrintWriter out, PageFormat pageFormat)
|
||
|
{
|
||
|
out.println("%%BeginPageSetup");
|
||
|
|
||
|
Paper p = pageFormat.getPaper();
|
||
|
double pWidth = p.getWidth();
|
||
|
double pHeight = p.getHeight();
|
||
|
|
||
|
if( pageFormat.getOrientation() == PageFormat.PORTRAIT )
|
||
|
out.println( "%%Orientation: Portrait" );
|
||
|
else
|
||
|
{
|
||
|
out.println( "%%Orientation: Landscape" );
|
||
|
double t = pWidth;
|
||
|
pWidth = pHeight;
|
||
|
pHeight = t;
|
||
|
}
|
||
|
|
||
|
out.println("gsave % first save");
|
||
|
|
||
|
// 595x842; 612x792 respectively
|
||
|
out.println("<< /PageSize [" +pWidth + " "+pHeight+ "] >> setpagedevice");
|
||
|
|
||
|
// invert the Y axis so that we get screen-like coordinates instead.
|
||
|
AffineTransform pageTransform = new AffineTransform();
|
||
|
if( pageFormat.getOrientation() == PageFormat.REVERSE_LANDSCAPE )
|
||
|
{
|
||
|
pageTransform.translate(pWidth, pHeight);
|
||
|
pageTransform.scale(-1.0, -1.0);
|
||
|
}
|
||
|
concatCTM(out, pageTransform);
|
||
|
out.println("%%EndPageSetup");
|
||
|
|
||
|
out.println("gsave");
|
||
|
|
||
|
|
||
|
// Draw the image
|
||
|
out.println(xSize+" "+ySize+" 8 [1 0 0 -1 0 "+ySize+" ]");
|
||
|
out.println("{currentfile 3 string readhexstring pop} bind");
|
||
|
out.println("false 3 colorimage");
|
||
|
int[] pixels = new int[xSize * ySize];
|
||
|
PixelGrabber pg = new PixelGrabber(image, 0, 0, xSize, ySize, pixels, 0, xSize);
|
||
|
|
||
|
try {
|
||
|
pg.grabPixels();
|
||
|
} catch (InterruptedException e) {
|
||
|
out.println("% Bug getting pixels!");
|
||
|
}
|
||
|
|
||
|
int n = 0;
|
||
|
for (int j = 0; j < ySize; j++) {
|
||
|
for (int i = 0; i < xSize; i++) {
|
||
|
out.print( colorTripleHex(pixels[j * xSize + i]) );
|
||
|
if(((++n)%11) == 0) out.println();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
out.println();
|
||
|
out.println("%%EOF");
|
||
|
out.println("grestore");
|
||
|
out.println("showpage");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get a nonsperated hex RGB triple, e.g. FFFFFF = white
|
||
|
*/
|
||
|
private String colorTripleHex(int num){
|
||
|
String s = "";
|
||
|
|
||
|
try {
|
||
|
s = Integer.toHexString( ( num & 0x00FFFFFF ) );
|
||
|
if( s.length() < 6 )
|
||
|
{
|
||
|
s = "000000"+s;
|
||
|
return s.substring(s.length()-6);
|
||
|
}
|
||
|
} catch (Exception e){
|
||
|
s = "FFFFFF";
|
||
|
}
|
||
|
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
private void concatCTM(PrintWriter out, 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");
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
/**
|
||
|
* PrinterGraphics method - Returns the printer job associated with this object.
|
||
|
*/
|
||
|
public PrinterJob getPrinterJob()
|
||
|
{
|
||
|
return printerJob;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The rest of the methods here are just pass-throughs to g.
|
||
|
*/
|
||
|
public void clearRect(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.clearRect(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void clipRect(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.clipRect(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void copyArea(int x, int y, int width, int height, int dx, int dy)
|
||
|
{
|
||
|
g.copyArea(x, y, width, height, dx, dy);
|
||
|
}
|
||
|
|
||
|
public Graphics create()
|
||
|
{
|
||
|
return g.create();
|
||
|
}
|
||
|
|
||
|
public void dispose()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public void drawArc(int x, int y, int width, int height, int startAngle,
|
||
|
int arcAngle)
|
||
|
{
|
||
|
g.drawArc(x, y, width, height, startAngle, arcAngle);
|
||
|
}
|
||
|
|
||
|
public boolean drawImage(Image img, int x, int y, Color bgcolor,
|
||
|
ImageObserver observer)
|
||
|
{
|
||
|
return g.drawImage(img, x, y, bgcolor, observer);
|
||
|
}
|
||
|
|
||
|
public boolean drawImage(Image img, int x, int y, ImageObserver observer)
|
||
|
{
|
||
|
return g.drawImage(img, x, y, observer);
|
||
|
}
|
||
|
|
||
|
public boolean drawImage(Image img, int x, int y, int width, int height,
|
||
|
Color bgcolor, ImageObserver observer)
|
||
|
{
|
||
|
return g.drawImage(img, x, y, width, height, bgcolor, observer);
|
||
|
}
|
||
|
|
||
|
public boolean drawImage(Image img, int x, int y, int width, int height,
|
||
|
ImageObserver observer)
|
||
|
{
|
||
|
return g.drawImage(img, x, y, width, height, 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)
|
||
|
{
|
||
|
return g.drawImage(img, dx1, dy1, dx2, dy2,
|
||
|
sx1, sy1, sx2, sy2, bgcolor, observer);
|
||
|
}
|
||
|
|
||
|
public boolean drawImage(Image img, int dx1, int dy1, int dx2, int dy2,
|
||
|
int sx1, int sy1, int sx2, int sy2, ImageObserver observer)
|
||
|
{
|
||
|
return g.drawImage(img, dx1, dy1, dx2, dy2,
|
||
|
sx1, sy1, sx2, sy2, observer);
|
||
|
}
|
||
|
|
||
|
public void drawLine(int x1, int y1, int x2, int y2)
|
||
|
{
|
||
|
g.drawLine(x1, y1, x2, y2);
|
||
|
}
|
||
|
|
||
|
public void drawOval(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.drawOval(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void drawPolygon(int[] xPoints, int[] yPoints, int nPoints)
|
||
|
{
|
||
|
g.drawPolygon(xPoints, yPoints, nPoints);
|
||
|
}
|
||
|
|
||
|
public void drawPolyline(int[] xPoints, int[] yPoints, int nPoints)
|
||
|
{
|
||
|
g.drawPolyline(xPoints, yPoints, nPoints);
|
||
|
}
|
||
|
|
||
|
public void drawRoundRect(int x, int y, int width, int height,
|
||
|
int arcWidth, int arcHeight)
|
||
|
{
|
||
|
g.drawRoundRect(x, y, width, height, arcWidth, arcHeight);
|
||
|
}
|
||
|
|
||
|
public void drawString(AttributedCharacterIterator iterator, int x, int y)
|
||
|
{
|
||
|
g.drawString(iterator, x, y);
|
||
|
}
|
||
|
|
||
|
public void drawString(String str, int x, int y)
|
||
|
{
|
||
|
g.drawString(str, x, y);
|
||
|
}
|
||
|
|
||
|
public void fillArc(int x, int y, int width, int height,
|
||
|
int startAngle, int arcAngle)
|
||
|
{
|
||
|
g.fillArc(x, y, width, height, startAngle, arcAngle);
|
||
|
}
|
||
|
|
||
|
public void fillOval(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.fillOval(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void fillPolygon(int[] xPoints, int[] yPoints, int nPoints)
|
||
|
{
|
||
|
g.fillPolygon(xPoints, yPoints, nPoints);
|
||
|
}
|
||
|
|
||
|
public void fillRect(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.fillRect(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void fillRoundRect(int x, int y, int width, int height,
|
||
|
int arcWidth, int arcHeight)
|
||
|
{
|
||
|
g.fillRoundRect(x, y, width, height, arcWidth, arcHeight);
|
||
|
}
|
||
|
|
||
|
public Shape getClip()
|
||
|
{
|
||
|
return g.getClip();
|
||
|
}
|
||
|
|
||
|
public Rectangle getClipBounds()
|
||
|
{
|
||
|
return g.getClipBounds();
|
||
|
}
|
||
|
|
||
|
public Color getColor()
|
||
|
{
|
||
|
return g.getColor();
|
||
|
}
|
||
|
|
||
|
public Font getFont()
|
||
|
{
|
||
|
return g.getFont();
|
||
|
}
|
||
|
|
||
|
public FontMetrics getFontMetrics(Font f)
|
||
|
{
|
||
|
return g.getFontMetrics(f);
|
||
|
}
|
||
|
|
||
|
public void setClip(int x, int y, int width, int height)
|
||
|
{
|
||
|
g.setClip(x, y, width, height);
|
||
|
}
|
||
|
|
||
|
public void setClip(Shape clip)
|
||
|
{
|
||
|
g.setClip(clip);
|
||
|
}
|
||
|
|
||
|
public void setColor(Color c)
|
||
|
{
|
||
|
g.setColor(c);
|
||
|
}
|
||
|
|
||
|
public void setFont(Font font)
|
||
|
{
|
||
|
g.setFont(font);
|
||
|
}
|
||
|
|
||
|
public void setPaintMode()
|
||
|
{
|
||
|
g.setPaintMode();
|
||
|
}
|
||
|
|
||
|
public void setXORMode(Color c1)
|
||
|
{
|
||
|
g.setXORMode(c1);
|
||
|
}
|
||
|
|
||
|
public void translate(int x, int y)
|
||
|
{
|
||
|
g.translate(x, y);
|
||
|
}
|
||
|
}
|
||
|
|