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