All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.twelvemonkeys.imageio.plugins.pict.QuickDrawContext Maven / Gradle / Ivy

/*
 * Copyright (c) 2008, Harald Kuhr
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * * Redistributions of source code must retain the above copyright notice, this
 *   list of conditions and the following disclaimer.
 *
 * * Redistributions in binary form must reproduce the above copyright notice,
 *   this list of conditions and the following disclaimer in the documentation
 *   and/or other materials provided with the distribution.
 *
 * * Neither the name of the copyright holder nor the names of its
 *   contributors may be used to endorse or promote products derived from
 *   this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package com.twelvemonkeys.imageio.plugins.pict;

import com.twelvemonkeys.lang.Validate;

import java.awt.*;
import java.awt.geom.*;
import java.awt.image.BufferedImage;

import static java.lang.Math.sqrt;

/**
 * Emulates an Apple QuickDraw rendering context, backed by a Java {@link Graphics2D}.
 * 

* * @author Harald Kuhr * @version $Id: QuickDrawContext.java,v 1.0 Oct 3, 2007 1:24:35 AM haraldk Exp$ */ // TODO: It would actually be possible to implement a version of this interface // that wrote opcodes/data to a stream... Or maybe a QDGraphics would be better. // TODO: Optimize for pensize 1,1? // TODO: Do we really need the Xxx2D stuff? // TODO: Support COPY_DITHER class QuickDrawContext { /* // The useful parts of the QD Graphics Port: portRect: Rect; {port rectangle} visRgn: RgnHandle; {visible region} clipRgn: RgnHandle; {clipping region} bkPat: Pattern; {background pattern} fillPat: Pattern; {fill pattern} pnLoc: Point; {pen location} pnSize: Point; {pen size} pnMode: Integer; {pattern mode} pnPat: Pattern; {pen pattern} pnVis: Integer; {pen visibility} txFont: Integer; {font number for text} txFace: Style; {text's font style} txMode: Integer; {source mode for text} txSize: Integer; {font size for text} spExtra: Fixed; {extra space} fgColor: LongInt; {foreground color} bkColor: LongInt; {background color} colrBit: Integer; {color bit} .. picSave: Handle; {picture being saved, used internally} rgnSave: Handle; {region being saved, used internally} polySave: Handle; {polygon being saved, used internally} */ /* // Color Graphics Port; chExtra: Integer; {added width for nonspace characters} pnLocHFrac: Integer; {pen fraction} portRect: Rect; {port rectangle} visRgn: RgnHandle; {visible region} clipRgn: RgnHandle; {clipping region} bkPixPat: PixPatHandle; {background pattern} rgbFgColor: RGBColor; {requested foreground color} rgbBkColor: RGBColor; {requested background color} pnLoc: Point; {pen location} pnSize: Point; {pen size} pnMode: Integer; {pattern mode} pnPixPat: PixPatHandle; {pen pattern} fillPixPat: PixPatHandle; {fill pattern} pnVis: Integer; {pen visibility} txFont: Integer; {font number for text} txFace: Style; {text's font style} txMode: Integer; {source mode for text} txSize: Integer; {font size for text} spExtra: Fixed; {added width for space characters} fgColor: LongInt; {actual foreground color} bkColor: LongInt; {actual background color} colrBit: Integer; {plane being drawn} .. picSave: Handle; {picture being saved, used internally} rgnSave: Handle; {region being saved, used internally} polySave: Handle; {polygon being saved, used internally} */ private final Graphics2D graphics; private Pattern background; // http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-68.html#HEADING68-0 // Upon the creation of a graphics port, QuickDraw assigns these initial // values to the graphics pen: a size of (1,1), a pattern of all-black pixels, // and the patCopy pattern mode. After changing any of these values, // you can use the PenNormal procedure to return these initial values to the // graphics pen. // TODO: Consider creating a Pen/PenState class? private int penVisibility = 0; private Point2D penPosition = new Point(); private Pattern penPattern; private Dimension2D penSize = new Dimension(); private int penMode; // TODO: Make sure setting bgColor/fgColor does not reset pattern, and pattern not resetting bg/fg! private Color bgColor = Color.WHITE; private Color fgColor = Color.BLACK; private int textMode; private Pattern textPattern = new BitMapPattern(Color.BLACK); private Pattern fillPattern; QuickDrawContext(final Graphics2D pGraphics) { graphics = Validate.notNull(pGraphics, "graphics"); graphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON); graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); setPenNormal(); } protected void dispose() { graphics.dispose(); } // ClosePicture public void closePicture() { dispose(); } // ClipRgn public void setClipRegion(Shape pClip) { graphics.setClip(pClip); } // Font number (sic), integer void setTextFont(int fontFamily) { // ..? System.err.println("QuickDrawContext.setTextFont: " + fontFamily); } public void setTextFont(final String fontName) { // TODO: Need mapping between known QD font names and Java font names? Font current = graphics.getFont(); graphics.setFont(Font.decode(fontName).deriveFont(current.getStyle(), (float) current.getSize())); } // Sets the text's font style (0..255) void setTextFace(final int face) { int style = 0; if ((face & QuickDraw.TX_BOLD_MASK) > 0) { style |= Font.BOLD; } if ((face & QuickDraw.TX_ITALIC_MASK) > 0) { style |= Font.ITALIC; } // TODO: Other face options, like underline, shadow, etc... graphics.setFont(graphics.getFont().deriveFont(style)); } void setTextMode(int pSourceMode) { // ..? System.err.println("QuickDrawContext.setTextMode"); textMode = pSourceMode; } public void setTextSize(int pSize) { graphics.setFont(graphics.getFont().deriveFont((float) pSize)); } // Numerator (Point), denominator (Point) void setTextRatio() { // TODO System.err.println("QuickDrawContext.setTextRatio"); } // TODO: spExtra added width for space characters // TODO: chExtra added width for nonspace characters public void setOrigin(Point2D pOrigin) { graphics.translate(pOrigin.getX(), pOrigin.getY()); } public void setForeground(final Color pColor) { fgColor = pColor; penPattern = new BitMapPattern(pColor); } Color getForeground() { return fgColor; } public void setBackground(final Color pColor) { bgColor = pColor; background = new BitMapPattern(pColor); } Color getBackground() { return bgColor; } /* // Pen management: // NOTE: The HidePen procedure is called by the OpenRgn, OpenPicture, and OpenPoly routines so that you can create regions, pictures, and polygons without drawing on the screen. // ShowPen is called by the procedures CloseRgn, ClosePoly, and ClosePicture GetPenState // All pen state incl. position (PenState type?) SetPenState */ /** * HidePen Visibility (decrements visibility by one!) */ public void hidePen() { penVisibility--; } /** * ShowPen Visibility (increments visibility by one!) */ public void showPen() { penVisibility++; } /** * Tells whether pen is visible. * * @return {@code true} if pen is visible */ private boolean isPenVisible() { return penVisibility >= 0; } /** * Returns the pen position. * GetPen * * @return the current pen position */ public Point2D getPenPosition() { return (Point2D) penPosition.clone(); } /** * Sets the pen size. * PenSize * * @param pSize the new size */ public void setPenSize(Dimension2D pSize) { penSize.setSize(pSize); graphics.setStroke(getStroke(penSize)); } /** * PenMode // Sets pen pattern mode * * @param pPenMode the new pen mode */ public void setPenMode(int pPenMode) { // TODO: Handle HILITE (+50) // TODO: Handle DITHER_COPY (+64) switch (pPenMode) { // Boolean source transfer modes case QuickDraw.SRC_COPY: case QuickDraw.SRC_OR: case QuickDraw.SRC_XOR: case QuickDraw.SRC_BIC: case QuickDraw.NOT_SRC_COPY: case QuickDraw.NOT_SRC_OR: case QuickDraw.NOT_SRC_XOR: case QuickDraw.NOT_SRC_BIC: // Boolean pattern transfer modes case QuickDraw.PAT_COPY: case QuickDraw.PAT_OR: case QuickDraw.PAT_XOR: case QuickDraw.PAT_BIC: case QuickDraw.NOT_PAT_COPY: case QuickDraw.NOT_PAT_OR: case QuickDraw.NOT_PAT_XOR: case QuickDraw.NOT_PAT_BIC: // Aritmetic transfer modes case QuickDraw.BLEND: case QuickDraw.ADD_PIN: case QuickDraw.ADD_OVER: case QuickDraw.SUB_PIN: case QuickDraw.TRANSPARENT: case QuickDraw.ADD_MAX: case QuickDraw.SUB_OVER: case QuickDraw.ADD_MIN: case QuickDraw.GRAYISH_TEXT_OR: penMode = pPenMode; break; default: throw new IllegalArgumentException("Undefined pen mode: " + pPenMode); } } /** * PenPat & PenPixPat // Sets pen bit pattern or pix pattern * * @param pPattern the new pattern */ public void setPenPattern(final Pattern pPattern) { penPattern = pPattern; } /** * PenNormal // Reset (except posiotion) */ public final void setPenNormal() { // NOTE: Shold not change pen location // TODO: What about visibility? Probably not touch setPenPattern(QuickDraw.BLACK); setPenSize(new Dimension(1, 1)); penMode = QuickDraw.SRC_COPY; } /* // Background pattern: BackPat // Used by the Erase* methods *BackPixPat */ public void setBackgroundPattern(final Pattern pPaint) { background = pPaint; } public void setFillPattern(final Pattern fillPattern) { this.fillPattern = fillPattern; } private Composite getCompositeFor(final int pMode) { switch (pMode & ~QuickDraw.DITHER_COPY) { // Boolean source transfer modes case QuickDraw.SRC_COPY: return AlphaComposite.Src; // Or, SRC_OVER? case QuickDraw.SRC_OR: return AlphaComposite.SrcOver; // Or, DST_OUT? case QuickDraw.SRC_XOR: return AlphaComposite.Xor; case QuickDraw.SRC_BIC: return AlphaComposite.Clear; case QuickDraw.NOT_SRC_XOR: return QuickDrawComposite.NotSrcXor; case QuickDraw.NOT_SRC_COPY: case QuickDraw.NOT_SRC_OR: case QuickDraw.NOT_SRC_BIC: throw new UnsupportedOperationException("Not implemented for mode " + pMode); // Boolean pattern transfer modes case QuickDraw.PAT_COPY: return AlphaComposite.Src; // Tested case QuickDraw.PAT_OR: return AlphaComposite.SrcOver; // Or, DST_OUT? case QuickDraw.PAT_XOR: return AlphaComposite.Xor; case QuickDraw.PAT_BIC: return AlphaComposite.Clear; case QuickDraw.NOT_PAT_COPY: case QuickDraw.NOT_PAT_OR: case QuickDraw.NOT_PAT_XOR: case QuickDraw.NOT_PAT_BIC: throw new UnsupportedOperationException("Not implemented for mode " + pMode); // Aritmetic transfer modes case QuickDraw.BLEND: return AlphaComposite.SrcOver.derive(.5f); case QuickDraw.ADD_PIN: case QuickDraw.ADD_OVER: case QuickDraw.SUB_PIN: case QuickDraw.TRANSPARENT: throw new UnsupportedOperationException("Not implemented for mode " + pMode); case QuickDraw.ADD_MAX: return QuickDrawComposite.AddMax; case QuickDraw.SUB_OVER: throw new UnsupportedOperationException("Not implemented for mode " + pMode); case QuickDraw.ADD_MIN: return QuickDrawComposite.AddMin; case QuickDraw.GRAYISH_TEXT_OR: throw new UnsupportedOperationException("Not implemented for mode " + pMode); default: throw new IllegalArgumentException("Unknown pnMode: " + pMode); } } /** * Sets up context for text drawing. */ protected void setupForText() { graphics.setPaint(textPattern); graphics.setComposite(getCompositeFor(textMode)); } /** * Sets up context for line drawing/painting. */ protected void setupForPaint() { graphics.setPaint(penPattern); graphics.setComposite(getCompositeFor(penMode)); //graphics.setStroke(getStroke(penSize)); } private Stroke getStroke(final Dimension2D pPenSize) { // TODO: OPTIMIZE: Only create stroke if changed! if (pPenSize.getWidth() <= 1.0 && pPenSize.getWidth() == pPenSize.getHeight()) { return new BasicStroke((float) pPenSize.getWidth()); } return new RectangleStroke(new Rectangle2D.Double(0, 0, pPenSize.getWidth(), pPenSize.getHeight())); } /** * Sets up paint context for fill. * * @param pPattern the pattern to use for filling. */ protected void setupForFill(final Pattern pPattern) { graphics.setPaint(pPattern); graphics.setComposite(getCompositeFor(QuickDraw.PAT_COPY)); } protected void setupForErase() { graphics.setPaint(background); graphics.setComposite(getCompositeFor(QuickDraw.PAT_COPY)); // TODO: Check spec } protected void setupForInvert() { // TODO: Setup for invert graphics.setColor(Color.BLACK); graphics.setXORMode(Color.WHITE); } /* // Line drawing: MoveTo // Moves to new pos Move // distance (MoveTo(h+dh,v+dv)) LineTo // Draws from current pos to new pos, stores new pos Line // dinstance (LineTo(h+dh,v+dv)) */ public void moveTo(final double pX, final double pY) { penPosition.setLocation(pX, pY); } public final void moveTo(final Point2D pPosition) { moveTo(pPosition.getX(), pPosition.getY()); } public final void move(final double pDeltaX, final double pDeltaY) { moveTo(penPosition.getX() + pDeltaX, penPosition.getY() + pDeltaY); } public void lineTo(final double pX, final double pY) { Shape line = new Line2D.Double(penPosition.getX(), penPosition.getY(), pX, pY); // TODO: Add line to current shape if recording... if (isPenVisible()) { // NOTE: Workaround for known Mac JDK bug: Paint, not frame paintShape(graphics.getStroke().createStrokedShape(line)); } moveTo(pX, pY); } public final void lineTo(final Point2D pPosition) { lineTo(pPosition.getX(), pPosition.getY()); } public final void line(final double pDeltaX, final double pDeltaY) { lineTo(penPosition.getX() + pDeltaX, penPosition.getY() + pDeltaY); } /* // Drawing With Color QuickDraw Colors: * RGBForeColor * RGBBackColor * SetCPixel <-- TODO * FillCRect * FillCRoundRect * FillCOval * FillCArc * FillCPoly * FillCRgn * OpColor // sets the maximum color values for the addPin and subPin arithmetic transfer modes, and the weight color for the blend arithmetic transfer mode. * HiliteColor // (SKIP?) See http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-199.html#MARKER-9-151 // Creating and Managing Rectangles (SKIP? Rely on awt Rectangle): SetRect OffsetRect InsetRect SectRect UnionRect PtInRect Pt2Rect PtToAngle EqualRect EmptyRect // Reset */ // Drawing Rectangles: /** * FrameRect(r) // outline rect with the size, pattern, and pattern mode of * the graphics pen. * * @param pRectangle the rectangle to frame */ public void frameRect(final Rectangle2D pRectangle) { frameShape(pRectangle); } /** * PaintRect(r) // fills a rectangle's interior with the pattern of the * graphics pen, using the pattern mode of the graphics pen. * * @param pRectangle the rectangle to paint */ public void paintRect(final Rectangle2D pRectangle) { paintShape(pRectangle); } /** * FillRect(r, pat) // fills a rectangle's interior with any pattern you * specify. The procedure transfers the pattern with the patCopy pattern * mode, which directly copies your requested pattern into the shape. * * @param pRectangle the rectangle to fill * @param pPattern the pattern to use */ public void fillRect(final Rectangle2D pRectangle, Pattern pPattern) { fillShape(pRectangle, pPattern); } /** * EraseRect(r) // fills the rectangle's interior with the background pattern * * @param pRectangle the rectangle to erase */ public void eraseRect(final Rectangle2D pRectangle) { eraseShape(pRectangle); } /** * InvertRect(r) // reverses the color of all pixels in the rect * * @param pRectangle the rectangle to invert */ public void invertRect(final Rectangle2D pRectangle) { invertShape(pRectangle); } // http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-102.html#HEADING102-0 // Drawing Rounded Rectangles: private static RoundRectangle2D.Double toRoundRect(final Rectangle2D pRectangle, final int pArcW, final int pArcH) { return new RoundRectangle2D.Double( pRectangle.getX(), pRectangle.getY(), pRectangle.getWidth(), pRectangle.getHeight(), pArcW, pArcH); } /** * FrameRoundRect(r,int,int) // outline round rect with the size, pattern, and pattern mode of * the graphics pen. * * @param pRectangle the rectangle to frame * @param pArcW width of the oval defining the rounded corner. * @param pArcH height of the oval defining the rounded corner. */ public void frameRoundRect(final Rectangle2D pRectangle, int pArcW, int pArcH) { frameShape(toRoundRect(pRectangle, pArcW, pArcH)); } /** * PaintRooundRect(r,int,int) // fills a rectangle's interior with the pattern of the * graphics pen, using the pattern mode of the graphics pen. * * @param pRectangle the rectangle to paint * @param pArcW width of the oval defining the rounded corner. * @param pArcH height of the oval defining the rounded corner. */ public void paintRoundRect(final Rectangle2D pRectangle, int pArcW, int pArcH) { paintShape(toRoundRect(pRectangle, pArcW, pArcH)); } /** * FillRoundRect(r,int,int,pat) // fills a rectangle's interior with any pattern you * specify. The procedure transfers the pattern with the patCopy pattern * mode, which directly copies your requested pattern into the shape. * * @param pRectangle the rectangle to fill * @param pArcW width of the oval defining the rounded corner. * @param pArcH height of the oval defining the rounded corner. * @param pPattern the pattern to use */ public void fillRoundRect(final Rectangle2D pRectangle, int pArcW, int pArcH, Pattern pPattern) { fillShape(toRoundRect(pRectangle, pArcW, pArcH), pPattern); } /** * EraseRoundRect(r,int,int) // fills the rectangle's interior with the background pattern * * @param pRectangle the rectangle to erase * @param pArcW width of the oval defining the rounded corner. * @param pArcH height of the oval defining the rounded corner. */ public void eraseRoundRect(final Rectangle2D pRectangle, int pArcW, int pArcH) { eraseShape(toRoundRect(pRectangle, pArcW, pArcH)); } /** * InvertRoundRect(r,int,int) // reverses the color of all pixels in the rect * * @param pRectangle the rectangle to invert * @param pArcW width of the oval defining the rounded corner. * @param pArcH height of the oval defining the rounded corner. */ public void invertRoundRect(final Rectangle2D pRectangle, int pArcW, int pArcH) { invertShape(toRoundRect(pRectangle, pArcW, pArcH)); } // Drawing Ovals: private static Ellipse2D.Double toOval(final Rectangle2D pRectangle) { Ellipse2D.Double ellipse = new Ellipse2D.Double(); ellipse.setFrame(pRectangle); return ellipse; } /** * FrameOval(r) // outline oval with the size, pattern, and pattern mode of * the graphics pen. * * @param pRectangle the rectangle to frame */ public void frameOval(final Rectangle2D pRectangle) { frameShape(toOval(pRectangle)); } /** * PaintOval(r) // fills an oval's interior with the pattern of the * graphics pen, using the pattern mode of the graphics pen. * * @param pRectangle the rectangle to paint */ public void paintOval(final Rectangle2D pRectangle) { paintShape(toOval(pRectangle)); } /** * FillOval(r, pat) // fills an oval's interior with any pattern you * specify. The procedure transfers the pattern with the patCopy pattern * mode, which directly copies your requested pattern into the shape. * * @param pRectangle the rectangle to fill * @param pPattern the pattern to use */ public void fillOval(final Rectangle2D pRectangle, Pattern pPattern) { fillShape(toOval(pRectangle), pPattern); } /** * EraseOval(r) // fills the oval's interior with the background pattern * * @param pRectangle the rectangle to erase */ public void eraseOval(final Rectangle2D pRectangle) { eraseShape(toOval(pRectangle)); } /** * InvertOval(r) // reverses the color of all pixels in the oval * * @param pRectangle the rectangle to invert */ public void invertOval(final Rectangle2D pRectangle) { invertShape(toOval(pRectangle)); } // http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-114.html#HEADING114-0 // NOTE: Differs from Java 2D arcs, in start angle, and rotation // Drawing Arcs and Wedges: /** * Converts a rectangle to an arc. * * @param pRectangle the framing rectangle * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) * @param pClosed specifies if the arc should be closed * @return the arc */ private static Arc2D.Double toArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle, final boolean pClosed) { return new Arc2D.Double(pRectangle, 90 - pStartAngle, -pArcAngle, pClosed ? Arc2D.PIE : Arc2D.OPEN); } /** * FrameArc(r,int,int) // outline arc with the size, pattern, and pattern mode of * the graphics pen. * * @param pRectangle the rectangle to frame * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) */ public void frameArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle) { frameShape(toArc(pRectangle, pStartAngle, pArcAngle, false)); } /** * PaintArc(r,int,int) // fills an arc's interior with the pattern of the * graphics pen, using the pattern mode of the graphics pen. * * @param pRectangle the rectangle to paint * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) */ public void paintArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle) { paintShape(toArc(pRectangle, pStartAngle, pArcAngle, true)); } /** * FillArc(r,int,int, pat) // fills an arc's interior with any pattern you * specify. The procedure transfers the pattern with the patCopy pattern * mode, which directly copies your requested pattern into the shape. * * @param pRectangle the rectangle to fill * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) * @param pPattern the pattern to use */ public void fillArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle, Pattern pPattern) { fillShape(toArc(pRectangle, pStartAngle, pArcAngle, true), pPattern); } /** * EraseArc(r,int,int) // fills the arc's interior with the background pattern * * @param pRectangle the rectangle to erase * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) */ public void eraseArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle) { eraseShape(toArc(pRectangle, pStartAngle, pArcAngle, true)); } /** * InvertArc(r,int,int) // reverses the color of all pixels in the arc * * @param pRectangle the rectangle to invert * @param pStartAngle start angle in degrees (starting from 12'o clock, this differs from Java) * @param pArcAngle rotation angle in degrees (starting from {@code pStartAngle}, this differs from Java arcs) */ public void invertArc(final Rectangle2D pRectangle, int pStartAngle, int pArcAngle) { invertShape(toArc(pRectangle, pStartAngle, pArcAngle, true)); } /* // http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-120.html#HEADING120-0 // Creating and Managing Polygons: // - Use Shape? OpenPoly // Returns a reference to the polygon, used by the paint or other methods ClosePoly // Close (use LineTo with invisible pen to create) OffsetPoly KillPoly // Set internal reference to null */ // Drawing Polygons: // TODO: What is the Xxx2D equivalent of Polygon!? GeneralPath? // FramePoly public void framePoly(final Polygon pPolygon) { // TODO: The old PICTImageReader does not draw the last connection line, // unless the start and end point is the same... // Find out what the spec says. //if (pPolygon.xpoints[0] == pPolygon.xpoints[pPolygon.npoints - 1] && // pPolygon.ypoints[0] == pPolygon.ypoints[pPolygon.npoints - 1]) { //} frameShape(pPolygon); } // From the source: // "Four of these procedures--PaintPoly, ErasePoly, InvertPoly, and FillPoly-- // temporarily convert the polygon into a region to perform their operations" // PaintPoly public void paintPoly(final Polygon pPolygon) { paintShape(pPolygon); } // FillPoly public void fillPoly(final Polygon pPolygon, final Pattern pPattern) { fillShape(pPolygon, pPattern); } // ErasePoly public void erasePoly(final Polygon pPolygon) { eraseShape(pPolygon); } // InvertPoly public void invertPoly(final Polygon pPolygon) { invertShape(pPolygon); } /* // http://developer.apple.com/documentation/mac/quickdraw/QuickDraw-131.html#HEADING131-0 // Creating and Managing Regions: // TODO: Java equiv? Area? NewRgn // new Region? OpenRgn // Start collecting region information CloseRgn DisposeRgn CopyRgn SetEmptyRgn SetRectRgn RectRgn OffsetRgn InsetRgn SectRgn UnionRgn DiffRgn XorRgn PtInRgn RectInRgn EqualRgn EmptyRgn */ // Drawing Regions: // FrameRgn public void frameRegion(final Area pArea) { frameShape(pArea); } // PaintRgn public void paintRegion(final Area pArea) { paintShape(pArea); } // FillRgn public void fillRegion(final Area pArea, final Pattern pPattern) { fillShape(pArea, pPattern); } // EraseRgn public void eraseRegion(final Area pArea) { eraseShape(pArea); } // InvertRgn public void invertRegion(final Area pArea) { invertShape(pArea); } // TODO: All other operations can delegate to these! :-) private void frameShape(final Shape pShape) { if (isPenVisible()) { setupForPaint(); Stroke stroke = getStroke(penSize); Shape shape = stroke.createStrokedShape(pShape); graphics.draw(shape); } } private void paintShape(final Shape pShape) { setupForPaint(); graphics.fill(pShape); // Yes, fill } private void fillShape(final Shape pShape, final Pattern pPattern) { setupForFill(pPattern); graphics.fill(pShape); } private void invertShape(final Shape pShape) { setupForInvert(); graphics.fill(pShape); } private void eraseShape(final Shape pShape) { setupForErase(); graphics.fill(pShape); } /* // Scaling and Mapping Points, Rectangles, Polygons, and Regions: ScalePt // Use the getXPtCoord/Y from the reader? MapPt MapRect MapRgn MapPoly // Calculating Black-and-White Fills (SKIP?): SeedFill // MacPaint paint-bucket tool CalcMask // Calculates where paint would not flow (see above) // Copying Images (SKIP?): */ /** * CopyBits. *

* Note that the destination is always {@code this}. * * @param pSrcBitmap the source bitmap to copy pixels from * @param pSrcRect the source rectangle * @param pDstRect the destination rectangle * @param pMode the blending mode * @param pMaskRgn the mask region */ public void copyBits(BufferedImage pSrcBitmap, Rectangle pSrcRect, Rectangle pDstRect, int pMode, Shape pMaskRgn) { graphics.setComposite(getCompositeFor(pMode)); if (pMaskRgn != null) { setClipRegion(pMaskRgn); } graphics.drawImage( pSrcBitmap, pDstRect.x, pDstRect.y, pDstRect.x + pDstRect.width, pDstRect.y + pDstRect.height, pSrcRect.x, pSrcRect.y, pSrcRect.x + pSrcRect.width, pSrcRect.y + pSrcRect.height, null ); setClipRegion(null); } /** * CopyMask */ public void copyMask(BufferedImage pSrcBitmap, BufferedImage pMaskBitmap, Rectangle pSrcRect, Rectangle pMaskRect, Rectangle pDstRect, int pSrcCopy, Shape pMaskRgn) { throw new UnsupportedOperationException("Method copyMask not implemented"); // TODO: Implement } /** * CopyDeepMask -- available to basic QuickDraw only in System 7, combines the functionality of both CopyBits and CopyMask */ public void copyDeepMask(BufferedImage pSrcBitmap, BufferedImage pMaskBitmap, Rectangle pSrcRect, Rectangle pMaskRect, Rectangle pDstRect, int pSrcCopy, Shape pMaskRgn) { throw new UnsupportedOperationException("Method copyDeepMask not implemented"); // TODO: Implement } /* // Drawing With the Eight-Color System: ForeColor // color of the "ink" used to frame, fill, and paint. BackColor ColorBit // Getting Pattern Resources: GetPattern GetIndPattern // http://developer.apple.com/documentation/mac/Text/Text-128.html#HEADING128-9 // Graphics Ports and Text Drawing // Setting Text Characteristics: TextFont // specifies the font to be used. TextFace // specifies the glyph style. TextMode // specifies the transfer mode. TextSize // specifies the font size. SpaceExtra // specifies the amount of pixels by which to widen or narrow each space character in a range of text. CharExtra // specifies the amount of pixels by which to widen or narrow each glyph other than the space characters in a range of text (CharExtra). GetFontInfo // Drawing Text: DrawChar // draws the glyph of a single 1-byte character. */ /** * DrawString - draws the text of a Pascal string. * * @param pString a Pascal string (a string of length less than or equal to 255 chars). */ public void drawString(String pString) { setupForText(); graphics.drawString(pString, (float) getPenPosition().getX(), (float) getPenPosition().getY()); } /* DrawText // draws the glyphs of a sequence of characters. DrawJustified // draws a sequence of text that is widened or narrowed by a specified number of pixels. // Measuring Text: CharWidth // returns the horizontal extension of a single glyph. StringWidth // returns the width of a Pascal string. TextWidth // returns the width of the glyphs of a text segment. MeasureText // fills an array with an entry for each character identifying the width of each character's glyph as measured from the left side of the entire text segment. MeasureJustified // fills an array with an entry for each character in a style run identifying the width of each character's glyph as measured from the left side of the text segment. // Laying Out a Line of Text: GetFormatOrder // determines the display order of style runs for a line of text containing multiple style runs with mixed directions VisibleLength // eliminates trailing spaces from the last style run on the line. PortionLine // determines how to distribute the total slop value for a line among the style runs on that line. // Determining the Caret Position, and Selecting and Highlighting Text: PixelToChar // converts a pixel location associated with a glyph in a range of text to a byte offset within the style run. CharToPixel // converts a byte offset to a pixel location. The pixel location is measured from the left edge of the style run. HiliteText // returns three pairs of offsets marking the endpoints of ranges of text to be highlighted. // Color Constants �whiteColor =�30; �blackColor = 33 �yellowColor = 69; magentaColor =�137; �redColor =�205; �cyanColor =�273; �greenColor =�341; �blueColor =�409; */ // TODO: Simplify! Extract to upper level class static class RectangleStroke implements Stroke { private Shape mShapes[]; private boolean repeat = true; private AffineTransform mTransform = new AffineTransform(); private static final float FLATNESS = 1; public RectangleStroke(Shape pShape) { this(new Shape[]{pShape}); } RectangleStroke(Shape pShapes[]) { mShapes = new Shape[pShapes.length]; for (int i = 0; i < mShapes.length; i++) { Rectangle2D bounds = pShapes[i].getBounds2D(); mTransform.setToTranslation(-bounds.getCenterX(), -bounds.getCenterY()); mShapes[i] = mTransform.createTransformedShape(pShapes[i]); } } public Shape createStrokedShape(Shape shape) { GeneralPath result = new GeneralPath(); PathIterator it = new FlatteningPathIterator(shape.getPathIterator(null), FLATNESS); float points[] = new float[6]; float moveX = 0, moveY = 0; float lastX = 0, lastY = 0; float thisX = 0, thisY = 0; int type = 0; float next = 0; int currentShape = 0; int length = mShapes.length; while (currentShape < length && !it.isDone()) { type = it.currentSegment(points); switch (type) { case PathIterator.SEG_MOVETO: moveX = lastX = points[0]; moveY = lastY = points[1]; result.moveTo(moveX, moveY); next = 0; break; case PathIterator.SEG_CLOSE: points[0] = moveX; points[1] = moveY; // Fall through case PathIterator.SEG_LINETO: thisX = points[0]; thisY = points[1]; float dx = thisX - lastX; float dy = thisY - lastY; float distance = (float) sqrt(dx * dx + dy * dy); if (distance >= next) { float r = 1.0f / distance; //float angle = (float) Math.atan2(dy, dx); while (currentShape < length && distance >= next) { float x = lastX + next * dx * r; float y = lastY + next * dy * r; mTransform.setToTranslation(x, y); //mTransform.rotate(angle); result.append(mTransform.createTransformedShape(mShapes[currentShape]), false); next += 1; currentShape++; if (repeat) { currentShape %= length; } } } next -= distance; lastX = thisX; lastY = thisY; break; } it.next(); } return result; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy