• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.graphics;
18 
19 import android.annotation.NonNull;
20 import android.annotation.Nullable;
21 
22 /**
23  * The Path class encapsulates compound (multiple contour) geometric paths
24  * consisting of straight line segments, quadratic curves, and cubic curves.
25  * It can be drawn with canvas.drawPath(path, paint), either filled or stroked
26  * (based on the paint's Style), or it can be used for clipping or to draw
27  * text on a path.
28  */
29 public class Path {
30     /**
31      * @hide
32      */
33     public long mNativePath;
34 
35     /**
36      * @hide
37      */
38     public boolean isSimplePath = true;
39     /**
40      * @hide
41      */
42     public Region rects;
43     private Direction mLastDirection = null;
44 
45     /**
46      * Create an empty path
47      */
Path()48     public Path() {
49         mNativePath = init1();
50     }
51 
52     /**
53      * Create a new path, copying the contents from the src path.
54      *
55      * @param src The path to copy from when initializing the new path
56      */
Path(Path src)57     public Path(Path src) {
58         long valNative = 0;
59         if (src != null) {
60             valNative = src.mNativePath;
61             isSimplePath = src.isSimplePath;
62             if (src.rects != null) {
63                 rects = new Region(src.rects);
64             }
65         }
66         mNativePath = init2(valNative);
67     }
68 
69     /**
70      * Clear any lines and curves from the path, making it empty.
71      * This does NOT change the fill-type setting.
72      */
reset()73     public void reset() {
74         isSimplePath = true;
75         mLastDirection = null;
76         if (rects != null) rects.setEmpty();
77         // We promised not to change this, so preserve it around the native
78         // call, which does now reset fill type.
79         final FillType fillType = getFillType();
80         native_reset(mNativePath);
81         setFillType(fillType);
82     }
83 
84     /**
85      * Rewinds the path: clears any lines and curves from the path but
86      * keeps the internal data structure for faster reuse.
87      */
rewind()88     public void rewind() {
89         isSimplePath = true;
90         mLastDirection = null;
91         if (rects != null) rects.setEmpty();
92         native_rewind(mNativePath);
93     }
94 
95     /** Replace the contents of this with the contents of src.
96     */
set(@onNull Path src)97     public void set(@NonNull Path src) {
98         if (this == src) {
99             return;
100         }
101         isSimplePath = src.isSimplePath;
102         native_set(mNativePath, src.mNativePath);
103         if (!isSimplePath) {
104             return;
105         }
106 
107         if (rects != null && src.rects != null) {
108             rects.set(src.rects);
109         } else if (rects != null && src.rects == null) {
110             rects.setEmpty();
111         } else if (src.rects != null) {
112             rects = new Region(src.rects);
113         }
114     }
115 
116     /**
117      * The logical operations that can be performed when combining two paths.
118      *
119      * @see #op(Path, android.graphics.Path.Op)
120      * @see #op(Path, Path, android.graphics.Path.Op)
121      */
122     public enum Op {
123         /**
124          * Subtract the second path from the first path.
125          */
126         DIFFERENCE,
127         /**
128          * Intersect the two paths.
129          */
130         INTERSECT,
131         /**
132          * Union (inclusive-or) the two paths.
133          */
134         UNION,
135         /**
136          * Exclusive-or the two paths.
137          */
138         XOR,
139         /**
140          * Subtract the first path from the second path.
141          */
142         REVERSE_DIFFERENCE
143     }
144 
145     /**
146      * Set this path to the result of applying the Op to this path and the specified path.
147      * The resulting path will be constructed from non-overlapping contours.
148      * The curve order is reduced where possible so that cubics may be turned
149      * into quadratics, and quadratics maybe turned into lines.
150      *
151      * @param path The second operand (for difference, the subtrahend)
152      *
153      * @return True if operation succeeded, false otherwise and this path remains unmodified.
154      *
155      * @see Op
156      * @see #op(Path, Path, android.graphics.Path.Op)
157      */
op(Path path, Op op)158     public boolean op(Path path, Op op) {
159         return op(this, path, op);
160     }
161 
162     /**
163      * Set this path to the result of applying the Op to the two specified paths.
164      * The resulting path will be constructed from non-overlapping contours.
165      * The curve order is reduced where possible so that cubics may be turned
166      * into quadratics, and quadratics maybe turned into lines.
167      *
168      * @param path1 The first operand (for difference, the minuend)
169      * @param path2 The second operand (for difference, the subtrahend)
170      *
171      * @return True if operation succeeded, false otherwise and this path remains unmodified.
172      *
173      * @see Op
174      * @see #op(Path, android.graphics.Path.Op)
175      */
op(Path path1, Path path2, Op op)176     public boolean op(Path path1, Path path2, Op op) {
177         if (native_op(path1.mNativePath, path2.mNativePath, op.ordinal(), this.mNativePath)) {
178             isSimplePath = false;
179             rects = null;
180             return true;
181         }
182         return false;
183     }
184 
185     /**
186      * Returns the path's convexity, as defined by the content of the path.
187      * <p>
188      * A path is convex if it has a single contour, and only ever curves in a
189      * single direction.
190      * <p>
191      * This function will calculate the convexity of the path from its control
192      * points, and cache the result.
193      *
194      * @return True if the path is convex.
195      */
isConvex()196     public boolean isConvex() {
197         return native_isConvex(mNativePath);
198     }
199 
200     /**
201      * Enum for the ways a path may be filled.
202      */
203     public enum FillType {
204         // these must match the values in SkPath.h
205         /**
206          * Specifies that "inside" is computed by a non-zero sum of signed
207          * edge crossings.
208          */
209         WINDING         (0),
210         /**
211          * Specifies that "inside" is computed by an odd number of edge
212          * crossings.
213          */
214         EVEN_ODD        (1),
215         /**
216          * Same as {@link #WINDING}, but draws outside of the path, rather than inside.
217          */
218         INVERSE_WINDING (2),
219         /**
220          * Same as {@link #EVEN_ODD}, but draws outside of the path, rather than inside.
221          */
222         INVERSE_EVEN_ODD(3);
223 
FillType(int ni)224         FillType(int ni) {
225             nativeInt = ni;
226         }
227 
228         final int nativeInt;
229     }
230 
231     // these must be in the same order as their native values
232     static final FillType[] sFillTypeArray = {
233         FillType.WINDING,
234         FillType.EVEN_ODD,
235         FillType.INVERSE_WINDING,
236         FillType.INVERSE_EVEN_ODD
237     };
238 
239     /**
240      * Return the path's fill type. This defines how "inside" is
241      * computed. The default value is WINDING.
242      *
243      * @return the path's fill type
244      */
getFillType()245     public FillType getFillType() {
246         return sFillTypeArray[native_getFillType(mNativePath)];
247     }
248 
249     /**
250      * Set the path's fill type. This defines how "inside" is computed.
251      *
252      * @param ft The new fill type for this path
253      */
setFillType(FillType ft)254     public void setFillType(FillType ft) {
255         native_setFillType(mNativePath, ft.nativeInt);
256     }
257 
258     /**
259      * Returns true if the filltype is one of the INVERSE variants
260      *
261      * @return true if the filltype is one of the INVERSE variants
262      */
isInverseFillType()263     public boolean isInverseFillType() {
264         final int ft = native_getFillType(mNativePath);
265         return (ft & FillType.INVERSE_WINDING.nativeInt) != 0;
266     }
267 
268     /**
269      * Toggles the INVERSE state of the filltype
270      */
toggleInverseFillType()271     public void toggleInverseFillType() {
272         int ft = native_getFillType(mNativePath);
273         ft ^= FillType.INVERSE_WINDING.nativeInt;
274         native_setFillType(mNativePath, ft);
275     }
276 
277     /**
278      * Returns true if the path is empty (contains no lines or curves)
279      *
280      * @return true if the path is empty (contains no lines or curves)
281      */
isEmpty()282     public boolean isEmpty() {
283         return native_isEmpty(mNativePath);
284     }
285 
286     /**
287      * Returns true if the path specifies a rectangle. If so, and if rect is
288      * not null, set rect to the bounds of the path. If the path does not
289      * specify a rectangle, return false and ignore rect.
290      *
291      * @param rect If not null, returns the bounds of the path if it specifies
292      *             a rectangle
293      * @return     true if the path specifies a rectangle
294      */
isRect(RectF rect)295     public boolean isRect(RectF rect) {
296         return native_isRect(mNativePath, rect);
297     }
298 
299     /**
300      * Compute the bounds of the control points of the path, and write the
301      * answer into bounds. If the path contains 0 or 1 points, the bounds is
302      * set to (0,0,0,0)
303      *
304      * @param bounds Returns the computed bounds of the path's control points.
305      * @param exact This parameter is no longer used.
306      */
307     @SuppressWarnings({"UnusedDeclaration"})
computeBounds(RectF bounds, boolean exact)308     public void computeBounds(RectF bounds, boolean exact) {
309         native_computeBounds(mNativePath, bounds);
310     }
311 
312     /**
313      * Hint to the path to prepare for adding more points. This can allow the
314      * path to more efficiently allocate its storage.
315      *
316      * @param extraPtCount The number of extra points that may be added to this
317      *                     path
318      */
incReserve(int extraPtCount)319     public void incReserve(int extraPtCount) {
320         native_incReserve(mNativePath, extraPtCount);
321     }
322 
323     /**
324      * Set the beginning of the next contour to the point (x,y).
325      *
326      * @param x The x-coordinate of the start of a new contour
327      * @param y The y-coordinate of the start of a new contour
328      */
moveTo(float x, float y)329     public void moveTo(float x, float y) {
330         native_moveTo(mNativePath, x, y);
331     }
332 
333     /**
334      * Set the beginning of the next contour relative to the last point on the
335      * previous contour. If there is no previous contour, this is treated the
336      * same as moveTo().
337      *
338      * @param dx The amount to add to the x-coordinate of the end of the
339      *           previous contour, to specify the start of a new contour
340      * @param dy The amount to add to the y-coordinate of the end of the
341      *           previous contour, to specify the start of a new contour
342      */
rMoveTo(float dx, float dy)343     public void rMoveTo(float dx, float dy) {
344         native_rMoveTo(mNativePath, dx, dy);
345     }
346 
347     /**
348      * Add a line from the last point to the specified point (x,y).
349      * If no moveTo() call has been made for this contour, the first point is
350      * automatically set to (0,0).
351      *
352      * @param x The x-coordinate of the end of a line
353      * @param y The y-coordinate of the end of a line
354      */
lineTo(float x, float y)355     public void lineTo(float x, float y) {
356         isSimplePath = false;
357         native_lineTo(mNativePath, x, y);
358     }
359 
360     /**
361      * Same as lineTo, but the coordinates are considered relative to the last
362      * point on this contour. If there is no previous point, then a moveTo(0,0)
363      * is inserted automatically.
364      *
365      * @param dx The amount to add to the x-coordinate of the previous point on
366      *           this contour, to specify a line
367      * @param dy The amount to add to the y-coordinate of the previous point on
368      *           this contour, to specify a line
369      */
rLineTo(float dx, float dy)370     public void rLineTo(float dx, float dy) {
371         isSimplePath = false;
372         native_rLineTo(mNativePath, dx, dy);
373     }
374 
375     /**
376      * Add a quadratic bezier from the last point, approaching control point
377      * (x1,y1), and ending at (x2,y2). If no moveTo() call has been made for
378      * this contour, the first point is automatically set to (0,0).
379      *
380      * @param x1 The x-coordinate of the control point on a quadratic curve
381      * @param y1 The y-coordinate of the control point on a quadratic curve
382      * @param x2 The x-coordinate of the end point on a quadratic curve
383      * @param y2 The y-coordinate of the end point on a quadratic curve
384      */
quadTo(float x1, float y1, float x2, float y2)385     public void quadTo(float x1, float y1, float x2, float y2) {
386         isSimplePath = false;
387         native_quadTo(mNativePath, x1, y1, x2, y2);
388     }
389 
390     /**
391      * Same as quadTo, but the coordinates are considered relative to the last
392      * point on this contour. If there is no previous point, then a moveTo(0,0)
393      * is inserted automatically.
394      *
395      * @param dx1 The amount to add to the x-coordinate of the last point on
396      *            this contour, for the control point of a quadratic curve
397      * @param dy1 The amount to add to the y-coordinate of the last point on
398      *            this contour, for the control point of a quadratic curve
399      * @param dx2 The amount to add to the x-coordinate of the last point on
400      *            this contour, for the end point of a quadratic curve
401      * @param dy2 The amount to add to the y-coordinate of the last point on
402      *            this contour, for the end point of a quadratic curve
403      */
rQuadTo(float dx1, float dy1, float dx2, float dy2)404     public void rQuadTo(float dx1, float dy1, float dx2, float dy2) {
405         isSimplePath = false;
406         native_rQuadTo(mNativePath, dx1, dy1, dx2, dy2);
407     }
408 
409     /**
410      * Add a cubic bezier from the last point, approaching control points
411      * (x1,y1) and (x2,y2), and ending at (x3,y3). If no moveTo() call has been
412      * made for this contour, the first point is automatically set to (0,0).
413      *
414      * @param x1 The x-coordinate of the 1st control point on a cubic curve
415      * @param y1 The y-coordinate of the 1st control point on a cubic curve
416      * @param x2 The x-coordinate of the 2nd control point on a cubic curve
417      * @param y2 The y-coordinate of the 2nd control point on a cubic curve
418      * @param x3 The x-coordinate of the end point on a cubic curve
419      * @param y3 The y-coordinate of the end point on a cubic curve
420      */
cubicTo(float x1, float y1, float x2, float y2, float x3, float y3)421     public void cubicTo(float x1, float y1, float x2, float y2,
422                         float x3, float y3) {
423         isSimplePath = false;
424         native_cubicTo(mNativePath, x1, y1, x2, y2, x3, y3);
425     }
426 
427     /**
428      * Same as cubicTo, but the coordinates are considered relative to the
429      * current point on this contour. If there is no previous point, then a
430      * moveTo(0,0) is inserted automatically.
431      */
rCubicTo(float x1, float y1, float x2, float y2, float x3, float y3)432     public void rCubicTo(float x1, float y1, float x2, float y2,
433                          float x3, float y3) {
434         isSimplePath = false;
435         native_rCubicTo(mNativePath, x1, y1, x2, y2, x3, y3);
436     }
437 
438     /**
439      * Append the specified arc to the path as a new contour. If the start of
440      * the path is different from the path's current last point, then an
441      * automatic lineTo() is added to connect the current contour to the
442      * start of the arc. However, if the path is empty, then we call moveTo()
443      * with the first point of the arc.
444      *
445      * @param oval        The bounds of oval defining shape and size of the arc
446      * @param startAngle  Starting angle (in degrees) where the arc begins
447      * @param sweepAngle  Sweep angle (in degrees) measured clockwise, treated
448      *                    mod 360.
449      * @param forceMoveTo If true, always begin a new contour with the arc
450      */
arcTo(RectF oval, float startAngle, float sweepAngle, boolean forceMoveTo)451     public void arcTo(RectF oval, float startAngle, float sweepAngle,
452                       boolean forceMoveTo) {
453         arcTo(oval.left, oval.top, oval.right, oval.bottom, startAngle, sweepAngle, forceMoveTo);
454     }
455 
456     /**
457      * Append the specified arc to the path as a new contour. If the start of
458      * the path is different from the path's current last point, then an
459      * automatic lineTo() is added to connect the current contour to the
460      * start of the arc. However, if the path is empty, then we call moveTo()
461      * with the first point of the arc.
462      *
463      * @param oval        The bounds of oval defining shape and size of the arc
464      * @param startAngle  Starting angle (in degrees) where the arc begins
465      * @param sweepAngle  Sweep angle (in degrees) measured clockwise
466      */
arcTo(RectF oval, float startAngle, float sweepAngle)467     public void arcTo(RectF oval, float startAngle, float sweepAngle) {
468         arcTo(oval.left, oval.top, oval.right, oval.bottom, startAngle, sweepAngle, false);
469     }
470 
471     /**
472      * Append the specified arc to the path as a new contour. If the start of
473      * the path is different from the path's current last point, then an
474      * automatic lineTo() is added to connect the current contour to the
475      * start of the arc. However, if the path is empty, then we call moveTo()
476      * with the first point of the arc.
477      *
478      * @param startAngle  Starting angle (in degrees) where the arc begins
479      * @param sweepAngle  Sweep angle (in degrees) measured clockwise, treated
480      *                    mod 360.
481      * @param forceMoveTo If true, always begin a new contour with the arc
482      */
arcTo(float left, float top, float right, float bottom, float startAngle, float sweepAngle, boolean forceMoveTo)483     public void arcTo(float left, float top, float right, float bottom, float startAngle,
484             float sweepAngle, boolean forceMoveTo) {
485         isSimplePath = false;
486         native_arcTo(mNativePath, left, top, right, bottom, startAngle, sweepAngle, forceMoveTo);
487     }
488 
489     /**
490      * Close the current contour. If the current point is not equal to the
491      * first point of the contour, a line segment is automatically added.
492      */
close()493     public void close() {
494         isSimplePath = false;
495         native_close(mNativePath);
496     }
497 
498     /**
499      * Specifies how closed shapes (e.g. rects, ovals) are oriented when they
500      * are added to a path.
501      */
502     public enum Direction {
503         /** clockwise */
504         CW  (0),    // must match enum in SkPath.h
505         /** counter-clockwise */
506         CCW (1);    // must match enum in SkPath.h
507 
Direction(int ni)508         Direction(int ni) {
509             nativeInt = ni;
510         }
511         final int nativeInt;
512     }
513 
detectSimplePath(float left, float top, float right, float bottom, Direction dir)514     private void detectSimplePath(float left, float top, float right, float bottom, Direction dir) {
515         if (mLastDirection == null) {
516             mLastDirection = dir;
517         }
518         if (mLastDirection != dir) {
519             isSimplePath = false;
520         } else {
521             if (rects == null) rects = new Region();
522             rects.op((int) left, (int) top, (int) right, (int) bottom, Region.Op.UNION);
523         }
524     }
525 
526     /**
527      * Add a closed rectangle contour to the path
528      *
529      * @param rect The rectangle to add as a closed contour to the path
530      * @param dir  The direction to wind the rectangle's contour
531      */
addRect(RectF rect, Direction dir)532     public void addRect(RectF rect, Direction dir) {
533         addRect(rect.left, rect.top, rect.right, rect.bottom, dir);
534     }
535 
536     /**
537      * Add a closed rectangle contour to the path
538      *
539      * @param left   The left side of a rectangle to add to the path
540      * @param top    The top of a rectangle to add to the path
541      * @param right  The right side of a rectangle to add to the path
542      * @param bottom The bottom of a rectangle to add to the path
543      * @param dir    The direction to wind the rectangle's contour
544      */
addRect(float left, float top, float right, float bottom, Direction dir)545     public void addRect(float left, float top, float right, float bottom, Direction dir) {
546         detectSimplePath(left, top, right, bottom, dir);
547         native_addRect(mNativePath, left, top, right, bottom, dir.nativeInt);
548     }
549 
550     /**
551      * Add a closed oval contour to the path
552      *
553      * @param oval The bounds of the oval to add as a closed contour to the path
554      * @param dir  The direction to wind the oval's contour
555      */
addOval(RectF oval, Direction dir)556     public void addOval(RectF oval, Direction dir) {
557         addOval(oval.left, oval.top, oval.right, oval.bottom, dir);
558     }
559 
560     /**
561      * Add a closed oval contour to the path
562      *
563      * @param dir The direction to wind the oval's contour
564      */
addOval(float left, float top, float right, float bottom, Direction dir)565     public void addOval(float left, float top, float right, float bottom, Direction dir) {
566         isSimplePath = false;
567         native_addOval(mNativePath, left, top, right, bottom, dir.nativeInt);
568     }
569 
570     /**
571      * Add a closed circle contour to the path
572      *
573      * @param x   The x-coordinate of the center of a circle to add to the path
574      * @param y   The y-coordinate of the center of a circle to add to the path
575      * @param radius The radius of a circle to add to the path
576      * @param dir    The direction to wind the circle's contour
577      */
addCircle(float x, float y, float radius, Direction dir)578     public void addCircle(float x, float y, float radius, Direction dir) {
579         isSimplePath = false;
580         native_addCircle(mNativePath, x, y, radius, dir.nativeInt);
581     }
582 
583     /**
584      * Add the specified arc to the path as a new contour.
585      *
586      * @param oval The bounds of oval defining the shape and size of the arc
587      * @param startAngle Starting angle (in degrees) where the arc begins
588      * @param sweepAngle Sweep angle (in degrees) measured clockwise
589      */
addArc(RectF oval, float startAngle, float sweepAngle)590     public void addArc(RectF oval, float startAngle, float sweepAngle) {
591         addArc(oval.left, oval.top, oval.right, oval.bottom, startAngle, sweepAngle);
592     }
593 
594     /**
595      * Add the specified arc to the path as a new contour.
596      *
597      * @param startAngle Starting angle (in degrees) where the arc begins
598      * @param sweepAngle Sweep angle (in degrees) measured clockwise
599      */
addArc(float left, float top, float right, float bottom, float startAngle, float sweepAngle)600     public void addArc(float left, float top, float right, float bottom, float startAngle,
601             float sweepAngle) {
602         isSimplePath = false;
603         native_addArc(mNativePath, left, top, right, bottom, startAngle, sweepAngle);
604     }
605 
606     /**
607         * Add a closed round-rectangle contour to the path
608      *
609      * @param rect The bounds of a round-rectangle to add to the path
610      * @param rx   The x-radius of the rounded corners on the round-rectangle
611      * @param ry   The y-radius of the rounded corners on the round-rectangle
612      * @param dir  The direction to wind the round-rectangle's contour
613      */
addRoundRect(RectF rect, float rx, float ry, Direction dir)614     public void addRoundRect(RectF rect, float rx, float ry, Direction dir) {
615         addRoundRect(rect.left, rect.top, rect.right, rect.bottom, rx, ry, dir);
616     }
617 
618     /**
619      * Add a closed round-rectangle contour to the path
620      *
621      * @param rx   The x-radius of the rounded corners on the round-rectangle
622      * @param ry   The y-radius of the rounded corners on the round-rectangle
623      * @param dir  The direction to wind the round-rectangle's contour
624      */
addRoundRect(float left, float top, float right, float bottom, float rx, float ry, Direction dir)625     public void addRoundRect(float left, float top, float right, float bottom, float rx, float ry,
626             Direction dir) {
627         isSimplePath = false;
628         native_addRoundRect(mNativePath, left, top, right, bottom, rx, ry, dir.nativeInt);
629     }
630 
631     /**
632      * Add a closed round-rectangle contour to the path. Each corner receives
633      * two radius values [X, Y]. The corners are ordered top-left, top-right,
634      * bottom-right, bottom-left
635      *
636      * @param rect The bounds of a round-rectangle to add to the path
637      * @param radii Array of 8 values, 4 pairs of [X,Y] radii
638      * @param dir  The direction to wind the round-rectangle's contour
639      */
addRoundRect(RectF rect, float[] radii, Direction dir)640     public void addRoundRect(RectF rect, float[] radii, Direction dir) {
641         if (rect == null) {
642             throw new NullPointerException("need rect parameter");
643         }
644         addRoundRect(rect.left, rect.top, rect.right, rect.bottom, radii, dir);
645     }
646 
647     /**
648      * Add a closed round-rectangle contour to the path. Each corner receives
649      * two radius values [X, Y]. The corners are ordered top-left, top-right,
650      * bottom-right, bottom-left
651      *
652      * @param radii Array of 8 values, 4 pairs of [X,Y] radii
653      * @param dir  The direction to wind the round-rectangle's contour
654      */
addRoundRect(float left, float top, float right, float bottom, float[] radii, Direction dir)655     public void addRoundRect(float left, float top, float right, float bottom, float[] radii,
656             Direction dir) {
657         if (radii.length < 8) {
658             throw new ArrayIndexOutOfBoundsException("radii[] needs 8 values");
659         }
660         isSimplePath = false;
661         native_addRoundRect(mNativePath, left, top, right, bottom, radii, dir.nativeInt);
662     }
663 
664     /**
665      * Add a copy of src to the path, offset by (dx,dy)
666      *
667      * @param src The path to add as a new contour
668      * @param dx  The amount to translate the path in X as it is added
669      */
addPath(Path src, float dx, float dy)670     public void addPath(Path src, float dx, float dy) {
671         isSimplePath = false;
672         native_addPath(mNativePath, src.mNativePath, dx, dy);
673     }
674 
675     /**
676      * Add a copy of src to the path
677      *
678      * @param src The path that is appended to the current path
679      */
addPath(Path src)680     public void addPath(Path src) {
681         isSimplePath = false;
682         native_addPath(mNativePath, src.mNativePath);
683     }
684 
685     /**
686      * Add a copy of src to the path, transformed by matrix
687      *
688      * @param src The path to add as a new contour
689      */
addPath(Path src, Matrix matrix)690     public void addPath(Path src, Matrix matrix) {
691         if (!src.isSimplePath) isSimplePath = false;
692         native_addPath(mNativePath, src.mNativePath, matrix.native_instance);
693     }
694 
695     /**
696      * Offset the path by (dx,dy)
697      *
698      * @param dx  The amount in the X direction to offset the entire path
699      * @param dy  The amount in the Y direction to offset the entire path
700      * @param dst The translated path is written here. If this is null, then
701      *            the original path is modified.
702      */
offset(float dx, float dy, @Nullable Path dst)703     public void offset(float dx, float dy, @Nullable Path dst) {
704         if (dst != null) {
705             dst.set(this);
706         } else {
707             dst = this;
708         }
709         dst.offset(dx, dy);
710     }
711 
712     /**
713      * Offset the path by (dx,dy)
714      *
715      * @param dx The amount in the X direction to offset the entire path
716      * @param dy The amount in the Y direction to offset the entire path
717      */
offset(float dx, float dy)718     public void offset(float dx, float dy) {
719         if (isSimplePath && rects == null) {
720             // nothing to offset
721             return;
722         }
723         if (isSimplePath && dx == Math.rint(dx) && dy == Math.rint(dy)) {
724             rects.translate((int) dx, (int) dy);
725         } else {
726             isSimplePath = false;
727         }
728         native_offset(mNativePath, dx, dy);
729     }
730 
731     /**
732      * Sets the last point of the path.
733      *
734      * @param dx The new X coordinate for the last point
735      * @param dy The new Y coordinate for the last point
736      */
setLastPoint(float dx, float dy)737     public void setLastPoint(float dx, float dy) {
738         isSimplePath = false;
739         native_setLastPoint(mNativePath, dx, dy);
740     }
741 
742     /**
743      * Transform the points in this path by matrix, and write the answer
744      * into dst. If dst is null, then the the original path is modified.
745      *
746      * @param matrix The matrix to apply to the path
747      * @param dst    The transformed path is written here. If dst is null,
748      *               then the the original path is modified
749      */
transform(Matrix matrix, Path dst)750     public void transform(Matrix matrix, Path dst) {
751         long dstNative = 0;
752         if (dst != null) {
753             dst.isSimplePath = false;
754             dstNative = dst.mNativePath;
755         }
756         native_transform(mNativePath, matrix.native_instance, dstNative);
757     }
758 
759     /**
760      * Transform the points in this path by matrix.
761      *
762      * @param matrix The matrix to apply to the path
763      */
transform(Matrix matrix)764     public void transform(Matrix matrix) {
765         isSimplePath = false;
766         native_transform(mNativePath, matrix.native_instance);
767     }
768 
finalize()769     protected void finalize() throws Throwable {
770         try {
771             finalizer(mNativePath);
772             mNativePath = 0;  //  Other finalizers can still call us.
773         } finally {
774             super.finalize();
775         }
776     }
777 
readOnlyNI()778     final long readOnlyNI() {
779         return mNativePath;
780     }
781 
mutateNI()782     final long mutateNI() {
783         isSimplePath = false;
784         return mNativePath;
785     }
786 
787     /**
788      * Approximate the <code>Path</code> with a series of line segments.
789      * This returns float[] with the array containing point components.
790      * There are three components for each point, in order:
791      * <ul>
792      *     <li>Fraction along the length of the path that the point resides</li>
793      *     <li>The x coordinate of the point</li>
794      *     <li>The y coordinate of the point</li>
795      * </ul>
796      * <p>Two points may share the same fraction along its length when there is
797      * a move action within the Path.</p>
798      *
799      * @param acceptableError The acceptable error for a line on the
800      *                        Path. Typically this would be 0.5 so that
801      *                        the error is less than half a pixel.
802      * @return An array of components for points approximating the Path.
803      * @hide
804      */
approximate(float acceptableError)805     public float[] approximate(float acceptableError) {
806         return native_approximate(mNativePath, acceptableError);
807     }
808 
init1()809     private static native long init1();
init2(long nPath)810     private static native long init2(long nPath);
native_reset(long nPath)811     private static native void native_reset(long nPath);
native_rewind(long nPath)812     private static native void native_rewind(long nPath);
native_set(long native_dst, long native_src)813     private static native void native_set(long native_dst, long native_src);
native_isConvex(long nPath)814     private static native boolean native_isConvex(long nPath);
native_getFillType(long nPath)815     private static native int native_getFillType(long nPath);
native_setFillType(long nPath, int ft)816     private static native void native_setFillType(long nPath, int ft);
native_isEmpty(long nPath)817     private static native boolean native_isEmpty(long nPath);
native_isRect(long nPath, RectF rect)818     private static native boolean native_isRect(long nPath, RectF rect);
native_computeBounds(long nPath, RectF bounds)819     private static native void native_computeBounds(long nPath, RectF bounds);
native_incReserve(long nPath, int extraPtCount)820     private static native void native_incReserve(long nPath, int extraPtCount);
native_moveTo(long nPath, float x, float y)821     private static native void native_moveTo(long nPath, float x, float y);
native_rMoveTo(long nPath, float dx, float dy)822     private static native void native_rMoveTo(long nPath, float dx, float dy);
native_lineTo(long nPath, float x, float y)823     private static native void native_lineTo(long nPath, float x, float y);
native_rLineTo(long nPath, float dx, float dy)824     private static native void native_rLineTo(long nPath, float dx, float dy);
native_quadTo(long nPath, float x1, float y1, float x2, float y2)825     private static native void native_quadTo(long nPath, float x1, float y1,
826                                              float x2, float y2);
native_rQuadTo(long nPath, float dx1, float dy1, float dx2, float dy2)827     private static native void native_rQuadTo(long nPath, float dx1, float dy1,
828                                               float dx2, float dy2);
native_cubicTo(long nPath, float x1, float y1, float x2, float y2, float x3, float y3)829     private static native void native_cubicTo(long nPath, float x1, float y1,
830                                         float x2, float y2, float x3, float y3);
native_rCubicTo(long nPath, float x1, float y1, float x2, float y2, float x3, float y3)831     private static native void native_rCubicTo(long nPath, float x1, float y1,
832                                         float x2, float y2, float x3, float y3);
native_arcTo(long nPath, float left, float top, float right, float bottom, float startAngle, float sweepAngle, boolean forceMoveTo)833     private static native void native_arcTo(long nPath, float left, float top,
834                                             float right, float bottom, float startAngle,
835                                             float sweepAngle, boolean forceMoveTo);
native_close(long nPath)836     private static native void native_close(long nPath);
native_addRect(long nPath, float left, float top, float right, float bottom, int dir)837     private static native void native_addRect(long nPath, float left, float top,
838                                             float right, float bottom, int dir);
native_addOval(long nPath, float left, float top, float right, float bottom, int dir)839     private static native void native_addOval(long nPath, float left, float top,
840             float right, float bottom, int dir);
native_addCircle(long nPath, float x, float y, float radius, int dir)841     private static native void native_addCircle(long nPath, float x, float y, float radius, int dir);
native_addArc(long nPath, float left, float top, float right, float bottom, float startAngle, float sweepAngle)842     private static native void native_addArc(long nPath, float left, float top,
843                                              float right, float bottom,
844                                              float startAngle, float sweepAngle);
native_addRoundRect(long nPath, float left, float top, float right, float bottom, float rx, float ry, int dir)845     private static native void native_addRoundRect(long nPath, float left, float top,
846                                                    float right, float bottom,
847                                                    float rx, float ry, int dir);
native_addRoundRect(long nPath, float left, float top, float right, float bottom, float[] radii, int dir)848     private static native void native_addRoundRect(long nPath, float left, float top,
849                                                    float right, float bottom,
850                                                    float[] radii, int dir);
native_addPath(long nPath, long src, float dx, float dy)851     private static native void native_addPath(long nPath, long src, float dx, float dy);
native_addPath(long nPath, long src)852     private static native void native_addPath(long nPath, long src);
native_addPath(long nPath, long src, long matrix)853     private static native void native_addPath(long nPath, long src, long matrix);
native_offset(long nPath, float dx, float dy)854     private static native void native_offset(long nPath, float dx, float dy);
native_setLastPoint(long nPath, float dx, float dy)855     private static native void native_setLastPoint(long nPath, float dx, float dy);
native_transform(long nPath, long matrix, long dst_path)856     private static native void native_transform(long nPath, long matrix, long dst_path);
native_transform(long nPath, long matrix)857     private static native void native_transform(long nPath, long matrix);
native_op(long path1, long path2, int op, long result)858     private static native boolean native_op(long path1, long path2, int op, long result);
finalizer(long nPath)859     private static native void finalizer(long nPath);
native_approximate(long nPath, float error)860     private static native float[] native_approximate(long nPath, float error);
861 }
862