org.apache.xmlgraphics.java2d.ps.AbstractPSDocumentGraphics2D Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/* $Id$ */
package org.apache.xmlgraphics.java2d.ps;
import java.awt.Color;
import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.io.IOException;
import java.io.OutputStream;
import org.apache.xmlgraphics.ps.DSCConstants;
import org.apache.xmlgraphics.ps.PSGenerator;
import org.apache.xmlgraphics.ps.PSProcSets;
/**
* This class is a wrapper for the PSGraphics2D that
* is used to create a full document around the PostScript rendering from
* PSGraphics2D.
*
* @version $Id$
* @see org.apache.xmlgraphics.java2d.ps.PSGraphics2D
*
* Originally authored by Keiron Liddle.
*/
public abstract class AbstractPSDocumentGraphics2D extends PSGraphics2D {
protected static final Integer ZERO = 0;
protected int width;
protected int height;
protected float viewportWidth;
protected float viewportHeight;
protected int pagecount;
protected boolean pagePending;
protected Shape initialClip;
protected AffineTransform initialTransform;
/**
* Create a new AbstractPSDocumentGraphics2D.
* This is used to create a new PostScript document, the height,
* width and output stream can be setup later.
* For use by the transcoder which needs font information
* for the bridge before the document size is known.
* The resulting document is written to the stream after rendering.
*
* @param textAsShapes set this to true so that text will be rendered
* using curves and not the font.
*/
AbstractPSDocumentGraphics2D(boolean textAsShapes) {
super(textAsShapes);
}
/**
* Setup the document.
* @param stream the output stream to write the document
* @param width the width of the page
* @param height the height of the page
* @throws IOException an io exception if there is a problem
* writing to the output stream
*/
public void setupDocument(OutputStream stream, int width, int height) throws IOException {
this.width = width;
this.height = height;
this.pagecount = 0;
this.pagePending = false;
//Setup for PostScript generation
setPSGenerator(new PSGenerator(stream));
writeFileHeader();
}
/**
* Writes the file header.
* @throws IOException if an I/O error occurs
*/
protected abstract void writeFileHeader() throws IOException;
/**
* Create a new AbstractPSDocumentGraphics2D.
* This is used to create a new PostScript document of the given height
* and width.
* The resulting document is written to the stream after rendering.
*
* @param textAsShapes set this to true so that text will be rendered
* using curves and not the font.
* @param stream the stream that the final document should be written to.
* @param width the width of the document
* @param height the height of the document
* @throws IOException an io exception if there is a problem
* writing to the output stream
*/
public AbstractPSDocumentGraphics2D(boolean textAsShapes, OutputStream stream,
int width, int height) throws IOException {
this(textAsShapes);
setupDocument(stream, width, height);
}
/**
* Set the dimensions of the document that will be drawn.
* This is useful if the dimensions of the document are different
* from the PostScript document that is to be created.
* The result is scaled so that the document fits correctly inside the
* PostScript document.
* @param w the width of the page
* @param h the height of the page
* @throws IOException in case of an I/O problem
*/
public void setViewportDimension(float w, float h) throws IOException {
this.viewportWidth = w;
this.viewportHeight = h;
/*
if (w != this.width || h != this.height) {
gen.concatMatrix(width / w, 0, 0, height / h, 0, 0);
}*/
}
/**
* Set the background of the PostScript document.
* This is used to set the background for the PostScript document
* Rather than leaving it as the default white.
* @param col the background colour to fill
*/
public void setBackgroundColor(Color col) {
/**(todo) Implement this */
/*
Color c = col;
PDFColor currentColour = new PDFColor(c.getRed(), c.getGreen(), c.getBlue());
currentStream.write("q\n");
currentStream.write(currentColour.getColorSpaceOut(true));
currentStream.write("0 0 " + width + " " + height + " re\n");
currentStream.write("f\n");
currentStream.write("Q\n");
*/
}
/**
* Returns the number of pages generated so far.
* @return the number of pages
*/
public int getPageCount() {
return this.pagecount;
}
/**
* Closes the current page and prepares to start a new one.
* @throws IOException if an I/O error occurs
*/
public void nextPage() throws IOException {
closePage();
}
/**
* Closes the current page.
* @throws IOException if an I/O error occurs
*/
protected void closePage() throws IOException {
if (!this.pagePending) {
return; //ignore
}
//Finish page
writePageTrailer();
this.pagePending = false;
}
/**
* Writes the page header for a page.
* @throws IOException In case an I/O error occurs
*/
protected abstract void writePageHeader() throws IOException;
/**
* Writes the page trailer for a page.
* @throws IOException In case an I/O error occurs
*/
protected abstract void writePageTrailer() throws IOException;
/**
* Writes the ProcSets ending up in the prolog to the PostScript file. Override to add your
* own ProcSets if so desired.
* @throws IOException In case an I/O error occurs
*/
protected void writeProcSets() throws IOException {
PSProcSets.writeStdProcSet(gen);
PSProcSets.writeEPSProcSet(gen);
}
/** {@inheritDoc} */
public void preparePainting() {
if (this.pagePending) {
return;
}
try {
startPage();
} catch (IOException ioe) {
handleIOException(ioe);
}
}
/**
* Starts a new page.
* @throws IOException if an I/O error occurs
*/
protected void startPage() throws IOException {
if (this.pagePending) {
throw new IllegalStateException("Close page first before starting another");
}
//Start page
this.pagecount++;
if (this.initialTransform == null) {
//Save initial transformation matrix
this.initialTransform = getTransform();
this.initialClip = getClip();
} else {
//Reset transformation matrix
setTransform(this.initialTransform);
setClip(this.initialClip);
}
writePageHeader();
AffineTransform at;
if ((this.viewportWidth != this.width
|| this.viewportHeight != this.height)
&& (this.viewportWidth > 0) && (this.viewportHeight > 0)) {
at = new AffineTransform(this.width / this.viewportWidth, 0,
0, -1 * (this.height / this.viewportHeight),
0, this.height);
} else {
at = new AffineTransform(1, 0, 0, -1, 0, this.height);
}
// Do not use concatMatrix, since it alters PSGenerator current state
//gen.concatMatrix(at);
gen.writeln(gen.formatMatrix(at) + " " + gen.mapCommand("concat"));
gen.writeDSCComment(DSCConstants.END_PAGE_SETUP);
this.pagePending = true;
}
/**
* The rendering process has finished.
* This should be called after the rendering has completed as there is
* no other indication it is complete.
* This will then write the results to the output stream.
* @throws IOException an io exception if there is a problem
* writing to the output stream
*/
public void finish() throws IOException {
if (this.pagePending) {
closePage();
}
//Finish document
gen.writeDSCComment(DSCConstants.TRAILER);
gen.writeDSCComment(DSCConstants.PAGES, this.pagecount);
gen.writeDSCComment(DSCConstants.EOF);
gen.flush();
}
/**
* This constructor supports the create method
* @param g the PostScript document graphics to make a copy of
*/
public AbstractPSDocumentGraphics2D(AbstractPSDocumentGraphics2D g) {
super(g);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy