123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366 |
- /* GeneralPath.java -- represents a shape built from subpaths
- Copyright (C) 2002 Free Software Foundation
- 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., 59 Temple Place, Suite 330, Boston, MA
- 02111-1307 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 java.awt.geom;
- import java.awt.Rectangle;
- import java.awt.Shape;
- /**
- * STUBS ONLY
- * XXX Implement and document. Note that Sun's implementation only expects
- * float precision, not double.
- */
- public final class GeneralPath implements Shape, Cloneable
- {
- public static final int WIND_EVEN_ODD = PathIterator.WIND_EVEN_ODD;
- public static final int WIND_NON_ZERO = PathIterator.WIND_NON_ZERO;
- /** Initial size if not specified. */
- private static final int INIT_SIZE = 20;
- /** The winding rule. */
- private int rule;
- /**
- * The path type in points. Note that points[index] maps to
- * types[index >> 1]; the control points of quad and cubic paths map as
- * well but are ignored.
- */
- private byte[] types;
- /**
- * The list of all points seen. Since you can only append floats, it makes
- * sense for this to be a float[]. I have no idea why Sun didn't choose to
- * allow a general path of double precision points.
- */
- private float[] points;
- /** The index of the most recent moveto point, or null. */
- private int subpath = -1;
- /** The next available index into points. */
- private int index;
- public GeneralPath()
- {
- this(WIND_NON_ZERO, INIT_SIZE);
- }
- public GeneralPath(int rule)
- {
- this(rule, INIT_SIZE);
- }
- public GeneralPath(int rule, int capacity)
- {
- if (rule != WIND_EVEN_ODD && rule != WIND_NON_ZERO)
- throw new IllegalArgumentException();
- this.rule = rule;
- if (capacity < INIT_SIZE)
- capacity = INIT_SIZE;
- types = new byte[capacity >> 1];
- points = new float[capacity];
- }
- public GeneralPath(Shape s)
- {
- types = new byte[INIT_SIZE >> 1];
- points = new float[INIT_SIZE];
- PathIterator pi = s.getPathIterator(null);
- setWindingRule(pi.getWindingRule());
- append(pi, false);
- }
- public void moveTo(float x, float y)
- {
- subpath = index;
- ensureSize(index + 2);
- types[index >> 1] = PathIterator.SEG_MOVETO;
- points[index++] = x;
- points[index++] = y;
- }
- public void lineTo(float x, float y)
- {
- ensureSize(index + 2);
- types[index >> 1] = PathIterator.SEG_LINETO;
- points[index++] = x;
- points[index++] = y;
- }
- public void quadTo(float x1, float y1, float x2, float y2)
- {
- ensureSize(index + 4);
- types[index >> 1] = PathIterator.SEG_QUADTO;
- points[index++] = x1;
- points[index++] = y1;
- points[index++] = x2;
- points[index++] = y2;
- }
- public void curveTo(float x1, float y1, float x2, float y2,
- float x3, float y3)
- {
- ensureSize(index + 6);
- types[index >> 1] = PathIterator.SEG_QUADTO;
- points[index++] = x1;
- points[index++] = y1;
- points[index++] = x2;
- points[index++] = y2;
- points[index++] = x3;
- points[index++] = y3;
- }
- public void closePath()
- {
- ensureSize(index + 2);
- types[index >> 1] = PathIterator.SEG_CLOSE;
- points[index++] = points[subpath];
- points[index++] = points[subpath + 1];
- }
- public void append(Shape s, boolean connect)
- {
- append(s.getPathIterator(null), connect);
- }
- public void append(PathIterator i, boolean connect)
- {
- float[] f = new float[6];
- while (! i.isDone())
- {
- int result = i.currentSegment(f);
- switch (result)
- {
- case PathIterator.SEG_MOVETO:
- if (! connect)
- {
- moveTo(f[0], f[1]);
- break;
- }
- if (subpath >= 0 && f[0] == points[subpath]
- && f[1] == points[subpath + 1])
- break;
- // Fallthrough.
- case PathIterator.SEG_LINETO:
- lineTo(f[0], f[1]);
- break;
- case PathIterator.SEG_QUADTO:
- quadTo(f[0], f[1], f[2], f[3]);
- break;
- case PathIterator.SEG_CUBICTO:
- curveTo(f[0], f[1], f[2], f[3], f[4], f[5]);
- break;
- default:
- closePath();
- }
- connect = false;
- }
- }
- public int getWindingRule()
- {
- return rule;
- }
- public void setWindingRule(int rule)
- {
- if (rule != WIND_EVEN_ODD && rule != WIND_NON_ZERO)
- throw new IllegalArgumentException();
- this.rule = rule;
- }
- public Point2D getCurrentPoint()
- {
- if (subpath < 0)
- return null;
- return new Point2D.Float(points[subpath], points[subpath + 1]);
- }
- public void reset()
- {
- subpath = -1;
- index = 0;
- }
- public void transform(AffineTransform xform)
- {
- xform.transform(points, 0, points, 0, index >> 1);
- }
- public Shape createTransformedShape(AffineTransform xform)
- {
- GeneralPath p = new GeneralPath(this);
- p.transform(xform);
- return p;
- }
- public Rectangle getBounds()
- {
- return getBounds2D().getBounds();
- }
- public Rectangle2D getBounds2D()
- {
- // XXX Implement.
- throw new Error("not implemented");
- }
- public boolean contains(double x, double y)
- {
- // XXX Implement.
- throw new Error("not implemented");
- }
- public boolean contains(Point2D p)
- {
- return contains(p.getX(), p.getY());
- }
- public boolean contains(double x, double y, double w, double h)
- {
- // XXX Implement.
- throw new Error("not implemented");
- }
- public boolean contains(Rectangle2D r)
- {
- return contains(r.getX(), r.getY(), r.getWidth(), r.getHeight());
- }
- public boolean intersects(double x, double y, double w, double h)
- {
- // XXX Implement.
- throw new Error("not implemented");
- }
- public boolean intersects(Rectangle2D r)
- {
- return intersects(r.getX(), r.getY(), r.getWidth(), r.getHeight());
- }
- public PathIterator getPathIterator(final AffineTransform at)
- {
- return new PathIterator()
- {
- int current = 0;
- public int getWindingRule()
- {
- return rule;
- }
- public boolean isDone()
- {
- return current >= index;
- }
- public void next()
- {
- current++;
- }
- public int currentSegment(float[] coords)
- {
- if (current >= index)
- return SEG_CLOSE;
- int result = types[current >> 1];
- int i = 0;
- if (result == 3)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- }
- if (result == 2)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- }
- if (result < 2)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- if (at != null)
- at.transform(coords, 0, coords, 0, result == 0 ? 1 : result);
- }
- return result;
- }
- public int currentSegment(double[] coords)
- {
- if (current >= index)
- return SEG_CLOSE;
- int result = types[current >> 1];
- int i = 0;
- if (result == 3)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- }
- if (result == 2)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- }
- if (result < 2)
- {
- coords[i++] = points[current++];
- coords[i++] = points[current++];
- if (at != null)
- at.transform(coords, 0, coords, 0, result == 0 ? 1 : result);
- }
- return result;
- }
- };
- }
- public PathIterator getPathIterator(AffineTransform at, double flatness)
- {
- return new FlatteningPathIterator(getPathIterator(at), flatness);
- }
- /**
- * Create a new shape of the same run-time type with the same contents as
- * this one.
- *
- * @return the clone
- *
- * @exception OutOfMemoryError If there is not enough memory available.
- *
- * @since 1.2
- */
- public Object clone()
- {
- // This class is final; no need to use super.clone().
- return new GeneralPath(this);
- }
- private void ensureSize(int size)
- {
- if (subpath < 0)
- throw new IllegalPathStateException("need initial moveto");
- if (size <= points.length)
- return;
- byte[] b = new byte[points.length];
- System.arraycopy(types, 0, b, 0, index >> 1);
- types = b;
- float[] f = new float[points.length << 1];
- System.arraycopy(points, 0, f, 0, index);
- points = f;
- }
- } // class GeneralPath
|