#Topic Path
#Alias Path_Reference ##
#Alias Paths ##
#Class SkPath
#Code
#Populate
##
Paths contain geometry. Paths may be empty, or contain one or more Verbs that
outline a figure. Path always starts with a move verb to a Cartesian_Coordinate,
and may be followed by additional verbs that add lines or curves.
Adding a close verb makes the geometry into a continuous loop, a closed contour.
Paths may contain any number of contours, each beginning with a move verb.
Path contours may contain only a move verb, or may also contain lines,
Quadratic_Beziers, Conics, and Cubic_Beziers. Path contours may be open or
closed.
When used to draw a filled area, Path describes whether the fill is inside or
outside the geometry. Path also describes the winding rule used to fill
overlapping contours.
Internally, Path lazily computes metrics likes bounds and convexity. Call
SkPath::updateBoundsCache to make Path thread safe.
#Subtopic Verb
#Alias Verbs ##
#Line # line and curve type ##
#Enum Verb
#Line # controls how Path Points are interpreted ##
#Code
enum Verb {
kMove_Verb,
kLine_Verb,
kQuad_Verb,
kConic_Verb,
kCubic_Verb,
kClose_Verb,
kDone_Verb,
};
##
Verb instructs Path how to interpret one or more Point and optional Conic_Weight;
manage Contour, and terminate Path.
#Const kMove_Verb 0
#Line # starts new Contour at next Point ##
Consecutive kMove_Verb are preserved but all but the last kMove_Verb is
ignored. kMove_Verb after other Verbs implicitly closes the previous Contour
if SkPaint::kFill_Style is set when drawn; otherwise, stroke is drawn open.
kMove_Verb as the last Verb is preserved but ignored.
##
#Const kLine_Verb 1
#Line # adds Line from Last_Point to next Point ##
Line is a straight segment from Point to Point. Consecutive kLine_Verb
extend Contour. kLine_Verb at same position as prior kMove_Verb is
preserved, and draws Point if SkPaint::kStroke_Style is set, and
SkPaint::Cap is SkPaint::kSquare_Cap or SkPaint::kRound_Cap. kLine_Verb
at same position as prior line or curve Verb is preserved but is ignored.
##
#Const kQuad_Verb 2
#Line # adds Quad from Last_Point ##
Adds Quad from Last_Point, using control Point, and end Point.
Quad is a parabolic section within tangents from Last_Point to control Point,
and control Point to end Point.
##
#Const kConic_Verb 3
#Line # adds Conic from Last_Point ##
Adds Conic from Last_Point, using control Point, end Point, and Conic_Weight.
Conic is a elliptical, parabolic, or hyperbolic section within tangents
from Last_Point to control Point, and control Point to end Point, constrained
by Conic_Weight. Conic_Weight less than one is elliptical; equal to one is
parabolic (and identical to Quad); greater than one hyperbolic.
##
#Const kCubic_Verb 4
#Line # adds Cubic from Last_Point ##
Adds Cubic from Last_Point, using two control Points, and end Point.
Cubic is a third-order Bezier_Curve section within tangents from Last_Point
to first control Point, and from second control Point to end Point.
##
#Const kClose_Verb 5
#Line # closes Contour ##
Closes Contour, connecting Last_Point to kMove_Verb Point. Consecutive
kClose_Verb are preserved but only first has an effect. kClose_Verb after
kMove_Verb has no effect.
##
#Const kDone_Verb 6
#Line # terminates Path ##
Not in Verb_Array, but returned by Path iterator.
##
Each Verb has zero or more Points stored in Path.
Path iterator returns complete curve descriptions, duplicating shared Points
for consecutive entries.
#Table
#Legend
# Verb # Allocated Points # Iterated Points # Weights ##
##
# kMove_Verb # 1 # 1 # 0 ##
# kLine_Verb # 1 # 2 # 0 ##
# kQuad_Verb # 2 # 3 # 0 ##
# kConic_Verb # 2 # 3 # 1 ##
# kCubic_Verb # 3 # 4 # 0 ##
# kClose_Verb # 0 # 1 # 0 ##
# kDone_Verb # -- # 0 # 0 ##
##
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.lineTo(20, 20);
path.quadTo(-10, -10, 30, 30);
path.close();
path.cubicTo(1, 2, 3, 4, 5, 6);
path.conicTo(0, 0, 0, 0, 2);
uint8_t verbs[7];
int count = path.getVerbs(verbs, (int) SK_ARRAY_COUNT(verbs));
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close" };
SkDebugf("verb count: %d\nverbs: ", count);
for (int i = 0; i < count; ++i) {
SkDebugf("k%s_Verb ", verbStr[verbs[i]]);
}
SkDebugf("\n");
}
#StdOut
verb count: 7
verbs: kMove_Verb kLine_Verb kQuad_Verb kClose_Verb kMove_Verb kCubic_Verb kConic_Verb
##
##
#Enum Verb ##
#Subtopic Verb ##
# ------------------------------------------------------------------------------
#Subtopic Direction
#Line # contour orientation, clockwise or counterclockwise ##
#Alias Directions ##
#Enum Direction
#Line # sets Contour clockwise or counterclockwise ##
#Code
enum Direction : int {
kCW_Direction,
kCCW_Direction,
};
##
Direction describes whether Contour is clockwise or counterclockwise.
When Path contains multiple overlapping Contours, Direction together with
Fill_Type determines whether overlaps are filled or form holes.
Direction also determines how Contour is measured. For instance, dashing
measures along Path to determine where to start and stop stroke; Direction
will change dashed results as it steps clockwise or counterclockwise.
Closed Contours like Rect, Round_Rect, Circle, and Oval added with
kCW_Direction travel clockwise; the same added with kCCW_Direction
travel counterclockwise.
#Const kCW_Direction 0
#Line # contour travels clockwise ##
##
#Const kCCW_Direction 1
#Line # contour travels counterclockwise ##
##
#Example
#Height 100
void draw(SkCanvas* canvas) {
const SkPoint arrow[] = { {40, -5}, {45, 0}, {40, 5} };
const SkRect rect = {10, 10, 90, 90};
SkPaint rectPaint;
rectPaint.setAntiAlias(true);
SkPaint textPaint(rectPaint);
rectPaint.setStyle(SkPaint::kStroke_Style);
SkPaint arrowPaint(rectPaint);
SkPath arrowPath;
arrowPath.addPoly(arrow, SK_ARRAY_COUNT(arrow), true);
arrowPaint.setPathEffect(SkPath1DPathEffect::Make(arrowPath, 320, 0,
SkPath1DPathEffect::kRotate_Style));
for (auto direction : { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
canvas->drawRect(rect, rectPaint);
for (unsigned start : { 0, 1, 2, 3 } ) {
SkPath path;
path.addRect(rect, direction, start);
canvas->drawPath(path, arrowPaint);
}
canvas->drawString(SkPath::kCW_Direction == direction ? "CW" : "CCW", rect.centerX(),
rect.centerY(), textPaint);
canvas->translate(120, 0);
}
}
##
#SeeAlso arcTo rArcTo isRect isNestedFillRects addRect addOval
#Enum Direction ##
#Subtopic Direction ##
# ------------------------------------------------------------------------------
#Method SkPath()
#In Constructors
#Line # constructs with default values ##
#Populate
#Example
SkPath path;
SkDebugf("path is " "%s" "empty", path.isEmpty() ? "" : "not ");
#StdOut
path is empty
##
##
#SeeAlso reset rewind
##
# ------------------------------------------------------------------------------
#Method SkPath(const SkPath& path)
#In Constructors
#Line # makes a shallow copy ##
#Populate
#Example
#Description
Modifying one path does not effect another, even if they started as copies
of each other.
##
SkPath path;
path.lineTo(20, 20);
SkPath path2(path);
path2.close();
SkDebugf("path verbs: %d\n", path.countVerbs());
SkDebugf("path2 verbs: %d\n", path2.countVerbs());
path.reset();
SkDebugf("after reset\n" "path verbs: %d\n", path.countVerbs());
SkDebugf("path2 verbs: %d\n", path2.countVerbs());
#StdOut
path verbs: 2
path2 verbs: 3
after reset
path verbs: 0
path2 verbs: 3
##
##
#SeeAlso operator=(const SkPath& path)
##
# ------------------------------------------------------------------------------
#Method ~SkPath()
#Line # decreases Reference_Count of owned objects ##
#Populate
#Example
#Description
delete calls Path destructor, but copy of original in path2 is unaffected.
##
void draw(SkCanvas* canvas) {
SkPath* path = new SkPath();
path->lineTo(20, 20);
SkPath path2(*path);
delete path;
SkDebugf("path2 is " "%s" "empty", path2.isEmpty() ? "" : "not ");
}
##
#SeeAlso SkPath() SkPath(const SkPath& path) operator=(const SkPath& path)
##
# ------------------------------------------------------------------------------
#Method SkPath& operator=(const SkPath& path)
#Line # makes a shallow copy ##
#Populate
#Example
SkPath path1;
path1.addRect({10, 20, 30, 40});
SkPath path2 = path1;
const SkRect& b1 = path1.getBounds();
SkDebugf("path1 bounds = %g, %g, %g, %g\n", b1.fLeft, b1.fTop, b1.fRight, b1.fBottom);
const SkRect& b2 = path2.getBounds();
SkDebugf("path2 bounds = %g, %g, %g, %g\n", b2.fLeft, b2.fTop, b2.fRight, b2.fBottom);
#StdOut
path1 bounds = 10, 20, 30, 40
path2 bounds = 10, 20, 30, 40
#StdOut ##
##
#SeeAlso swap SkPath(const SkPath& path)
##
# ------------------------------------------------------------------------------
#Method bool operator==(const SkPath& a, const SkPath& b)
#Line # compares Paths for equality ##
#Populate
#Example
#Description
rewind() removes Verb_Array but leaves storage; since storage is not compared,
Path pair are equivalent.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& a, const SkPath& b) -> void {
SkDebugf("%s one %c= two\n", prefix, a == b ? '=' : '!');
};
SkPath one;
SkPath two;
debugster("empty", one, two);
one.moveTo(0, 0);
debugster("moveTo", one, two);
one.rewind();
debugster("rewind", one, two);
one.moveTo(0, 0);
one.reset();
debugster("reset", one, two);
}
#StdOut
empty one == two
moveTo one != two
rewind one == two
reset one == two
##
##
#SeeAlso operator!=(const SkPath& a, const SkPath& b) operator=(const SkPath& path)
##
# ------------------------------------------------------------------------------
#Method bool operator!=(const SkPath& a, const SkPath& b)
#Line # compares paths for inequality ##
#Populate
#Example
#Description
Path pair are equal though their convexity is not equal.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& a, const SkPath& b) -> void {
SkDebugf("%s one %c= two\n", prefix, a != b ? '!' : '=');
};
SkPath one;
SkPath two;
debugster("empty", one, two);
one.addRect({10, 20, 30, 40});
two.addRect({10, 20, 30, 40});
debugster("add rect", one, two);
one.setConvexity(SkPath::kConcave_Convexity);
debugster("setConvexity", one, two);
SkDebugf("convexity %c=\n", one.getConvexity() == two.getConvexity() ? '=' : '!');
}
#StdOut
empty one == two
add rect one == two
setConvexity one == two
convexity !=
##
##
##
# ------------------------------------------------------------------------------
#Subtopic Property
#Line # metrics and attributes ##
##
#Method bool isInterpolatable(const SkPath& compare) const
#In Property
#In Interpolate
#Line # returns if pair contains equal counts of Verb_Array and Weights ##
#Populate
#Example
SkPath path, path2;
path.moveTo(20, 20);
path.lineTo(40, 40);
path.lineTo(20, 20);
path.lineTo(40, 40);
path.close();
path2.addRect({20, 20, 40, 40});
SkDebugf("paths are " "%s" "interpolatable", path.isInterpolatable(path2) ? "" : "not ");
#StdOut
paths are interpolatable
##
##
#SeeAlso isInterpolatable
##
# ------------------------------------------------------------------------------
#Subtopic Interpolate
#Line # weighted average of Path pair ##
##
#Method bool interpolate(const SkPath& ending, SkScalar weight, SkPath* out) const
#In Interpolate
#Line # interpolates between Path pair ##
Interpolates between Paths with Point_Array of equal size.
Copy Verb_Array and Weights to out, and set out Point_Array to a weighted
average of this Point_Array and ending Point_Array, using the formula:
#Formula # (Path Point * weight) + ending Point * (1 - weight) ##.
weight is most useful when between zero (ending Point_Array) and
one (this Point_Array); will work with values outside of this
range.
interpolate() returns false and leaves out unchanged if Point_Array is not
the same size as ending Point_Array. Call isInterpolatable to check Path
compatibility prior to calling interpolate().
#Param ending Point_Array averaged with this Point_Array ##
#Param weight contribution of this Point_Array, and
one minus contribution of ending Point_Array
##
#Param out Path replaced by interpolated averages ##
#Return true if Paths contain same number of Points ##
#Example
#Height 60
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path, path2;
path.moveTo(20, 20);
path.lineTo(40, 40);
path.lineTo(20, 40);
path.lineTo(40, 20);
path.close();
path2.addRect({20, 20, 40, 40});
for (SkScalar i = 0; i <= 1; i += 1.f / 6) {
SkPath interp;
path.interpolate(path2, i, &interp);
canvas->drawPath(interp, paint);
canvas->translate(30, 0);
}
}
##
#SeeAlso isInterpolatable
##
# ------------------------------------------------------------------------------
#Subtopic Fill_Type
#Line # fill rule, normal and inverted ##
#Enum FillType
#Line # sets winding rule and inverse fill ##
#Code
enum FillType {
kWinding_FillType,
kEvenOdd_FillType,
kInverseWinding_FillType,
kInverseEvenOdd_FillType,
};
##
Fill_Type selects the rule used to fill Path. Path set to kWinding_FillType
fills if the sum of Contour edges is not zero, where clockwise edges add one, and
counterclockwise edges subtract one. Path set to kEvenOdd_FillType fills if the
number of Contour edges is odd. Each Fill_Type has an inverse variant that
reverses the rule:
kInverseWinding_FillType fills where the sum of Contour edges is zero;
kInverseEvenOdd_FillType fills where the number of Contour edges is even.
#Example
#Height 100
#Description
The top row has two clockwise rectangles. The second row has one clockwise and
one counterclockwise rectangle. The even-odd variants draw the same. The
winding variants draw the top rectangle overlap, which has a winding of 2, the
same as the outer parts of the top rectangles, which have a winding of 1.
##
void draw(SkCanvas* canvas) {
SkPath path;
path.addRect({10, 10, 30, 30}, SkPath::kCW_Direction);
path.addRect({20, 20, 40, 40}, SkPath::kCW_Direction);
path.addRect({10, 60, 30, 80}, SkPath::kCW_Direction);
path.addRect({20, 70, 40, 90}, SkPath::kCCW_Direction);
SkPaint strokePaint;
strokePaint.setStyle(SkPaint::kStroke_Style);
SkRect clipRect = {0, 0, 51, 100};
canvas->drawPath(path, strokePaint);
SkPaint fillPaint;
for (auto fillType : { SkPath::kWinding_FillType, SkPath::kEvenOdd_FillType,
SkPath::kInverseWinding_FillType, SkPath::kInverseEvenOdd_FillType } ) {
canvas->translate(51, 0);
canvas->save();
canvas->clipRect(clipRect);
path.setFillType(fillType);
canvas->drawPath(path, fillPaint);
canvas->restore();
}
}
##
#Const kWinding_FillType 0
#Line # is enclosed by a non-zero sum of Contour Directions ##
##
#Const kEvenOdd_FillType 1
#Line # is enclosed by an odd number of Contours ##
##
#Const kInverseWinding_FillType 2
#Line # is enclosed by a zero sum of Contour Directions ##
##
#Const kInverseEvenOdd_FillType 3
#Line # is enclosed by an even number of Contours ##
##
#Example
#Height 230
void draw(SkCanvas* canvas) {
SkPath path;
path.addRect({20, 10, 80, 70}, SkPath::kCW_Direction);
path.addRect({40, 30, 100, 90}, SkPath::kCW_Direction);
SkPaint strokePaint;
strokePaint.setStyle(SkPaint::kStroke_Style);
SkRect clipRect = {0, 0, 128, 128};
canvas->drawPath(path, strokePaint);
canvas->drawLine({0, 50}, {120, 50}, strokePaint);
SkPaint textPaint;
textPaint.setAntiAlias(true);
SkScalar textHPos[] = { 10, 30, 60, 90, 110 };
canvas->drawPosTextH("01210", 5, textHPos, 48, textPaint);
textPaint.setTextSize(18);
canvas->translate(0, 128);
canvas->scale(.5f, .5f);
canvas->drawString("inverse", 384, 150, textPaint);
SkPaint fillPaint;
for (auto fillType : { SkPath::kWinding_FillType, SkPath::kEvenOdd_FillType,
SkPath::kInverseWinding_FillType, SkPath::kInverseEvenOdd_FillType } ) {
canvas->save();
canvas->clipRect(clipRect);
path.setFillType(fillType);
canvas->drawPath(path, fillPaint);
canvas->restore();
canvas->drawString(fillType & 1 ? "even-odd" : "winding", 64, 170, textPaint);
canvas->translate(128, 0);
}
}
##
#SeeAlso SkPaint::Style Direction getFillType setFillType
##
# ------------------------------------------------------------------------------
#Method FillType getFillType() const
#In Fill_Type
#Line # returns Fill_Type: winding, even-odd, inverse ##
#Populate
#Example
SkPath path;
SkDebugf("default path fill type is %s\n",
path.getFillType() == SkPath::kWinding_FillType ? "kWinding_FillType" :
path.getFillType() == SkPath::kEvenOdd_FillType ? "kEvenOdd_FillType" :
path.getFillType() == SkPath::kInverseWinding_FillType ? "kInverseWinding_FillType" :
"kInverseEvenOdd_FillType");
#StdOut
default path fill type is kWinding_FillType
##
##
#SeeAlso FillType setFillType isInverseFillType
##
# ------------------------------------------------------------------------------
#Method void setFillType(FillType ft)
#In Fill_Type
#Line # sets Fill_Type: winding, even-odd, inverse ##
#Populate
#Example
#Description
If empty Path is set to inverse FillType, it fills all pixels.
##
#Height 64
SkPath path;
path.setFillType(SkPath::kInverseWinding_FillType);
SkPaint paint;
paint.setColor(SK_ColorBLUE);
canvas->drawPath(path, paint);
##
#SeeAlso FillType getFillType toggleInverseFillType
##
# ------------------------------------------------------------------------------
#Method bool isInverseFillType() const
#In Fill_Type
#Line # returns if Fill_Type fills outside geometry ##
#Populate
#Example
SkPath path;
SkDebugf("default path fill type is inverse: %s\n",
path.isInverseFillType() ? "true" : "false");
#StdOut
default path fill type is inverse: false
##
##
#SeeAlso FillType getFillType setFillType toggleInverseFillType
##
# ------------------------------------------------------------------------------
#Method void toggleInverseFillType()
#In Fill_Type
#Line # toggles Fill_Type between inside and outside geometry ##
Replaces FillType with its inverse. The inverse of FillType describes the area
unmodified by the original FillType.
#Table
#Legend
# FillType # toggled FillType ##
##
# kWinding_FillType # kInverseWinding_FillType ##
# kEvenOdd_FillType # kInverseEvenOdd_FillType ##
# kInverseWinding_FillType # kWinding_FillType ##
# kInverseEvenOdd_FillType # kEvenOdd_FillType ##
##
#Example
#Description
Path drawn normally and through its inverse touches every pixel once.
##
#Height 100
SkPath path;
SkPaint paint;
paint.setColor(SK_ColorRED);
paint.setTextSize(80);
paint.getTextPath("ABC", 3, 20, 80, &path);
canvas->drawPath(path, paint);
path.toggleInverseFillType();
paint.setColor(SK_ColorGREEN);
canvas->drawPath(path, paint);
##
#SeeAlso FillType getFillType setFillType isInverseFillType
##
#Subtopic Fill_Type ##
# ------------------------------------------------------------------------------
#Subtopic Convexity
#Line # if Path is concave or convex ##
#Enum Convexity
#Line # returns if Path is convex or concave ##
#Code
enum Convexity : uint8_t {
kUnknown_Convexity,
kConvex_Convexity,
kConcave_Convexity,
};
##
Path is convex if it contains one Contour and Contour loops no more than
360 degrees, and Contour angles all have same Direction. Convex Path
may have better performance and require fewer resources on GPU_Surface.
Path is concave when either at least one Direction change is clockwise and
another is counterclockwise, or the sum of the changes in Direction is not 360
degrees.
Initially Path Convexity is kUnknown_Convexity. Path Convexity is computed
if needed by destination Surface.
#Const kUnknown_Convexity 0
#Line # indicates Convexity has not been determined ##
##
#Const kConvex_Convexity 1
#Line # one Contour made of a simple geometry without indentations ##
##
#Const kConcave_Convexity 2
#Line # more than one Contour, or a geometry with indentations ##
##
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPoint quad[] = {{70, 70}, {20, 20}, {120, 20}, {120, 120}};
const char* labels[] = { "unknown", "convex", "concave" };
for (SkScalar x : { 40, 100 } ) {
SkPath path;
quad[0].fX = x;
path.addPoly(quad, SK_ARRAY_COUNT(quad), true);
canvas->drawPath(path, paint);
canvas->drawString(labels[(int) path.getConvexity()], 30, 100, paint);
canvas->translate(100, 100);
}
}
##
#SeeAlso Contour Direction getConvexity getConvexityOrUnknown setConvexity isConvex
#Enum Convexity ##
#Method Convexity getConvexity() const
#In Convexity
#Line # returns geometry convexity, computing if necessary ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s path convexity is %s\n", prefix,
SkPath::kUnknown_Convexity == path.getConvexity() ? "unknown" :
SkPath::kConvex_Convexity == path.getConvexity() ? "convex" : "concave"); };
SkPath path;
debugster("initial", path);
path.lineTo(50, 0);
debugster("first line", path);
path.lineTo(50, 50);
debugster("second line", path);
path.lineTo(100, 50);
debugster("third line", path);
}
##
#SeeAlso Convexity Contour Direction getConvexityOrUnknown setConvexity isConvex
##
# ------------------------------------------------------------------------------
#Method Convexity getConvexityOrUnknown() const
#In Convexity
#Line # returns geometry convexity if known ##
#Populate
#Example
#Description
Convexity is unknown unless getConvexity is called without a subsequent call
that alters the path.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s path convexity is %s\n", prefix,
SkPath::kUnknown_Convexity == path.getConvexityOrUnknown() ? "unknown" :
SkPath::kConvex_Convexity == path.getConvexityOrUnknown() ? "convex" : "concave"); };
SkPath path;
debugster("initial", path);
path.lineTo(50, 0);
debugster("first line", path);
path.getConvexity();
path.lineTo(50, 50);
debugster("second line", path);
path.lineTo(100, 50);
path.getConvexity();
debugster("third line", path);
}
##
#SeeAlso Convexity Contour Direction getConvexity setConvexity isConvex
##
# ------------------------------------------------------------------------------
#Method void setConvexity(Convexity convexity)
#In Convexity
#Line # sets if geometry is convex to avoid future computation ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s path convexity is %s\n", prefix,
SkPath::kUnknown_Convexity == path.getConvexity() ? "unknown" :
SkPath::kConvex_Convexity == path.getConvexity() ? "convex" : "concave"); };
SkPoint quad[] = {{70, 70}, {20, 20}, {120, 20}, {120, 120}};
SkPath path;
path.addPoly(quad, SK_ARRAY_COUNT(quad), true);
debugster("initial", path);
path.setConvexity(SkPath::kConcave_Convexity);
debugster("after forcing concave", path);
path.setConvexity(SkPath::kUnknown_Convexity);
debugster("after forcing unknown", path);
}
##
#SeeAlso Convexity Contour Direction getConvexity getConvexityOrUnknown isConvex
##
# ------------------------------------------------------------------------------
#Method bool isConvex() const
#In Convexity
#Line # returns if geometry is convex ##
#Populate
#Example
#Description
Concave shape is erroneously considered convex after a forced call to
setConvexity.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPoint quad[] = {{70, 70}, {20, 20}, {120, 20}, {120, 120}};
for (SkScalar x : { 40, 100 } ) {
SkPath path;
quad[0].fX = x;
path.addPoly(quad, SK_ARRAY_COUNT(quad), true);
path.setConvexity(SkPath::kConvex_Convexity);
canvas->drawPath(path, paint);
canvas->drawString(path.isConvex() ? "convex" : "not convex", 30, 100, paint);
canvas->translate(100, 100);
}
}
##
#SeeAlso Convexity Contour Direction getConvexity getConvexityOrUnknown setConvexity
##
#Subtopic Convexity ##
# ------------------------------------------------------------------------------
#Method bool isOval(SkRect* bounds) const
#In Property
#Line # returns if describes Oval ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPath path;
path.addOval({20, 20, 220, 220});
SkRect bounds;
if (path.isOval(&bounds)) {
paint.setColor(0xFF9FBFFF);
canvas->drawRect(bounds, paint);
}
paint.setColor(0x3f000000);
canvas->drawPath(path, paint);
}
##
#SeeAlso Oval addCircle addOval
##
# ------------------------------------------------------------------------------
#Method bool isRRect(SkRRect* rrect) const
#In Property
#Line # returns if describes Round_Rect ##
#Populate
#Example
#Description
Draw rounded rectangle and its bounds.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPath path;
path.addRRect(SkRRect::MakeRectXY({20, 20, 220, 220}, 30, 50));
SkRRect rrect;
if (path.isRRect(&rrect)) {
const SkRect& bounds = rrect.rect();
paint.setColor(0xFF9FBFFF);
canvas->drawRect(bounds, paint);
}
paint.setColor(0x3f000000);
canvas->drawPath(path, paint);
}
##
#SeeAlso Round_Rect addRoundRect addRRect
##
# ------------------------------------------------------------------------------
#Method SkPath& reset()
#In Constructors
#Line # removes Verb_Array, Point_Array, and Weights; frees memory ##
#Populate
#Example
SkPath path1, path2;
path1.setFillType(SkPath::kInverseWinding_FillType);
path1.addRect({10, 20, 30, 40});
SkDebugf("path1 %c= path2\n", path1 == path2 ? '=' : '!');
path1.reset();
SkDebugf("path1 %c= path2\n", path1 == path2 ? '=' : '!');
##
#SeeAlso rewind()
##
# ------------------------------------------------------------------------------
#Method SkPath& rewind()
#In Constructors
#Line # removes Verb_Array, Point_Array, and Weights, keeping memory ##
#Populate
#Example
#Description
Although path1 retains its internal storage, it is indistinguishable from
a newly initialized path.
##
SkPath path1, path2;
path1.setFillType(SkPath::kInverseWinding_FillType);
path1.addRect({10, 20, 30, 40});
SkDebugf("path1 %c= path2\n", path1 == path2 ? '=' : '!');
path1.rewind();
SkDebugf("path1 %c= path2\n", path1 == path2 ? '=' : '!');
##
#SeeAlso reset()
##
# ------------------------------------------------------------------------------
#Method bool isEmpty() const
#In Property
#Line # returns if verb count is zero ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s path is %s" "empty\n", prefix, path.isEmpty() ? "" : "not ");
};
SkPath path;
debugster("initial", path);
path.moveTo(0, 0);
debugster("after moveTo", path);
path.rewind();
debugster("after rewind", path);
path.lineTo(0, 0);
debugster("after lineTo", path);
path.reset();
debugster("after reset", path);
}
#StdOut
initial path is empty
after moveTo path is not empty
after rewind path is empty
after lineTo path is not empty
after reset path is empty
##
##
#SeeAlso SkPath() reset() rewind()
##
# ------------------------------------------------------------------------------
#Method bool isLastContourClosed() const
#In Property
#Line # returns if final Contour forms a loop ##
#Populate
#Example
#Description
close() has no effect if Path is empty; isLastContourClosed() returns
false until Path has geometry followed by close().
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s last contour is %s" "closed\n", prefix,
path.isLastContourClosed() ? "" : "not ");
};
SkPath path;
debugster("initial", path);
path.close();
debugster("after close", path);
path.lineTo(0, 0);
debugster("after lineTo", path);
path.close();
debugster("after close", path);
}
#StdOut
initial last contour is not closed
after close last contour is not closed
after lineTo last contour is not closed
after close last contour is closed
##
##
#SeeAlso close()
##
# ------------------------------------------------------------------------------
#Method bool isFinite() const
#In Property
#Line # returns if all Point values are finite ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s path is %s" "finite\n", prefix, path.isFinite() ? "" : "not ");
};
SkPath path;
debugster("initial", path);
path.lineTo(SK_ScalarMax, SK_ScalarMax);
debugster("after line", path);
SkMatrix matrix;
matrix.setScale(2, 2);
path.transform(matrix);
debugster("after scale", path);
}
#StdOut
initial path is finite
after line path is finite
after scale path is not finite
##
##
#SeeAlso SkScalar
##
# ------------------------------------------------------------------------------
#Method bool isVolatile() const
#In Property
#In Volatile
#Line # returns if Device should not cache ##
#Populate
#Example
SkPath path;
SkDebugf("volatile by default is %s\n", path.isVolatile() ? "true" : "false");
#StdOut
volatile by default is false
##
##
#SeeAlso setIsVolatile
##
# ------------------------------------------------------------------------------
#Subtopic Volatile
#Line # caching attribute ##
##
#Method void setIsVolatile(bool isVolatile)
#In Volatile
#Line # sets if Device should not cache ##
#Populate
#Example
#Height 50
#Width 50
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.setIsVolatile(true);
path.lineTo(40, 40);
canvas->drawPath(path, paint);
path.rewind();
path.moveTo(0, 40);
path.lineTo(40, 0);
canvas->drawPath(path, paint);
##
#ToDo tie example to bench to show how volatile affects speed or dm to show resource usage ##
#SeeAlso isVolatile
##
# ------------------------------------------------------------------------------
#Method static bool IsLineDegenerate(const SkPoint& p1, const SkPoint& p2, bool exact)
#In Property
#Line # returns if Line is very small ##
#Populate
#Example
#Description
As single precision floats, 100 and 100.000001 have the same bit representation,
and are exactly equal. 100 and 100.0001 have different bit representations, and
are not exactly equal, but are nearly equal.
##
void draw(SkCanvas* canvas) {
SkPoint points[] = { {100, 100}, {100.000001f, 100.000001f}, {100.0001f, 100.0001f} };
for (size_t i = 0; i < SK_ARRAY_COUNT(points) - 1; ++i) {
for (bool exact : { false, true } ) {
SkDebugf("line from (%1.8g,%1.8g) to (%1.8g,%1.8g) is %s" "degenerate, %s\n",
points[i].fX, points[i].fY, points[i + 1].fX, points[i + 1].fY,
SkPath::IsLineDegenerate(points[i], points[i + 1], exact)
? "" : "not ", exact ? "exactly" : "nearly");
}
}
}
#StdOut
line from (100,100) to (100,100) is degenerate, nearly
line from (100,100) to (100,100) is degenerate, exactly
line from (100,100) to (100.0001,100.0001) is degenerate, nearly
line from (100,100) to (100.0001,100.0001) is not degenerate, exactly
#StdOut ##
##
#SeeAlso IsQuadDegenerate IsCubicDegenerate
##
# ------------------------------------------------------------------------------
#Method static bool IsQuadDegenerate(const SkPoint& p1, const SkPoint& p2,
const SkPoint& p3, bool exact)
#In Property
#Line # returns if Quad is very small ##
#Populate
#Example
#Description
As single precision floats: 100, 100.00001, and 100.00002 have different bit representations
but nearly the same value. Translating all three by 1000 gives them the same bit representation;
the fractional portion of the number can not be represented by the float and is lost.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const SkPath& path, bool exact) -> void {
SkDebugf("quad (%1.8g,%1.8g), (%1.8g,%1.8g), (%1.8g,%1.8g) is %s" "degenerate, %s\n",
path.getPoint(0).fX, path.getPoint(0).fY, path.getPoint(1).fX,
path.getPoint(1).fY, path.getPoint(2).fX, path.getPoint(2).fY,
SkPath::IsQuadDegenerate(path.getPoint(0), path.getPoint(1), path.getPoint(2), exact) ?
"" : "not ", exact ? "exactly" : "nearly");
};
SkPath path, offset;
path.moveTo({100, 100});
path.quadTo({100.00001f, 100.00001f}, {100.00002f, 100.00002f});
offset.addPath(path, 1000, 1000);
for (bool exact : { false, true } ) {
debugster(path, exact);
debugster(offset, exact);
}
}
#StdOut
quad (100,100), (100.00001,100.00001), (100.00002,100.00002) is degenerate, nearly
quad (1100,1100), (1100,1100), (1100,1100) is degenerate, nearly
quad (100,100), (100.00001,100.00001), (100.00002,100.00002) is not degenerate, exactly
quad (1100,1100), (1100,1100), (1100,1100) is degenerate, exactly
#StdOut ##
##
#SeeAlso IsLineDegenerate IsCubicDegenerate
##
# ------------------------------------------------------------------------------
#Method static bool IsCubicDegenerate(const SkPoint& p1, const SkPoint& p2,
const SkPoint& p3, const SkPoint& p4, bool exact)
#In Property
#Line # returns if Cubic is very small ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPoint points[] = {{1, 0}, {0, 0}, {0, 0}, {0, 0}};
SkScalar step = 1;
SkScalar prior, length = 0, degenerate = 0;
do {
prior = points[0].fX;
step /= 2;
if (SkPath::IsCubicDegenerate(points[0], points[1], points[2], points[3], false)) {
degenerate = prior;
points[0].fX += step;
} else {
length = prior;
points[0].fX -= step;
}
} while (prior != points[0].fX);
SkDebugf("%1.8g is degenerate\n", degenerate);
SkDebugf("%1.8g is length\n", length);
}
#StdOut
0.00024414062 is degenerate
0.00024414065 is length
#StdOut ##
##
##
# ------------------------------------------------------------------------------
#Method bool isLine(SkPoint line[2]) const
#In Property
#Line # returns if describes Line ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkPoint line[2];
if (path.isLine(line)) {
SkDebugf("%s is line (%1.8g,%1.8g) (%1.8g,%1.8g)\n", prefix,
line[0].fX, line[0].fY, line[1].fX, line[1].fY);
} else {
SkDebugf("%s is not line\n", prefix);
}
};
SkPath path;
debugster("empty", path);
path.lineTo(0, 0);
debugster("zero line", path);
path.rewind();
path.moveTo(10, 10);
path.lineTo(20, 20);
debugster("line", path);
path.moveTo(20, 20);
debugster("second move", path);
}
#StdOut
empty is not line
zero line is line (0,0) (0,0)
line is line (10,10) (20,20)
second move is not line
##
##
##
# ------------------------------------------------------------------------------
#Subtopic Point_Array
#Line # end points and control points for lines and curves ##
#Substitute SkPoint array
Point_Array contains Points satisfying the allocated Points for
each Verb in Verb_Array. For instance, Path containing one Contour with Line
and Quad is described by Verb_Array: kMove_Verb, kLine_Verb, kQuad_Verb; and
one Point for move, one Point for Line, two Points for Quad; totaling four Points.
Point_Array may be read directly from Path with getPoints, or inspected with
getPoint, with Iter, or with RawIter.
#Method int getPoints(SkPoint points[], int max) const
#In Point_Array
#Line # returns Point_Array ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path, SkPoint* points, int max) -> void {
int count = path.getPoints(points, max);
SkDebugf("%s point count: %d ", prefix, count);
for (int i = 0; i < SkTMin(count, max) && points; ++i) {
SkDebugf("(%1.8g,%1.8g) ", points[i].fX, points[i].fY);
}
SkDebugf("\n");
};
SkPath path;
path.lineTo(20, 20);
path.lineTo(-10, -10);
SkPoint points[3];
debugster("no points", path, nullptr, 0);
debugster("zero max", path, points, 0);
debugster("too small", path, points, 2);
debugster("just right", path, points, path.countPoints());
}
#StdOut
no points point count: 3
zero max point count: 3
too small point count: 3 (0,0) (20,20)
just right point count: 3 (0,0) (20,20) (-10,-10)
##
##
#SeeAlso countPoints getPoint
##
#Method int countPoints() const
#In Point_Array
#Line # returns Point_Array length ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkDebugf("%s point count: %d\n", prefix, path.countPoints());
};
SkPath path;
debugster("empty", path);
path.lineTo(0, 0);
debugster("zero line", path);
path.rewind();
path.moveTo(10, 10);
path.lineTo(20, 20);
debugster("line", path);
path.moveTo(20, 20);
debugster("second move", path);
}
#StdOut
empty point count: 0
zero line point count: 2
line point count: 2
second move point count: 3
##
##
#SeeAlso getPoints
##
#Method SkPoint getPoint(int index) const
#In Point_Array
#Line # returns entry from Point_Array ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.lineTo(20, 20);
path.offset(-10, -10);
for (int i= 0; i < path.countPoints(); ++i) {
SkDebugf("point %d: (%1.8g,%1.8g)\n", i, path.getPoint(i).fX, path.getPoint(i).fY);
}
}
#StdOut
point 0: (-10,-10)
point 1: (10,10)
##
##
#SeeAlso countPoints getPoints
##
#Subtopic Point_Array ##
# ------------------------------------------------------------------------------
#Subtopic Verb_Array
#Line # line and curve type for points ##
Verb_Array always starts with kMove_Verb.
If kClose_Verb is not the last entry, it is always followed by kMove_Verb;
the quantity of kMove_Verb equals the Contour count.
Verb_Array does not include or count kDone_Verb; it is a convenience
returned when iterating through Verb_Array.
Verb_Array may be read directly from Path with getVerbs, or inspected with Iter,
or with RawIter.
#Method int countVerbs() const
#In Verb_Array
#Line # returns Verb_Array length ##
#Populate
#Example
SkPath path;
SkDebugf("empty verb count: %d\n", path.countVerbs());
path.addRoundRect({10, 20, 30, 40}, 5, 5);
SkDebugf("round rect verb count: %d\n", path.countVerbs());
#StdOut
empty verb count: 0
round rect verb count: 10
##
##
#SeeAlso getVerbs Iter RawIter
##
#Method int getVerbs(uint8_t verbs[], int max) const
#In Verb_Array
#Line # returns Verb_Array ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path, uint8_t* verbs, int max) -> void {
int count = path.getVerbs(verbs, max);
SkDebugf("%s verb count: %d ", prefix, count);
const char* verbStr[] = { "move", "line", "quad", "conic", "cubic", "close" };
for (int i = 0; i < SkTMin(count, max) && verbs; ++i) {
SkDebugf("%s ", verbStr[verbs[i]]);
}
SkDebugf("\n");
};
SkPath path;
path.lineTo(20, 20);
path.lineTo(-10, -10);
uint8_t verbs[3];
debugster("no verbs", path, nullptr, 0);
debugster("zero max", path, verbs, 0);
debugster("too small", path, verbs, 2);
debugster("just right", path, verbs, path.countVerbs());
}
#StdOut
no verbs verb count: 3
zero max verb count: 3
too small verb count: 3 move line
just right verb count: 3 move line line
##
##
#SeeAlso countVerbs getPoints Iter RawIter
##
#Subtopic Verb_Array ##
# ------------------------------------------------------------------------------
#Method void swap(SkPath& other)
#In Operators
#Line # exchanges Path pair ##
#Populate
#Example
SkPath path1, path2;
path1.addRect({10, 20, 30, 40});
path1.swap(path2);
const SkRect& b1 = path1.getBounds();
SkDebugf("path1 bounds = %g, %g, %g, %g\n", b1.fLeft, b1.fTop, b1.fRight, b1.fBottom);
const SkRect& b2 = path2.getBounds();
SkDebugf("path2 bounds = %g, %g, %g, %g\n", b2.fLeft, b2.fTop, b2.fRight, b2.fBottom);
#StdOut
path1 bounds = 0, 0, 0, 0
path2 bounds = 10, 20, 30, 40
#StdOut ##
##
#SeeAlso operator=(const SkPath& path)
##
# ------------------------------------------------------------------------------
#Method const SkRect& getBounds() const
#In Property
#Line # returns maximum and minimum of Point_Array ##
#Populate
#Example
#Description
Bounds of upright Circle can be predicted from center and radius.
Bounds of rotated Circle includes control Points outside of filled area.
##
auto debugster = [](const char* prefix, const SkPath& path) -> void {
const SkRect& bounds = path.getBounds();
SkDebugf("%s bounds = %g, %g, %g, %g\n", prefix,
bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
};
SkPath path;
debugster("empty", path);
path.addCircle(50, 45, 25);
debugster("circle", path);
SkMatrix matrix;
matrix.setRotate(45, 50, 45);
path.transform(matrix);
debugster("rotated circle", path);
#StdOut
empty bounds = 0, 0, 0, 0
circle bounds = 25, 20, 75, 70
rotated circle bounds = 14.6447, 9.64466, 85.3553, 80.3553
##
##
#SeeAlso computeTightBounds updateBoundsCache
##
# ------------------------------------------------------------------------------
#Subtopic Utility
#Line # rarely called management functions ##
##
#Method void updateBoundsCache() const
#In Utility
#Line # refreshes result of getBounds ##
#Populate
#Example
double times[2] = { 0, 0 };
for (int i = 0; i < 10000; ++i) {
SkPath path;
for (int j = 1; j < 100; ++ j) {
path.addCircle(50 + j, 45 + j, 25 + j);
}
if (1 & i) {
path.updateBoundsCache();
}
double start = SkTime::GetNSecs();
(void) path.getBounds();
times[1 & i] += SkTime::GetNSecs() - start;
}
SkDebugf("uncached avg: %g ms\n", times[0] * 1e-6);
SkDebugf("cached avg: %g ms\n", times[1] * 1e-6);
#StdOut
#Volatile
uncached avg: 0.18048 ms
cached avg: 0.182784 ms
##
##
#SeeAlso getBounds
#ToDo the results don't make sense, need to profile to figure this out ##
##
# ------------------------------------------------------------------------------
#Method SkRect computeTightBounds() const
#In Property
#Line # returns extent of geometry ##
#Populate
#Example
auto debugster = [](const char* prefix, const SkPath& path) -> void {
const SkRect& bounds = path.computeTightBounds();
SkDebugf("%s bounds = %g, %g, %g, %g\n", prefix,
bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom);
};
SkPath path;
debugster("empty", path);
path.addCircle(50, 45, 25);
debugster("circle", path);
SkMatrix matrix;
matrix.setRotate(45, 50, 45);
path.transform(matrix);
debugster("rotated circle", path);
#StdOut
empty bounds = 0, 0, 0, 0
circle bounds = 25, 20, 75, 70
rotated circle bounds = 25, 20, 75, 70
##
##
#SeeAlso getBounds
##
# ------------------------------------------------------------------------------
#Method bool conservativelyContainsRect(const SkRect& rect) const
#In Property
#Line # returns true if Rect may be inside ##
#Populate
#Example
#Height 140
#Description
Rect is drawn in blue if it is contained by red Path.
##
void draw(SkCanvas* canvas) {
SkPath path;
path.addRoundRect({10, 20, 54, 120}, 10, 20);
SkRect tests[] = {
{ 10, 40, 54, 80 },
{ 25, 20, 39, 120 },
{ 15, 25, 49, 115 },
{ 13, 27, 51, 113 },
};
for (unsigned i = 0; i < SK_ARRAY_COUNT(tests); ++i) {
SkPaint paint;
paint.setColor(SK_ColorRED);
canvas->drawPath(path, paint);
bool rectInPath = path.conservativelyContainsRect(tests[i]);
paint.setColor(rectInPath ? SK_ColorBLUE : SK_ColorBLACK);
canvas->drawRect(tests[i], paint);
canvas->translate(64, 0);
}
}
##
#SeeAlso contains Op Rect Convexity
##
# ------------------------------------------------------------------------------
#Method void incReserve(int extraPtCount)
#In Utility
#Line # reserves space for additional data ##
#Populate
#Example
#Height 192
void draw(SkCanvas* canvas) {
auto addPoly = [](int sides, SkScalar size, SkPath* path) -> void {
path->moveTo(size, 0);
for (int i = 1; i < sides; i++) {
SkScalar c, s = SkScalarSinCos(SK_ScalarPI * 2 * i / sides, &c);
path->lineTo(c * size, s * size);
}
path->close();
};
SkPath path;
path.incReserve(3 + 4 + 5 + 6 + 7 + 8 + 9);
for (int sides = 3; sides < 10; ++sides) {
addPoly(sides, sides, &path);
}
SkMatrix matrix;
matrix.setScale(10, 10, -10, -10);
path.transform(matrix);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
canvas->drawPath(path, paint);
}
##
#SeeAlso Point_Array
##
#Method void shrinkToFit()
#In Utility
#Line # removes unused reserved space ##
#Populate
#NoExample
#Height 192
void draw(SkCanvas* canvas) {
SkPath skinnyPath;
skinnyPath.lineTo(100, 100);
skinnyPath.shrinkToFit();
SkDebugf("no wasted Path capacity in my house!\n");
}
##
#SeeAlso incReserve
##
# ------------------------------------------------------------------------------
#Subtopic Build
#Line # adds points and verbs to path ##
##
#Method SkPath& moveTo(SkScalar x, SkScalar y)
#In Build
#Line # starts Contour ##
#Populate
#Example
#Width 140
#Height 100
void draw(SkCanvas* canvas) {
SkRect rect = { 20, 20, 120, 80 };
SkPath path;
path.addRect(rect);
path.moveTo(rect.fLeft, rect.fTop);
path.lineTo(rect.fRight, rect.fBottom);
path.moveTo(rect.fLeft, rect.fBottom);
path.lineTo(rect.fRight, rect.fTop);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour lineTo rMoveTo quadTo conicTo cubicTo close()
##
#Method SkPath& moveTo(const SkPoint& p)
#Populate
#Example
#Width 128
#Height 128
void draw(SkCanvas* canvas) {
SkPoint data[][3] = {{{30,40},{60,60},{90,30}}, {{30,120},{60,100},{90,120}},
{{60,100},{60,40},{70,30}}, {{60,40},{50,20},{70,30}}};
SkPath path;
for (unsigned i = 0; i < SK_ARRAY_COUNT(data); ++i) {
path.moveTo(data[i][0]);
path.lineTo(data[i][1]);
path.lineTo(data[i][2]);
}
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour lineTo rMoveTo quadTo conicTo cubicTo close()
##
#Method SkPath& rMoveTo(SkScalar dx, SkScalar dy)
#In Build
#Line # starts Contour relative to Last_Point ##
#Populate
#Example
#Height 100
SkPath path;
path.addRect({20, 20, 80, 80}, SkPath::kCW_Direction, 2);
path.rMoveTo(25, 2);
SkVector arrow[] = {{0, -4}, {-20, 0}, {0, -3}, {-5, 5}, {5, 5}, {0, -3}, {20, 0}};
for (unsigned i = 0; i < SK_ARRAY_COUNT(arrow); ++i) {
path.rLineTo(arrow[i].fX, arrow[i].fY);
}
SkPaint paint;
canvas->drawPath(path, paint);
SkPoint lastPt;
path.getLastPt(&lastPt);
canvas->drawString("start", lastPt.fX, lastPt.fY, paint);
##
#SeeAlso Contour lineTo moveTo quadTo conicTo cubicTo close()
##
# ------------------------------------------------------------------------------
#Method SkPath& lineTo(SkScalar x, SkScalar y)
#In Build
#Line # appends Line ##
#Populate
#Example
#Height 100
###$
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setTextSize(72);
canvas->drawString("#", 120, 80, paint);
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(5);
SkPath path;
SkPoint hash[] = {{58, 28}, {43, 80}, {37, 45}, {85, 45}};
SkVector offsets[] = {{0, 0}, {17, 0}, {0, 0}, {-5, 17}};
unsigned o = 0;
for (unsigned i = 0; i < SK_ARRAY_COUNT(hash); i += 2) {
for (unsigned j = 0; j < 2; o++, j++) {
path.moveTo(hash[i].fX + offsets[o].fX, hash[i].fY + offsets[o].fY);
path.lineTo(hash[i + 1].fX + offsets[o].fX, hash[i + 1].fY + offsets[o].fY);
}
}
canvas->drawPath(path, paint);
}
$$$#
##
#SeeAlso Contour moveTo rLineTo addRect
##
# ------------------------------------------------------------------------------
#Method SkPath& lineTo(const SkPoint& p)
#Populate
#Example
#Height 100
SkPath path;
SkVector oxo[] = {{25, 25}, {35, 35}, {25, 35}, {35, 25},
{40, 20}, {40, 80}, {60, 20}, {60, 80},
{20, 40}, {80, 40}, {20, 60}, {80, 60}};
for (unsigned i = 0; i < SK_ARRAY_COUNT(oxo); i += 2) {
path.moveTo(oxo[i]);
path.lineTo(oxo[i + 1]);
}
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
canvas->drawPath(path, paint);
##
#SeeAlso Contour moveTo rLineTo addRect
##
# ------------------------------------------------------------------------------
#Method SkPath& rLineTo(SkScalar dx, SkScalar dy)
#In Build
#Line # appends Line relative to Last_Point ##
#Populate
#Example
#Height 128
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.moveTo(10, 98);
SkScalar x = 0, y = 0;
for (int i = 10; i < 100; i += 5) {
x += i * ((i & 2) - 1);
y += i * (((i + 1) & 2) - 1);
path.rLineTo(x, y);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo lineTo addRect
##
# ------------------------------------------------------------------------------
#Subtopic Quad
#Alias Quad ##
#Alias Quads ##
#Alias Quadratic_Bezier ##
#Alias Quadratic_Beziers ##
#Line # curve described by second-order polynomial ##
Quad describes a Quadratic_Bezier, a second-order curve identical to a section
of a parabola. Quad begins at a start Point, curves towards a control Point,
and then curves to an end Point.
#Example
#Height 110
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPoint quadPts[] = {{20, 90}, {120, 10}, {220, 90}};
canvas->drawLine(quadPts[0], quadPts[1], paint);
canvas->drawLine(quadPts[1], quadPts[2], paint);
SkPath path;
path.moveTo(quadPts[0]);
path.quadTo(quadPts[1], quadPts[2]);
paint.setStrokeWidth(3);
canvas->drawPath(path, paint);
}
##
Quad is a special case of Conic where Conic_Weight is set to one.
Quad is always contained by the triangle connecting its three Points. Quad
begins tangent to the line between start Point and control Point, and ends
tangent to the line between control Point and end Point.
#Example
#Height 160
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPoint quadPts[] = {{20, 150}, {120, 10}, {220, 150}};
SkColor colors[] = { 0xff88ff00, 0xff0088bb, 0xff6600cc, 0xffbb3377 };
for (unsigned i = 0; i < SK_ARRAY_COUNT(colors); ++i) {
paint.setColor(0x7fffffff & colors[i]);
paint.setStrokeWidth(1);
canvas->drawLine(quadPts[0], quadPts[1], paint);
canvas->drawLine(quadPts[1], quadPts[2], paint);
SkPath path;
path.moveTo(quadPts[0]);
path.quadTo(quadPts[1], quadPts[2]);
paint.setStrokeWidth(3);
paint.setColor(colors[i]);
canvas->drawPath(path, paint);
quadPts[1].fY += 30;
}
}
##
#Method SkPath& quadTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2)
#In Quad
#Line # appends Quad ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.moveTo(0, -10);
for (int i = 0; i < 128; i += 16) {
path.quadTo( 10 + i, -10 - i, 10 + i, 0);
path.quadTo( 14 + i, 14 + i, 0, 14 + i);
path.quadTo(-18 - i, 18 + i, -18 - i, 0);
path.quadTo(-22 - i, -22 - i, 0, -22 - i);
}
path.offset(128, 128);
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo conicTo rQuadTo
##
#Method SkPath& quadTo(const SkPoint& p1, const SkPoint& p2)
#In Build
#In Quad
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
paint.setAntiAlias(true);
SkPath path;
SkPoint pts[] = {{128, 10}, {10, 214}, {236, 214}};
path.moveTo(pts[1]);
for (int i = 0; i < 3; ++i) {
path.quadTo(pts[i % 3], pts[(i + 2) % 3]);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo conicTo rQuadTo
##
#Method SkPath& rQuadTo(SkScalar dx1, SkScalar dy1, SkScalar dx2, SkScalar dy2)
#In Build
#In Quad
#Line # appends Quad relative to Last_Point ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
SkPath path;
path.moveTo(128, 20);
path.rQuadTo(-6, 10, -7, 10);
for (int i = 1; i < 32; i += 4) {
path.rQuadTo(10 + i, 10 + i, 10 + i * 4, 10);
path.rQuadTo(-10 - i, 10 + i, -10 - (i + 2) * 4, 10);
}
path.quadTo(92, 220, 128, 215);
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo conicTo quadTo
##
#Subtopic Quad ##
# ------------------------------------------------------------------------------
#Subtopic Conic
#Alias Conic ##
#Alias Conics ##
#Line # conic section defined by three points and a weight ##
Conic describes a conical section: a piece of an ellipse, or a piece of a
parabola, or a piece of a hyperbola. Conic begins at a start Point,
curves towards a control Point, and then curves to an end Point. The influence
of the control Point is determined by Conic_Weight.
Each Conic in Path adds two Points and one Conic_Weight. Conic_Weights in Path
may be inspected with Iter, or with RawIter.
#Subtopic Weight
#Alias Conic_Weights ##
#Alias Weights ##
#Line # strength of Conic control Point ##
Weight determines both the strength of the control Point and the type of Conic.
Weight varies from zero to infinity. At zero, Weight causes the control Point to
have no effect; Conic is identical to a line segment from start Point to end
point. If Weight is less than one, Conic follows an elliptical arc.
If Weight is exactly one, then Conic is identical to Quad; Conic follows a
parabolic arc. If Weight is greater than one, Conic follows a hyperbolic
arc. If Weight is infinity, Conic is identical to two line segments, connecting
start Point to control Point, and control Point to end Point.
#Example
#Description
When Conic_Weight is one, Quad is added to path; the two are identical.
##
void draw(SkCanvas* canvas) {
const char* verbNames[] = { "move", "line", "quad", "conic", "cubic", "close", "done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath path;
path.conicTo(20, 30, 50, 60, 1);
SkPath::Iter iter(path, false);
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("%s ", verbNames[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%g, %g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
}
#StdOut
move {0, 0},
quad {0, 0}, {20, 30}, {50, 60},
done
##
##
If weight is less than one, Conic is an elliptical segment.
#Example
#Description
A 90 degree circular arc has the weight #Formula # 1 / sqrt(2) ##.
##
void draw(SkCanvas* canvas) {
const char* verbNames[] = { "move", "line", "quad", "conic", "cubic", "close", "done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath path;
path.arcTo(20, 0, 20, 20, 20);
SkPath::Iter iter(path, false);
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("%s ", verbNames[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%g, %g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
}
#StdOut
move {0, 0},
conic {0, 0}, {20, 0}, {20, 20}, weight = 0.707107
done
##
##
If weight is greater than one, Conic is a hyperbolic segment. As weight gets large,
a hyperbolic segment can be approximated by straight lines connecting the
control Point with the end Points.
#Example
void draw(SkCanvas* canvas) {
const char* verbNames[] = { "move", "line", "quad", "conic", "cubic", "close", "done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath path;
path.conicTo(20, 0, 20, 20, SK_ScalarInfinity);
SkPath::Iter iter(path, false);
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("%s ", verbNames[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%g, %g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
}
#StdOut
move {0, 0},
line {0, 0}, {20, 0},
line {20, 0}, {20, 20},
done
##
##
#Subtopic Weight ##
#Method SkPath& conicTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2,
SkScalar w)
#In Conic
#In Build
#Line # appends Conic ##
#Populate
#Example
#Height 160
#Description
As weight increases, curve is pulled towards control point.
The bottom two curves are elliptical; the next is parabolic; the
top curve is hyperbolic.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPoint conicPts[] = {{20, 150}, {120, 10}, {220, 150}};
canvas->drawLine(conicPts[0], conicPts[1], paint);
canvas->drawLine(conicPts[1], conicPts[2], paint);
SkColor colors[] = { 0xff88ff00, 0xff0088bb, 0xff6600cc, 0xffbb3377 };
paint.setStrokeWidth(3);
SkScalar weight = 0.5f;
for (unsigned i = 0; i < SK_ARRAY_COUNT(colors); ++i) {
SkPath path;
path.moveTo(conicPts[0]);
path.conicTo(conicPts[1], conicPts[2], weight);
paint.setColor(colors[i]);
canvas->drawPath(path, paint);
weight += 0.25f;
}
}
##
#SeeAlso rConicTo arcTo addArc quadTo
##
#Method SkPath& conicTo(const SkPoint& p1, const SkPoint& p2, SkScalar w)
#In Build
#In Conic
#Populate
#Example
#Height 128
#Description
Conics and arcs use identical representations. As the arc sweep increases
the Conic_Weight also increases, but remains smaller than one.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkRect oval = {0, 20, 120, 140};
SkPath path;
for (int i = 0; i < 4; ++i) {
path.moveTo(oval.centerX(), oval.fTop);
path.arcTo(oval, -90, 90 - 20 * i, false);
oval.inset(15, 15);
}
path.offset(100, 0);
SkScalar conicWeights[] = { 0.707107f, 0.819152f, 0.906308f, 0.965926f };
SkPoint conicPts[][3] = { { {40, 20}, {100, 20}, {100, 80} },
{ {40, 35}, {71.509f, 35}, {82.286f, 64.6091f} },
{ {40, 50}, {53.9892f, 50}, {62.981f, 60.7164f} },
{ {40, 65}, {44.0192f, 65}, {47.5f, 67.0096f} } };
for (int i = 0; i < 4; ++i) {
path.moveTo(conicPts[i][0]);
path.conicTo(conicPts[i][1], conicPts[i][2], conicWeights[i]);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso rConicTo arcTo addArc quadTo
##
#Method SkPath& rConicTo(SkScalar dx1, SkScalar dy1, SkScalar dx2, SkScalar dy2,
SkScalar w)
#In Build
#In Conic
#Line # appends Conic relative to Last_Point ##
#Populate
#Example
#Height 140
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.moveTo(20, 80);
path.rConicTo( 60, 0, 60, 60, 0.707107f);
path.rConicTo( 0, -60, 60, -60, 0.707107f);
path.rConicTo(-60, 0, -60, -60, 0.707107f);
path.rConicTo( 0, 60, -60, 60, 0.707107f);
canvas->drawPath(path, paint);
}
##
#SeeAlso conicTo arcTo addArc quadTo
##
#Subtopic Conic ##
# ------------------------------------------------------------------------------
#Subtopic Cubic
#Alias Cubic ##
#Alias Cubics ##
#Alias Cubic_Bezier ##
#Alias Cubic_Beziers ##
#Line # curve described by third-order polynomial ##
Cubic describes a Bezier_Curve segment described by a third-order polynomial.
Cubic begins at a start Point, curving towards the first control Point;
and curves from the end Point towards the second control Point.
#Example
#Height 160
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPoint cubicPts[] = {{20, 150}, {90, 10}, {160, 150}, {230, 10}};
SkColor colors[] = { 0xff88ff00, 0xff0088bb, 0xff6600cc, 0xffbb3377 };
for (unsigned i = 0; i < SK_ARRAY_COUNT(colors); ++i) {
paint.setColor(0x7fffffff & colors[i]);
paint.setStrokeWidth(1);
for (unsigned j = 0; j < 3; ++j) {
canvas->drawLine(cubicPts[j], cubicPts[j + 1], paint);
}
SkPath path;
path.moveTo(cubicPts[0]);
path.cubicTo(cubicPts[1], cubicPts[2], cubicPts[3]);
paint.setStrokeWidth(3);
paint.setColor(colors[i]);
canvas->drawPath(path, paint);
cubicPts[1].fY += 30;
cubicPts[2].fX += 30;
}
}
##
#Method SkPath& cubicTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2,
SkScalar x3, SkScalar y3)
#In Build
#In Cubic
#Line # appends Cubic ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.moveTo(0, -10);
for (int i = 0; i < 128; i += 16) {
SkScalar c = i * 0.5f;
path.cubicTo( 10 + c, -10 - i, 10 + i, -10 - c, 10 + i, 0);
path.cubicTo( 14 + i, 14 + c, 14 + c, 14 + i, 0, 14 + i);
path.cubicTo(-18 - c, 18 + i, -18 - i, 18 + c, -18 - i, 0);
path.cubicTo(-22 - i, -22 - c, -22 - c, -22 - i, 0, -22 - i);
}
path.offset(128, 128);
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo rCubicTo quadTo
##
# ------------------------------------------------------------------------------
#Method SkPath& cubicTo(const SkPoint& p1, const SkPoint& p2, const SkPoint& p3)
#In Build
#In Cubic
#Populate
#Example
#Height 84
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPoint pts[] = { {20, 20}, {300, 80}, {-140, 90}, {220, 10} };
SkPath path;
path.moveTo(pts[0]);
path.cubicTo(pts[1], pts[2], pts[3]);
canvas->drawPath(path, paint);
##
#SeeAlso Contour moveTo rCubicTo quadTo
##
# ------------------------------------------------------------------------------
#Method SkPath& rCubicTo(SkScalar dx1, SkScalar dy1, SkScalar dx2, SkScalar dy2,
SkScalar dx3, SkScalar dy3)
#In Build
#In Cubic
#Line # appends Cubic relative to Last_Point ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
SkPath path;
path.moveTo(24, 108);
for (int i = 0; i < 16; i++) {
SkScalar sx, sy;
sx = SkScalarSinCos(i * SK_ScalarPI / 8, &sy);
path.rCubicTo(40 * sx, 4 * sy, 4 * sx, 40 * sy, 40 * sx, 40 * sy);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso Contour moveTo cubicTo quadTo
##
#Subtopic Cubic ##
# ------------------------------------------------------------------------------
#Subtopic Arc
#Line # part of Oval or Circle ##
Arc can be constructed in a number of ways. Arc may be described by part of Oval and angles,
by start point and end point, and by radius and tangent lines. Each construction has advantages,
and some constructions correspond to Arc drawing in graphics standards.
All Arc draws are implemented by one or more Conic draws. When Conic_Weight is less than one,
Conic describes an Arc of some Oval or Circle.
arcTo(const SkRect& oval, SkScalar startAngle, SkScalar sweepAngle, bool forceMoveTo)
describes Arc as a piece of Oval, beginning at start angle, sweeping clockwise or counterclockwise,
which may continue Contour or start a new one. This construction is similar to PostScript and
HTML_Canvas arcs. Variation addArc always starts new Contour. SkCanvas::drawArc draws without
requiring Path.
arcTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2, SkScalar radius)
describes Arc as tangent to the line segment from last Point added to Path to (x1, y1); and tangent
to the line segment from (x1, y1) to (x2, y2). This construction is similar to PostScript and
HTML_Canvas arcs.
arcTo(SkScalar rx, SkScalar ry, SkScalar xAxisRotate, ArcSize largeArc, Direction sweep,
SkScalar x, SkScalar y)
describes Arc as part of Oval with radii (rx, ry), beginning at
last Point added to Path and ending at (x, y). More than one Arc satisfies this criteria,
so additional values choose a single solution. This construction is similar to SVG arcs.
conicTo describes Arc of less than 180 degrees as a pair of tangent lines and Conic_Weight.
conicTo can represent any Arc with a sweep less than 180 degrees at any rotation. All arcTo
constructions are converted to Conic data when added to Path.
#ToDo example is spaced correctly on fiddle but spacing is too wide on pc
##
#Illustration
#List
# 1 arcTo(const SkRect& oval, SkScalar startAngle, SkScalar sweepAngle, bool forceMoveTo) ##
# 2 parameter adds move to first point ##
# 3 start angle must be multiple of 90 degrees ##
# 4 arcTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2, SkScalar radius) ##
# 5 arcTo(SkScalar rx, SkScalar ry, SkScalar xAxisRotate, ArcSize largeArc,
Direction sweep, SkScalar x, SkScalar y) ##
#List ##
#Example
#Height 128
void draw(SkCanvas* canvas) {
SkRect oval = {8, 8, 56, 56};
SkPaint ovalPaint;
ovalPaint.setAntiAlias(true);
SkPaint textPaint(ovalPaint);
ovalPaint.setStyle(SkPaint::kStroke_Style);
SkPaint arcPaint(ovalPaint);
arcPaint.setStrokeWidth(5);
arcPaint.setColor(SK_ColorBLUE);
canvas->translate(-64, 0);
for (char arcStyle = '1'; arcStyle <= '6'; ++arcStyle) {
'4' == arcStyle ? canvas->translate(-96, 55) : canvas->translate(64, 0);
canvas->drawText(&arcStyle, 1, 30, 36, textPaint);
canvas->drawOval(oval, ovalPaint);
SkPath path;
path.moveTo({56, 32});
switch (arcStyle) {
case '1':
path.arcTo(oval, 0, 90, false);
break;
case '2':
canvas->drawArc(oval, 0, 90, false, arcPaint);
continue;
case '3':
path.addArc(oval, 0, 90);
break;
case '4':
path.arcTo({56, 56}, {32, 56}, 24);
break;
case '5':
path.arcTo({24, 24}, 0, SkPath::kSmall_ArcSize, SkPath::kCW_Direction, {32, 56});
break;
case '6':
path.conicTo({56, 56}, {32, 56}, SK_ScalarRoot2Over2);
break;
}
canvas->drawPath(path, arcPaint);
}
}
#Example ##
In the example above:
#List
# 1 describes an arc from an oval, a starting angle, and a sweep angle. ##
# 2 is similar to 1, but does not require building a path to draw. ##
# 3 is similar to 1, but always begins new Contour. ##
# 4 describes an arc from a pair of tangent lines and a radius. ##
# 5 describes an arc from Oval center, arc start Point and arc end Point. ##
# 6 describes an arc from a pair of tangent lines and a Conic_Weight. ##
#List ##
#Method SkPath& arcTo(const SkRect& oval, SkScalar startAngle, SkScalar sweepAngle, bool forceMoveTo)
#In Build
#In Arc
#Line # appends Arc ##
#Populate
#Example
#Height 200
#Description
arcTo continues a previous contour when forceMoveTo is false and when Path
is not empty.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPath path;
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(4);
path.moveTo(0, 0);
path.arcTo({20, 20, 120, 120}, -90, 90, false);
canvas->drawPath(path, paint);
path.rewind();
path.arcTo({120, 20, 220, 120}, -90, 90, false);
canvas->drawPath(path, paint);
path.rewind();
path.moveTo(0, 0);
path.arcTo({20, 120, 120, 220}, -90, 90, true);
canvas->drawPath(path, paint);
}
##
#SeeAlso addArc SkCanvas::drawArc conicTo
##
# ------------------------------------------------------------------------------
#Method SkPath& arcTo(SkScalar x1, SkScalar y1, SkScalar x2, SkScalar y2, SkScalar radius)
#In Build
#In Arc
#Populate
#Example
#Height 226
void draw(SkCanvas* canvas) {
SkPaint tangentPaint;
tangentPaint.setAntiAlias(true);
SkPaint textPaint(tangentPaint);
tangentPaint.setStyle(SkPaint::kStroke_Style);
tangentPaint.setColor(SK_ColorGRAY);
SkPaint arcPaint(tangentPaint);
arcPaint.setStrokeWidth(5);
arcPaint.setColor(SK_ColorBLUE);
SkPath path;
SkPoint pts[] = { {56, 20}, {200, 20}, {90, 190} };
SkScalar radius = 50;
path.moveTo(pts[0]);
path.arcTo(pts[1], pts[2], radius);
canvas->drawLine(pts[0], pts[1], tangentPaint);
canvas->drawLine(pts[1], pts[2], tangentPaint);
SkPoint lastPt;
(void) path.getLastPt(&lastPt);
SkVector radial = pts[2] - pts[1];
radial.setLength(radius);
SkPoint center = { lastPt.fX - radial.fY, lastPt.fY + radial.fX };
canvas->drawCircle(center, radius, tangentPaint);
canvas->drawLine(lastPt, center, tangentPaint);
radial = pts[1] - pts[0];
radial.setLength(radius);
SkPoint arcStart = { center.fX + radial.fY, center.fY - radial.fX };
canvas->drawLine(center, arcStart, tangentPaint);
canvas->drawPath(path, arcPaint);
canvas->drawString("(x0, y0)", pts[0].fX - 5, pts[0].fY, textPaint);
canvas->drawString("(x1, y1)", pts[1].fX + 5, pts[1].fY, textPaint);
canvas->drawString("(x2, y2)", pts[2].fX, pts[2].fY + 15, textPaint);
canvas->drawString("radius", center.fX + 15, center.fY + 25, textPaint);
canvas->drawString("radius", center.fX - 3, center.fY - 16, textPaint);
}
##
#Example
#Height 128
void draw(SkCanvas* canvas) {
SkPaint tangentPaint;
tangentPaint.setAntiAlias(true);
SkPaint textPaint(tangentPaint);
tangentPaint.setStyle(SkPaint::kStroke_Style);
tangentPaint.setColor(SK_ColorGRAY);
SkPaint arcPaint(tangentPaint);
arcPaint.setStrokeWidth(5);
arcPaint.setColor(SK_ColorBLUE);
SkPath path;
SkPoint pts[] = { {156, 20}, {200, 20}, {170, 50} };
SkScalar radius = 50;
path.moveTo(pts[0]);
path.arcTo(pts[1], pts[2], radius);
canvas->drawLine(pts[0], pts[1], tangentPaint);
canvas->drawLine(pts[1], pts[2], tangentPaint);
SkPoint lastPt;
(void) path.getLastPt(&lastPt);
SkVector radial = pts[2] - pts[1];
radial.setLength(radius);
SkPoint center = { lastPt.fX - radial.fY, lastPt.fY + radial.fX };
canvas->drawLine(lastPt, center, tangentPaint);
radial = pts[1] - pts[0];
radial.setLength(radius);
SkPoint arcStart = { center.fX + radial.fY, center.fY - radial.fX };
canvas->drawLine(center, arcStart, tangentPaint);
canvas->drawPath(path, arcPaint);
canvas->drawString("(x0, y0)", pts[0].fX, pts[0].fY - 7, textPaint);
canvas->drawString("(x1, y1)", pts[1].fX + 5, pts[1].fY, textPaint);
canvas->drawString("(x2, y2)", pts[2].fX, pts[2].fY + 15, textPaint);
canvas->drawString("radius", center.fX + 15, center.fY + 25, textPaint);
canvas->drawString("radius", center.fX - 5, center.fY - 20, textPaint);
}
##
#Example
#Description
arcTo is represented by Line and circular Conic in Path.
##
void draw(SkCanvas* canvas) {
SkPath path;
path.moveTo({156, 20});
path.arcTo(200, 20, 170, 50, 50);
SkPath::Iter iter(path, false);
SkPoint p[4];
SkPath::Verb verb;
while (SkPath::kDone_Verb != (verb = iter.next(p))) {
switch (verb) {
case SkPath::kMove_Verb:
SkDebugf("move to (%g,%g)\n", p[0].fX, p[0].fY);
break;
case SkPath::kLine_Verb:
SkDebugf("line (%g,%g),(%g,%g)\n", p[0].fX, p[0].fY, p[1].fX, p[1].fY);
break;
case SkPath::kConic_Verb:
SkDebugf("conic (%g,%g),(%g,%g),(%g,%g) weight %g\n",
p[0].fX, p[0].fY, p[1].fX, p[1].fY, p[2].fX, p[2].fY, iter.conicWeight());
break;
default:
SkDebugf("unexpected verb\n");
}
}
}
#StdOut
move to (156,20)
line (156,20),(79.2893,20)
conic (79.2893,20),(200,20),(114.645,105.355) weight 0.382683
##
##
#SeeAlso conicTo
##
# ------------------------------------------------------------------------------
#Method SkPath& arcTo(const SkPoint p1, const SkPoint p2, SkScalar radius)
#In Build
#In Arc
#Populate
#Example
#Description
Because tangent lines are parallel, arcTo appends line from last Path Point to
p1, but does not append a circular Conic.
##
void draw(SkCanvas* canvas) {
SkPath path;
path.moveTo({156, 20});
path.arcTo({200, 20}, {170, 20}, 50);
SkPath::Iter iter(path, false);
SkPoint p[4];
SkPath::Verb verb;
while (SkPath::kDone_Verb != (verb = iter.next(p))) {
switch (verb) {
case SkPath::kMove_Verb:
SkDebugf("move to (%g,%g)\n", p[0].fX, p[0].fY);
break;
case SkPath::kLine_Verb:
SkDebugf("line (%g,%g),(%g,%g)\n", p[0].fX, p[0].fY, p[1].fX, p[1].fY);
break;
case SkPath::kConic_Verb:
SkDebugf("conic (%g,%g),(%g,%g),(%g,%g) weight %g\n",
p[0].fX, p[0].fY, p[1].fX, p[1].fY, p[2].fX, p[2].fY, iter.conicWeight());
break;
default:
SkDebugf("unexpected verb\n");
}
}
}
#StdOut
move to (156,20)
line (156,20),(200,20)
##
##
#SeeAlso conicTo
##
# ------------------------------------------------------------------------------
#Enum ArcSize
#Line # used by arcTo variation ##
#Code
enum ArcSize {
kSmall_ArcSize,
kLarge_ArcSize,
};
##
Four axis-aligned Ovals with the same height and width intersect a pair of Points.
ArcSize and Direction select one of the four Ovals, by choosing the larger or smaller
arc between the Points; and by choosing the arc Direction, clockwise
or counterclockwise.
#Const kSmall_ArcSize 0
#Line # smaller of Arc pair ##
##
#Const kLarge_ArcSize 1
#Line # larger of Arc pair ##
##
#Example
#Height 160
#Description
Arc begins at top of Oval pair and ends at bottom. Arc can take four routes to get there.
Two routes are large, and two routes are counterclockwise. The one route both large
and counterclockwise is blue.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
for (auto sweep: { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
for (auto arcSize : { SkPath::kSmall_ArcSize, SkPath::kLarge_ArcSize } ) {
SkPath path;
path.moveTo({120, 50});
path.arcTo(70, 40, 30, arcSize, sweep, 156, 100);
if (SkPath::kCCW_Direction == sweep && SkPath::kLarge_ArcSize == arcSize) {
paint.setColor(SK_ColorBLUE);
paint.setStrokeWidth(3);
}
canvas->drawPath(path, paint);
}
}
}
##
#SeeAlso arcTo Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& arcTo(SkScalar rx, SkScalar ry, SkScalar xAxisRotate, ArcSize largeArc,
Direction sweep, SkScalar x, SkScalar y)
#In Build
#In Arc
#Populate
#Example
#Height 160
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
for (auto sweep: { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
for (auto arcSize : { SkPath::kSmall_ArcSize, SkPath::kLarge_ArcSize } ) {
SkPath path;
path.moveTo({120, 50});
path.arcTo(70, 40, 30, arcSize, sweep, 120.1, 50);
if (SkPath::kCCW_Direction == sweep && SkPath::kLarge_ArcSize == arcSize) {
paint.setColor(SK_ColorBLUE);
paint.setStrokeWidth(3);
}
canvas->drawPath(path, paint);
}
}
}
##
#SeeAlso rArcTo ArcSize Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& arcTo(const SkPoint r, SkScalar xAxisRotate, ArcSize largeArc, Direction sweep,
const SkPoint xy)
#In Build
#In Arc
#Populate
#Example
#Height 108
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPath path;
const SkPoint starts[] = {{20, 20}, {120, 20}, {70, 60}};
for (auto start : starts) {
path.moveTo(start.fX, start.fY);
path.rArcTo(20, 20, 0, SkPath::kSmall_ArcSize, SkPath::kCCW_Direction, 60, 0);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso rArcTo ArcSize Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& rArcTo(SkScalar rx, SkScalar ry, SkScalar xAxisRotate, ArcSize largeArc,
Direction sweep, SkScalar dx, SkScalar dy)
#In Build
#In Arc
#Line # appends Arc relative to Last_Point ##
Appends Arc to Path, relative to last Path Point. Arc is implemented by one or
more Conic, weighted to describe part of Oval with radii (rx, ry) rotated by
xAxisRotate degrees. Arc curves from last Path Point to relative end Point
(dx, dy), choosing one of four possible routes: clockwise or
counterclockwise, and smaller or larger. If Path is empty, the start Arc Point
is (0, 0).
Arc sweep is always less than 360 degrees. arcTo appends Line to end Point
if either radii are zero, or if last Path Point equals end Point.
arcTo scales radii (rx, ry) to fit last Path Point and end Point if both are
greater than zero but too small to describe an arc.
arcTo appends up to four Conic curves.
arcTo implements the functionality of SVG_Arc, although SVG "sweep-flag" value is
opposite the integer value of sweep; SVG "sweep-flag" uses 1 for clockwise, while
kCW_Direction cast to int is zero.
#Param rx radius before x-axis rotation ##
#Param ry radius before x-axis rotation ##
#Param xAxisRotate x-axis rotation in degrees; positive values are clockwise ##
#Param largeArc chooses smaller or larger Arc ##
#Param sweep chooses clockwise or counterclockwise Arc ##
#Param dx x-axis offset end of Arc from last Path Point ##
#Param dy y-axis offset end of Arc from last Path Point ##
#Return reference to Path ##
#Example
#Height 108
void draw(SkCanvas* canvas) {
SkPaint paint;
SkPath path;
const SkPoint starts[] = {{20, 20}, {120, 20}, {70, 60}};
for (auto start : starts) {
path.moveTo(start.fX, start.fY);
path.rArcTo(20, 20, 0, SkPath::kSmall_ArcSize, SkPath::kCCW_Direction, 60, 0);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso arcTo ArcSize Direction
##
#Subtopic Arc ##
# ------------------------------------------------------------------------------
#Method SkPath& close()
#In Build
#Line # makes last Contour a loop ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStrokeWidth(15);
paint.setStrokeCap(SkPaint::kRound_Cap);
SkPath path;
const SkPoint points[] = {{20, 20}, {70, 20}, {40, 90}};
path.addPoly(points, SK_ARRAY_COUNT(points), false);
for (int loop = 0; loop < 2; ++loop) {
for (auto style : {SkPaint::kStroke_Style, SkPaint::kFill_Style,
SkPaint::kStrokeAndFill_Style} ) {
paint.setStyle(style);
canvas->drawPath(path, paint);
canvas->translate(85, 0);
}
path.close();
canvas->translate(-255, 128);
}
}
##
#SeeAlso
##
# ------------------------------------------------------------------------------
#Method static bool IsInverseFillType(FillType fill)
#In Property
#Line # returns if Fill_Type represents outside geometry ##
Returns true if fill is inverted and Path with fill represents area outside
of its geometric bounds.
#Table
#Legend
# FillType # is inverse ##
##
# kWinding_FillType # false ##
# kEvenOdd_FillType # false ##
# kInverseWinding_FillType # true ##
# kInverseEvenOdd_FillType # true ##
##
#Param fill one of: kWinding_FillType, kEvenOdd_FillType,
kInverseWinding_FillType, kInverseEvenOdd_FillType
##
#Return true if Path fills outside its bounds ##
#Example
#Function
###$
#define nameValue(fill) { SkPath::fill, #fill }
$$$#
##
void draw(SkCanvas* canvas) {
struct {
SkPath::FillType fill;
const char* name;
} fills[] = {
nameValue(kWinding_FillType),
nameValue(kEvenOdd_FillType),
nameValue(kInverseWinding_FillType),
nameValue(kInverseEvenOdd_FillType),
};
for (auto fill: fills ) {
SkDebugf("IsInverseFillType(%s) == %s\n", fill.name, SkPath::IsInverseFillType(fill.fill) ?
"true" : "false");
}
}
#StdOut
IsInverseFillType(kWinding_FillType) == false
IsInverseFillType(kEvenOdd_FillType) == false
IsInverseFillType(kInverseWinding_FillType) == true
IsInverseFillType(kInverseEvenOdd_FillType) == true
##
##
#SeeAlso FillType getFillType setFillType ConvertToNonInverseFillType
##
# ------------------------------------------------------------------------------
#Method static FillType ConvertToNonInverseFillType(FillType fill)
#In Utility
#Line # returns Fill_Type representing inside geometry ##
Returns equivalent Fill_Type representing Path fill inside its bounds.
#Table
#Legend
# FillType # inside FillType ##
##
# kWinding_FillType # kWinding_FillType ##
# kEvenOdd_FillType # kEvenOdd_FillType ##
# kInverseWinding_FillType # kWinding_FillType ##
# kInverseEvenOdd_FillType # kEvenOdd_FillType ##
##
#Param fill one of: kWinding_FillType, kEvenOdd_FillType,
kInverseWinding_FillType, kInverseEvenOdd_FillType
##
#Return fill, or kWinding_FillType or kEvenOdd_FillType if fill is inverted ##
#Example
#Function
###$
#define nameValue(fill) { SkPath::fill, #fill }
$$$#
##
void draw(SkCanvas* canvas) {
struct {
SkPath::FillType fill;
const char* name;
} fills[] = {
nameValue(kWinding_FillType),
nameValue(kEvenOdd_FillType),
nameValue(kInverseWinding_FillType),
nameValue(kInverseEvenOdd_FillType),
};
for (unsigned i = 0; i < SK_ARRAY_COUNT(fills); ++i) {
if (fills[i].fill != (SkPath::FillType) i) {
SkDebugf("fills array order does not match FillType enum order");
break;
}
SkDebugf("ConvertToNonInverseFillType(%s) == %s\n", fills[i].name,
fills[(int) SkPath::ConvertToNonInverseFillType(fills[i].fill)].name);
}
}
#StdOut
ConvertToNonInverseFillType(kWinding_FillType) == kWinding_FillType
ConvertToNonInverseFillType(kEvenOdd_FillType) == kEvenOdd_FillType
ConvertToNonInverseFillType(kInverseWinding_FillType) == kWinding_FillType
ConvertToNonInverseFillType(kInverseEvenOdd_FillType) == kEvenOdd_FillType
##
##
#SeeAlso FillType getFillType setFillType IsInverseFillType
##
# ------------------------------------------------------------------------------
#Method static int ConvertConicToQuads(const SkPoint& p0, const SkPoint& p1, const SkPoint& p2,
SkScalar w, SkPoint pts[], int pow2)
#In Utility
#Line # approximates Conic with Quad array ##
Approximates Conic with Quad array. Conic is constructed from start Point p0,
control Point p1, end Point p2, and weight w.
Quad array is stored in pts; this storage is supplied by caller.
Maximum Quad count is 2 to the pow2.
Every third point in array shares last Point of previous Quad and first Point of
next Quad. Maximum pts storage size is given by:
#Formula # (1 + 2 * (1 << pow2)) * sizeof(SkPoint) ##.
Returns Quad count used the approximation, which may be smaller
than the number requested.
Conic_Weight determines the amount of influence Conic control point has on the curve.
w less than one represents an elliptical section. w greater than one represents
a hyperbolic section. w equal to one represents a parabolic section.
Two Quad curves are sufficient to approximate an elliptical Conic with a sweep
of up to 90 degrees; in this case, set pow2 to one.
#Param p0 Conic start Point ##
#Param p1 Conic control Point ##
#Param p2 Conic end Point ##
#Param w Conic weight ##
#Param pts storage for Quad array ##
#Param pow2 Quad count, as power of two, normally 0 to 5 (1 to 32 Quad curves) ##
#Return number of Quad curves written to pts ##
#Example
#Description
A pair of Quad curves are drawn in red on top of the elliptical Conic curve in black.
The middle curve is nearly circular. The top-right curve is parabolic, which can
be drawn exactly with a single Quad.
##
void draw(SkCanvas* canvas) {
SkPaint conicPaint;
conicPaint.setAntiAlias(true);
conicPaint.setStyle(SkPaint::kStroke_Style);
SkPaint quadPaint(conicPaint);
quadPaint.setColor(SK_ColorRED);
SkPoint conic[] = { {20, 170}, {80, 170}, {80, 230} };
for (auto weight : { .25f, .5f, .707f, .85f, 1.f } ) {
SkPoint quads[5];
SkPath::ConvertConicToQuads(conic[0], conic[1], conic[2], weight, quads, 1);
SkPath path;
path.moveTo(conic[0]);
path.conicTo(conic[1], conic[2], weight);
canvas->drawPath(path, conicPaint);
path.rewind();
path.moveTo(quads[0]);
path.quadTo(quads[1], quads[2]);
path.quadTo(quads[3], quads[4]);
canvas->drawPath(path, quadPaint);
canvas->translate(50, -50);
}
}
##
#SeeAlso Conic Quad
##
# ------------------------------------------------------------------------------
#Method bool isRect(SkRect* rect, bool* isClosed = nullptr, Direction* direction = nullptr) const
#In Property
#Line # returns if describes Rect ##
#Populate
#Example
#Description
After addRect, isRect returns true. Following moveTo permits isRect to return true, but
following lineTo does not. addPoly returns true even though rect is not closed, and one
side of rect is made up of consecutive line segments.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path) -> void {
SkRect rect;
SkPath::Direction direction;
bool isClosed;
path.isRect(&rect, &isClosed, &direction) ?
SkDebugf("%s is rect (%g, %g, %g, %g); is %s" "closed; direction %s\n", prefix,
rect.fLeft, rect.fTop, rect.fRight, rect.fBottom, isClosed ? "" : "not ",
SkPath::kCW_Direction == direction ? "CW" : "CCW") :
SkDebugf("%s is not rect\n", prefix);
};
SkPath path;
debugster("empty", path);
path.addRect({10, 20, 30, 40});
debugster("addRect", path);
path.moveTo(60, 70);
debugster("moveTo", path);
path.lineTo(60, 70);
debugster("lineTo", path);
path.reset();
const SkPoint pts[] = { {0, 0}, {0, 80}, {80, 80}, {80, 0}, {40, 0}, {20, 0} };
path.addPoly(pts, SK_ARRAY_COUNT(pts), false);
debugster("addPoly", path);
}
#StdOut
empty is not rect
addRect is rect (10, 20, 30, 40); is closed; direction CW
moveTo is rect (10, 20, 30, 40); is closed; direction CW
lineTo is not rect
addPoly is rect (0, 0, 80, 80); is not closed; direction CCW
##
##
#SeeAlso computeTightBounds conservativelyContainsRect getBounds isConvex isLastContourClosed isNestedFillRects
##
# ------------------------------------------------------------------------------
#Method bool isNestedFillRects(SkRect rect[2], Direction dirs[2] = nullptr) const
#In Property
#Line # returns if describes Rect pair, one inside the other ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(5);
SkPath path;
path.addRect({10, 20, 30, 40});
paint.getFillPath(path, &path);
SkRect rects[2];
SkPath::Direction directions[2];
if (path.isNestedFillRects(rects, directions)) {
for (int i = 0; i < 2; ++i) {
SkDebugf("%s (%g, %g, %g, %g); direction %s\n", i ? "inner" : "outer",
rects[i].fLeft, rects[i].fTop, rects[i].fRight, rects[i].fBottom,
SkPath::kCW_Direction == directions[i] ? "CW" : "CCW");
}
} else {
SkDebugf("is not nested rectangles\n");
}
}
#StdOut
outer (7.5, 17.5, 32.5, 42.5); direction CW
inner (12.5, 22.5, 27.5, 37.5); direction CCW
##
##
#SeeAlso computeTightBounds conservativelyContainsRect getBounds isConvex isLastContourClosed isRect
##
# ------------------------------------------------------------------------------
#Method SkPath& addRect(const SkRect& rect, Direction dir = kCW_Direction)
#In Build
#Line # adds one Contour containing Rect ##
#Populate
#Example
#Description
The left Rect dashes starting at the top-left corner, to the right.
The right Rect dashes starting at the top-left corner, towards the bottom.
##
#Height 128
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStrokeWidth(15);
paint.setStrokeCap(SkPaint::kSquare_Cap);
float intervals[] = { 5, 21.75f };
paint.setStyle(SkPaint::kStroke_Style);
paint.setPathEffect(SkDashPathEffect::Make(intervals, SK_ARRAY_COUNT(intervals), 0));
SkPath path;
path.addRect({20, 20, 100, 100}, SkPath::kCW_Direction);
canvas->drawPath(path, paint);
path.rewind();
path.addRect({140, 20, 220, 100}, SkPath::kCCW_Direction);
canvas->drawPath(path, paint);
}
##
#SeeAlso SkCanvas::drawRect Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& addRect(const SkRect& rect, Direction dir, unsigned start)
Adds Rect to Path, appending kMove_Verb, three kLine_Verb, and kClose_Verb.
If dir is kCW_Direction, Rect corners are added clockwise; if dir is
kCCW_Direction, Rect corners are added counterclockwise.
start determines the first corner added.
#Table
#Legend
# start # first corner ##
#Legend ##
# 0 # top-left ##
# 1 # top-right ##
# 2 # bottom-right ##
# 3 # bottom-left ##
#Table ##
#Param rect Rect to add as a closed contour ##
#Param dir Direction to wind added contour ##
#Param start initial corner of Rect to add ##
#Return reference to Path ##
#Example
#Height 128
#Description
The arrow is just after the initial corner and points towards the next
corner appended to Path.
##
void draw(SkCanvas* canvas) {
const SkPoint arrow[] = { {5, -5}, {15, -5}, {20, 0}, {15, 5}, {5, 5}, {10, 0} };
const SkRect rect = {10, 10, 54, 54};
SkPaint rectPaint;
rectPaint.setAntiAlias(true);
rectPaint.setStyle(SkPaint::kStroke_Style);
SkPaint arrowPaint(rectPaint);
SkPath arrowPath;
arrowPath.addPoly(arrow, SK_ARRAY_COUNT(arrow), true);
arrowPaint.setPathEffect(SkPath1DPathEffect::Make(arrowPath, 176, 0,
SkPath1DPathEffect::kRotate_Style));
for (auto direction : { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
for (unsigned start : { 0, 1, 2, 3 } ) {
SkPath path;
path.addRect(rect, direction, start);
canvas->drawPath(path, rectPaint);
canvas->drawPath(path, arrowPaint);
canvas->translate(64, 0);
}
canvas->translate(-256, 64);
}
}
##
#SeeAlso SkCanvas::drawRect Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& addRect(SkScalar left, SkScalar top, SkScalar right, SkScalar bottom,
Direction dir = kCW_Direction)
#Populate
#Example
#Description
The left Rect dashes start at the top-left corner, and continue to the right.
The right Rect dashes start at the top-left corner, and continue down.
##
#Height 128
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStrokeWidth(15);
paint.setStrokeCap(SkPaint::kSquare_Cap);
float intervals[] = { 5, 21.75f };
paint.setStyle(SkPaint::kStroke_Style);
paint.setPathEffect(SkDashPathEffect::Make(intervals, SK_ARRAY_COUNT(intervals), 0));
for (auto direction : { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
SkPath path;
path.addRect(20, 20, 100, 100, direction);
canvas->drawPath(path, paint);
canvas->translate(128, 0);
}
}
##
#SeeAlso SkCanvas::drawRect Direction
##
# ------------------------------------------------------------------------------
#Method SkPath& addOval(const SkRect& oval, Direction dir = kCW_Direction)
#In Build
#Line # adds one Contour containing Oval ##
#Populate
#Example
#Height 120
SkPaint paint;
SkPath oval;
oval.addOval({20, 20, 160, 80});
canvas->drawPath(oval, paint);
##
#SeeAlso SkCanvas::drawOval Direction Oval
##
# ------------------------------------------------------------------------------
#Method SkPath& addOval(const SkRect& oval, Direction dir, unsigned start)
Adds Oval to Path, appending kMove_Verb, four kConic_Verb, and kClose_Verb.
Oval is upright ellipse bounded by Rect oval with radii equal to half oval width
and half oval height. Oval begins at start and continues
clockwise if dir is kCW_Direction, counterclockwise if dir is kCCW_Direction.
#Table
#Legend
# start # Point ##
#Legend ##
# 0 # oval.centerX(), oval.fTop ##
# 1 # oval.fRight, oval.centerY() ##
# 2 # oval.centerX(), oval.fBottom ##
# 3 # oval.fLeft, oval.centerY() ##
#Table ##
#Param oval bounds of ellipse added ##
#Param dir Direction to wind ellipse ##
#Param start index of initial point of ellipse ##
#Return reference to Path ##
#Example
#Height 160
void draw(SkCanvas* canvas) {
const SkPoint arrow[] = { {0, -5}, {10, 0}, {0, 5} };
const SkRect rect = {10, 10, 54, 54};
SkPaint ovalPaint;
ovalPaint.setAntiAlias(true);
SkPaint textPaint(ovalPaint);
ovalPaint.setStyle(SkPaint::kStroke_Style);
SkPaint arrowPaint(ovalPaint);
SkPath arrowPath;
arrowPath.addPoly(arrow, SK_ARRAY_COUNT(arrow), true);
arrowPaint.setPathEffect(SkPath1DPathEffect::Make(arrowPath, 176, 0,
SkPath1DPathEffect::kRotate_Style));
for (auto direction : { SkPath::kCW_Direction, SkPath::kCCW_Direction } ) {
for (unsigned start : { 0, 1, 2, 3 } ) {
SkPath path;
path.addOval(rect, direction, start);
canvas->drawPath(path, ovalPaint);
canvas->drawPath(path, arrowPaint);
canvas->drawText(&"0123"[start], 1, rect.centerX(), rect.centerY() + 5, textPaint);
canvas->translate(64, 0);
}
canvas->translate(-256, 72);
canvas->drawString(SkPath::kCW_Direction == direction ? "clockwise" : "counterclockwise",
128, 0, textPaint);
}
}
##
#SeeAlso SkCanvas::drawOval Direction Oval
##
# ------------------------------------------------------------------------------
#Method SkPath& addCircle(SkScalar x, SkScalar y, SkScalar radius,
Direction dir = kCW_Direction)
#In Build
#Line # adds one Contour containing Circle ##
Adds Circle centered at (x, y) of size radius to Path, appending kMove_Verb,
four kConic_Verb, and kClose_Verb. Circle begins at: #Formula # (x + radius, y) ##, continuing
clockwise if dir is kCW_Direction, and counterclockwise if dir is kCCW_Direction.
Has no effect if radius is zero or negative.
#Param x center of Circle ##
#Param y center of Circle ##
#Param radius distance from center to edge ##
#Param dir Direction to wind Circle ##
#Return reference to Path ##
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(10);
for (int size = 10; size < 300; size += 20) {
SkPath path;
path.addCircle(128, 128, size, SkPath::kCW_Direction);
canvas->drawPath(path, paint);
}
}
##
#SeeAlso SkCanvas::drawCircle Direction Circle
##
# ------------------------------------------------------------------------------
#Method SkPath& addArc(const SkRect& oval, SkScalar startAngle, SkScalar sweepAngle)
#In Build
#Line # adds one Contour containing Arc ##
#Populate
#Example
#Description
The middle row of the left and right columns draw differently from the entries
above and below because sweepAngle is outside of the range of +/-360,
and startAngle modulo 90 is not zero.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
for (auto start : { 0, 90, 135, 180, 270 } ) {
for (auto sweep : { -450.f, -180.f, -90.f, 90.f, 180.f, 360.1f } ) {
SkPath path;
path.addArc({10, 10, 35, 45}, start, sweep);
canvas->drawPath(path, paint);
canvas->translate(252 / 6, 0);
}
canvas->translate(-252, 255 / 5);
}
}
##
#SeeAlso Arc arcTo SkCanvas::drawArc
##
# ------------------------------------------------------------------------------
#Method SkPath& addRoundRect(const SkRect& rect, SkScalar rx, SkScalar ry,
Direction dir = kCW_Direction)
#In Build
#Line # adds one Contour containing Round_Rect with common corner radii ##
#Populate
#Example
#Description
If either radius is zero, path contains Rect and is drawn red.
If sides are only radii, path contains Oval and is drawn blue.
All remaining path draws are convex, and are drawn in gray; no
paths constructed from addRoundRect are concave, so none are
drawn in green.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
for (auto xradius : { 0, 7, 13, 20 } ) {
for (auto yradius : { 0, 9, 18, 40 } ) {
SkPath path;
path.addRoundRect({10, 10, 36, 46}, xradius, yradius);
paint.setColor(path.isRect(nullptr) ? SK_ColorRED : path.isOval(nullptr) ?
SK_ColorBLUE : path.isConvex() ? SK_ColorGRAY : SK_ColorGREEN);
canvas->drawPath(path, paint);
canvas->translate(64, 0);
}
canvas->translate(-256, 64);
}
}
##
#SeeAlso addRRect SkCanvas::drawRoundRect
##
# ------------------------------------------------------------------------------
#Method SkPath& addRoundRect(const SkRect& rect, const SkScalar radii[],
Direction dir = kCW_Direction)
Appends Round_Rect to Path, creating a new closed Contour. Round_Rect has bounds
equal to rect; each corner is 90 degrees of an ellipse with radii from the
array.
#Table
#Legend
# radii index # location ##
#Legend ##
# 0 # x-axis radius of top-left corner ##
# 1 # y-axis radius of top-left corner ##
# 2 # x-axis radius of top-right corner ##
# 3 # y-axis radius of top-right corner ##
# 4 # x-axis radius of bottom-right corner ##
# 5 # y-axis radius of bottom-right corner ##
# 6 # x-axis radius of bottom-left corner ##
# 7 # y-axis radius of bottom-left corner ##
#Table ##
If dir is kCW_Direction, Round_Rect starts at top-left of the lower-left corner
and winds clockwise. If dir is kCCW_Direction, Round_Rect starts at the
bottom-left of the upper-left corner and winds counterclockwise.
If both radii on any side of rect exceed its length, all radii are scaled
uniformly until the corners fit. If either radius of a corner is less than or
equal to zero, both are treated as zero.
After appending, Path may be empty, or may contain: Rect, Oval, or Round_Rect.
#Param rect bounds of Round_Rect ##
#Param radii array of 8 SkScalar values, a radius pair for each corner ##
#Param dir Direction to wind Round_Rect ##
#Return reference to Path ##
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
SkScalar radii[] = { 80, 100, 0, 0, 40, 60, 0, 0 };
SkPath path;
SkMatrix rotate90;
rotate90.setRotate(90, 128, 128);
for (int i = 0; i < 4; ++i) {
path.addRoundRect({10, 10, 110, 110}, radii);
path.transform(rotate90);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso addRRect SkCanvas::drawRoundRect
##
# ------------------------------------------------------------------------------
#Method SkPath& addRRect(const SkRRect& rrect, Direction dir = kCW_Direction)
#In Build
#Line # adds one Contour containing Round_Rect ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
SkRRect rrect;
SkVector radii[] = {{50, 50}, {0, 0}, {0, 0}, {50, 50}};
rrect.setRectRadii({10, 10, 110, 110}, radii);
SkPath path;
SkMatrix rotate90;
rotate90.setRotate(90, 128, 128);
for (int i = 0; i < 4; ++i) {
path.addRRect(rrect);
path.transform(rotate90);
}
canvas->drawPath(path, paint);
}
##
#SeeAlso addRoundRect SkCanvas::drawRRect
##
# ------------------------------------------------------------------------------
#Method SkPath& addRRect(const SkRRect& rrect, Direction dir, unsigned start)
Adds rrect to Path, creating a new closed Contour. If dir is kCW_Direction, rrect
winds clockwise; if dir is kCCW_Direction, rrect winds counterclockwise.
start determines the first point of rrect to add.
#Table
#Legend
# start # location ##
#Legend ##
# 0 # right of top-left corner ##
# 1 # left of top-right corner ##
# 2 # bottom of top-right corner ##
# 3 # top of bottom-right corner ##
# 4 # left of bottom-right corner ##
# 5 # right of bottom-left corner ##
# 6 # top of bottom-left corner ##
# 7 # bottom of top-left corner ##
#Table ##
After appending, Path may be empty, or may contain: Rect, Oval, or Round_Rect.
#Param rrect bounds and radii of rounded rectangle ##
#Param dir Direction to wind Round_Rect ##
#Param start index of initial point of Round_Rect ##
#Return reference to Path ##
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
SkRRect rrect;
rrect.setRectXY({40, 40, 215, 215}, 50, 50);
SkPath path;
path.addRRect(rrect);
canvas->drawPath(path, paint);
for (int start = 0; start < 8; ++start) {
SkPath textPath;
textPath.addRRect(rrect, SkPath::kCW_Direction, start);
SkPathMeasure pathMeasure(textPath, false);
SkPoint position;
SkVector tangent;
if (!pathMeasure.getPosTan(0, &position, &tangent)) {
continue;
}
SkRSXform rsxForm = SkRSXform::Make(tangent.fX, tangent.fY,
position.fX + tangent.fY * 5, position.fY - tangent.fX * 5);
canvas->drawTextRSXform(&"01234567"[start], 1, &rsxForm, nullptr, paint);
}
}
##
#SeeAlso addRoundRect SkCanvas::drawRRect
##
# ------------------------------------------------------------------------------
#Method SkPath& addPoly(const SkPoint pts[], int count, bool close)
#In Build
#Line # adds one Contour containing connected lines ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStrokeWidth(15);
paint.setStrokeCap(SkPaint::kRound_Cap);
const SkPoint points[] = {{20, 20}, {70, 20}, {40, 90}};
for (bool close : { false, true } ) {
SkPath path;
path.addPoly(points, SK_ARRAY_COUNT(points), close);
for (auto style : {SkPaint::kStroke_Style, SkPaint::kFill_Style,
SkPaint::kStrokeAndFill_Style} ) {
paint.setStyle(style);
canvas->drawPath(path, paint);
canvas->translate(85, 0);
}
canvas->translate(-255, 128);
}
}
##
#SeeAlso SkCanvas::drawPoints
##
#Method SkPath& addPoly(const std::initializer_list& list, bool close)
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setStrokeWidth(15);
paint.setStrokeCap(SkPaint::kRound_Cap);
for (bool close : { false, true } ) {
SkPath path;
path.addPoly({{20, 20}, {70, 20}, {40, 90}}, close);
for (auto style : {SkPaint::kStroke_Style, SkPaint::kFill_Style,
SkPaint::kStrokeAndFill_Style} ) {
paint.setStyle(style);
canvas->drawPath(path, paint);
canvas->translate(85, 0);
}
canvas->translate(-255, 128);
}
}
##
#SeeAlso SkCanvas::drawPoints
##
# ------------------------------------------------------------------------------
#Enum AddPathMode
#Line # sets addPath options ##
#Code
enum AddPathMode {
kAppend_AddPathMode,
kExtend_AddPathMode,
};
##
AddPathMode chooses how addPath appends. Adding one Path to another can extend
the last Contour or start a new Contour.
#Const kAppend_AddPathMode
#Line # appended to destination unaltered ##
Path Verbs, Points, and Conic_Weights are appended to destination unaltered.
Since Path Verb_Array begins with kMove_Verb if src is not empty, this
starts a new Contour.
##
#Const kExtend_AddPathMode
#Line # add line if prior Contour is not closed ##
If destination is closed or empty, start a new Contour. If destination
is not empty, add Line from Last_Point to added Path first Point. Skip added
Path initial kMove_Verb, then append remaining Verbs, Points, and Conic_Weights.
##
#Example
#Description
test is built from path, open on the top row, and closed on the bottom row.
The left column uses kAppend_AddPathMode; the right uses kExtend_AddPathMode.
The top right composition is made up of one contour; the other three have two.
##
#Height 180
SkPath path, path2;
path.moveTo(20, 20);
path.lineTo(20, 40);
path.lineTo(40, 20);
path2.moveTo(60, 60);
path2.lineTo(80, 60);
path2.lineTo(80, 40);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 2; i++) {
for (auto addPathMode : { SkPath::kAppend_AddPathMode, SkPath::kExtend_AddPathMode } ) {
SkPath test(path);
test.addPath(path2, addPathMode);
canvas->drawPath(test, paint);
canvas->translate(100, 0);
}
canvas->translate(-200, 100);
path.close();
}
##
#SeeAlso addPath reverseAddPath
##
# ------------------------------------------------------------------------------
#Method SkPath& addPath(const SkPath& src, SkScalar dx, SkScalar dy,
AddPathMode mode = kAppend_AddPathMode)
#In Build
#Line # adds contents of Path ##
#Populate
#Example
#Height 180
SkPaint paint;
paint.setTextSize(128);
paint.setFakeBoldText(true);
SkPath dest, text;
paint.getTextPath("O", 1, 50, 120, &text);
for (int i = 0; i < 3; i++) {
dest.addPath(text, i * 20, i * 20);
}
Simplify(dest, &dest);
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(3);
canvas->drawPath(dest, paint);
##
#SeeAlso AddPathMode offset reverseAddPath
##
# ------------------------------------------------------------------------------
#Method SkPath& addPath(const SkPath& src, AddPathMode mode = kAppend_AddPathMode)
#Populate
#Example
#Height 80
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
SkPath dest, path;
path.addOval({-80, 20, 0, 60}, SkPath::kCW_Direction, 1);
for (int i = 0; i < 2; i++) {
dest.addPath(path, SkPath::kExtend_AddPathMode);
dest.offset(100, 0);
}
canvas->drawPath(dest, paint);
##
#SeeAlso AddPathMode reverseAddPath
##
# ------------------------------------------------------------------------------
#Method SkPath& addPath(const SkPath& src, const SkMatrix& matrix, AddPathMode mode = kAppend_AddPathMode)
#Populate
#Example
#Height 160
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
SkPath dest, path;
path.addOval({20, 20, 200, 120}, SkPath::kCW_Direction, 1);
for (int i = 0; i < 6; i++) {
SkMatrix matrix;
matrix.reset();
matrix.setPerspX(i / 400.f);
dest.addPath(path, matrix);
}
canvas->drawPath(dest, paint);
##
#SeeAlso AddPathMode transform offset reverseAddPath
##
# ------------------------------------------------------------------------------
#Method SkPath& reverseAddPath(const SkPath& src)
#In Build
#Line # adds contents of Path back to front ##
#Populate
#Example
#Height 200
SkPath path;
path.moveTo(20, 20);
path.lineTo(20, 40);
path.lineTo(40, 20);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 2; i++) {
SkPath path2;
path2.moveTo(60, 60);
path2.lineTo(80, 60);
path2.lineTo(80, 40);
for (int j = 0; j < 2; j++) {
SkPath test(path);
test.reverseAddPath(path2);
canvas->drawPath(test, paint);
canvas->translate(100, 0);
path2.close();
}
canvas->translate(-200, 100);
path.close();
}
##
#SeeAlso AddPathMode transform offset addPath
##
# ------------------------------------------------------------------------------
#Method void offset(SkScalar dx, SkScalar dy, SkPath* dst) const
#In Transform
#Line # translates Point_Array ##
#Populate
#Example
#Height 60
SkPath pattern;
pattern.moveTo(20, 20);
pattern.lineTo(20, 40);
pattern.lineTo(40, 20);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 10; i++) {
SkPath path;
pattern.offset(20 * i, 0, &path);
canvas->drawPath(path, paint);
}
##
#SeeAlso addPath transform
##
# ------------------------------------------------------------------------------
#Subtopic Transform
#Line # modify all points ##
##
#Method void offset(SkScalar dx, SkScalar dy)
#In Transform
#Populate
#Example
#Height 60
SkPath path;
path.moveTo(20, 20);
path.lineTo(20, 40);
path.lineTo(40, 20);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 10; i++) {
canvas->drawPath(path, paint);
path.offset(20, 0);
}
##
#SeeAlso addPath transform SkCanvas::translate()
##
# ------------------------------------------------------------------------------
#Method void transform(const SkMatrix& matrix, SkPath* dst) const
#In Transform
#Line # applies Matrix to Point_Array and Weights ##
#Populate
#Example
#Height 200
SkPath pattern;
pattern.moveTo(100, 100);
pattern.lineTo(100, 20);
pattern.lineTo(20, 100);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 10; i++) {
SkPath path;
SkMatrix matrix;
matrix.setRotate(36 * i, 100, 100);
pattern.transform(matrix, &path);
canvas->drawPath(path, paint);
}
##
#SeeAlso addPath offset SkCanvas::concat() SkMatrix
##
# ------------------------------------------------------------------------------
#Method void transform(const SkMatrix& matrix)
#Populate
#Example
#Height 200
SkPath path;
path.moveTo(100, 100);
path.quadTo(100, 20, 20, 100);
SkPaint paint;
paint.setStyle(SkPaint::kStroke_Style);
for (int i = 0; i < 10; i++) {
SkMatrix matrix;
matrix.setRotate(36, 100, 100);
path.transform(matrix);
canvas->drawPath(path, paint);
}
##
#SeeAlso addPath offset SkCanvas::concat() SkMatrix
##
# ------------------------------------------------------------------------------
#Subtopic Last_Point
#Line # final Point in Contour ##
Path is defined cumulatively, often by adding a segment to the end of last
Contour. Last_Point of Contour is shared as first Point of added Line or Curve.
Last_Point can be read and written directly with getLastPt and setLastPt.
#Method bool getLastPt(SkPoint* lastPt) const
#In Property
#In Last_Point
#Line # returns Last_Point ##
#Populate
#Example
SkPath path;
path.moveTo(100, 100);
path.quadTo(100, 20, 20, 100);
SkMatrix matrix;
matrix.setRotate(36, 100, 100);
path.transform(matrix);
SkPoint last;
path.getLastPt(&last);
SkDebugf("last point: %g, %g\n", last.fX, last.fY);
#StdOut
last point: 35.2786, 52.9772
##
##
#SeeAlso setLastPt
##
#Method void setLastPt(SkScalar x, SkScalar y)
#In Utility
#In Last_Point
#Line # replaces Last_Point ##
#Populate
#Example
#Height 128
SkPaint paint;
paint.setTextSize(128);
SkPath path;
paint.getTextPath("@", 1, 60, 100, &path);
path.setLastPt(20, 120);
canvas->drawPath(path, paint);
##
#SeeAlso getLastPt
##
#Method void setLastPt(const SkPoint& p)
#Populate
#Example
#Height 128
SkPaint paint;
paint.setTextSize(128);
SkPath path, path2;
paint.getTextPath("A", 1, 60, 100, &path);
paint.getTextPath("Z", 1, 60, 100, &path2);
SkPoint pt, pt2;
path.getLastPt(&pt);
path2.getLastPt(&pt2);
path.setLastPt(pt2);
path2.setLastPt(pt);
canvas->drawPath(path, paint);
canvas->drawPath(path2, paint);
##
#SeeAlso getLastPt
##
#Subtopic Last_Point ##
# ------------------------------------------------------------------------------
#Enum SegmentMask
#Line # returns Verb types in Path ##
#Code
enum SegmentMask {
kLine_SegmentMask = 1 << 0,
kQuad_SegmentMask = 1 << 1,
kConic_SegmentMask = 1 << 2,
kCubic_SegmentMask = 1 << 3,
};
##
SegmentMask constants correspond to each drawing Verb type in Path; for
instance, if Path only contains Lines, only the kLine_SegmentMask bit is set.
#Bug 6785
#Const kLine_SegmentMask 1
#Line # contains one or more Lines ##
Set if Verb_Array contains kLine_Verb.
##
#Const kQuad_SegmentMask 2
#Line # contains one or more Quads ##
Set if Verb_Array contains kQuad_Verb. Note that conicTo may add a Quad.
##
#Const kConic_SegmentMask 4
#Line # contains one or more Conics ##
Set if Verb_Array contains kConic_Verb.
##
#Const kCubic_SegmentMask 8
#Line # contains one or more Cubics ##
Set if Verb_Array contains kCubic_Verb.
##
#Example
#Description
When conicTo has a weight of one, Quad is added to Path.
##
SkPath path;
path.conicTo(10, 10, 20, 30, 1);
SkDebugf("Path kConic_SegmentMask is %s\n", path.getSegmentMasks() &
SkPath::kConic_SegmentMask ? "set" : "clear");
SkDebugf("Path kQuad_SegmentMask is %s\n", path.getSegmentMasks() &
SkPath::kQuad_SegmentMask ? "set" : "clear");
#StdOut
Path kConic_SegmentMask is clear
Path kQuad_SegmentMask is set
##
##
#SeeAlso getSegmentMasks Verb
##
# ------------------------------------------------------------------------------
#Method uint32_t getSegmentMasks() const
#In Utility
#In Property
#Line # returns types in Verb_Array ##
#Populate
#Example
SkPath path;
path.quadTo(20, 30, 40, 50);
path.close();
const char* masks[] = { "line", "quad", "conic", "cubic" };
int index = 0;
for (auto mask : { SkPath::kLine_SegmentMask, SkPath::kQuad_SegmentMask,
SkPath::kConic_SegmentMask, SkPath::kCubic_SegmentMask } ) {
if (mask & path.getSegmentMasks()) {
SkDebugf("mask %s set\n", masks[index]);
}
++index;
}
#StdOut
mask quad set
##
##
#SeeAlso getSegmentMasks Verb
##
# ------------------------------------------------------------------------------
#Method bool contains(SkScalar x, SkScalar y) const
#In Property
#Line # returns if Point is in fill area ##
Returns true if the point (x, y) is contained by Path, taking into
account FillType.
#Table
#Legend
# FillType # contains() returns true if Point is enclosed by ##
##
# kWinding_FillType # a non-zero sum of Contour Directions. ##
# kEvenOdd_FillType # an odd number of Contours. ##
# kInverseWinding_FillType # a zero sum of Contour Directions. ##
# kInverseEvenOdd_FillType # and even number of Contours. ##
##
#Param x x-axis value of containment test ##
#Param y y-axis value of containment test ##
#Return true if Point is in Path ##
#Example
SkPath path;
SkPaint paint;
paint.setTextSize(256);
paint.getTextPath("&", 1, 30, 220, &path);
for (int y = 2; y < 256; y += 9) {
for (int x = 2; x < 256; x += 9) {
int coverage = 0;
for (int iy = -4; iy <= 4; iy += 2) {
for (int ix = -4; ix <= 4; ix += 2) {
coverage += path.contains(x + ix, y + iy);
}
}
paint.setColor(SkColorSetARGB(0x5f, 0xff * coverage / 25, 0, 0xff * (25 - coverage) / 25));
canvas->drawCircle(x, y, 8, paint);
}
}
##
#SeeAlso conservativelyContainsRect Fill_Type Op
##
# ------------------------------------------------------------------------------
#Method void dump(SkWStream* stream, bool forceClose, bool dumpAsHex) const
#In Utility
#Line # sends text representation to stream ##
#Populate
#Example
SkPath path;
path.quadTo(20, 30, 40, 50);
for (bool forceClose : { false, true } ) {
for (bool dumpAsHex : { false, true } ) {
path.dump(nullptr, forceClose, dumpAsHex);
SkDebugf("\n");
}
}
#StdOut
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(0, 0);
path.quadTo(20, 30, 40, 50);
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0
path.quadTo(SkBits2Float(0x41a00000), SkBits2Float(0x41f00000), SkBits2Float(0x42200000), SkBits2Float(0x42480000)); // 20, 30, 40, 50
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(0, 0);
path.quadTo(20, 30, 40, 50);
path.lineTo(0, 0);
path.close();
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0
path.quadTo(SkBits2Float(0x41a00000), SkBits2Float(0x41f00000), SkBits2Float(0x42200000), SkBits2Float(0x42480000)); // 20, 30, 40, 50
path.lineTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0
path.close();
##
##
#SeeAlso dumpHex SkRect::dump() SkRRect::dump() SkPathMeasure::dump()
##
# ------------------------------------------------------------------------------
#Method void dump() const
#Populate
#Example
SkPath path, copy;
path.lineTo(6.f / 7, 2.f / 3);
path.dump();
copy.setFillType(SkPath::kWinding_FillType);
copy.moveTo(0, 0);
copy.lineTo(0.857143f, 0.666667f);
SkDebugf("path is " "%s" "equal to copy\n", path == copy ? "" : "not ");
#StdOut
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(0, 0);
path.lineTo(0.857143f, 0.666667f);
path is not equal to copy
##
##
#SeeAlso dumpHex SkRect::dump() SkRRect::dump() SkPathMeasure::dump() writeToMemory
##
# ------------------------------------------------------------------------------
#Method void dumpHex() const
#In Utility
#Line # sends text representation using hexadecimal to standard output ##
Writes text representation of Path to standard output. The representation may be
directly compiled as C++ code. Floating point values are written
in hexadecimal to preserve their exact bit pattern. The output reconstructs the
original Path.
Use instead of dump() when submitting
#A bug reports against Skia # https://bug.skia.org ##
.
#Example
SkPath path, copy;
path.lineTo(6.f / 7, 2.f / 3);
path.dumpHex();
copy.setFillType(SkPath::kWinding_FillType);
copy.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0
copy.lineTo(SkBits2Float(0x3f5b6db7), SkBits2Float(0x3f2aaaab)); // 0.857143f, 0.666667f
SkDebugf("path is " "%s" "equal to copy\n", path == copy ? "" : "not ");
#StdOut
path.setFillType(SkPath::kWinding_FillType);
path.moveTo(SkBits2Float(0x00000000), SkBits2Float(0x00000000)); // 0, 0
path.lineTo(SkBits2Float(0x3f5b6db7), SkBits2Float(0x3f2aaaab)); // 0.857143f, 0.666667f
path is equal to copy
##
##
#SeeAlso dump SkRect::dumpHex SkRRect::dumpHex writeToMemory
##
# ------------------------------------------------------------------------------
#Method size_t writeToMemory(void* buffer) const
#In Utility
#Line # copies data to buffer ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path, copy;
path.lineTo(6.f / 7, 2.f / 3);
size_t size = path.writeToMemory(nullptr);
SkTDArray storage;
storage.setCount(size);
path.writeToMemory(storage.begin());
copy.readFromMemory(storage.begin(), size);
SkDebugf("path is " "%s" "equal to copy\n", path == copy ? "" : "not ");
}
#StdOut
path is equal to copy
##
##
#SeeAlso serialize readFromMemory dump dumpHex
##
#Method sk_sp serialize() const
#In Utility
#Line # copies data to buffer ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path, copy;
path.lineTo(6.f / 7, 2.f / 3);
sk_sp data = path.serialize();
copy.readFromMemory(data->data(), data->size());
SkDebugf("path is " "%s" "equal to copy\n", path == copy ? "" : "not ");
}
#StdOut
path is equal to copy
##
##
#SeeAlso writeToMemory readFromMemory dump dumpHex
##
# ------------------------------------------------------------------------------
#Method size_t readFromMemory(const void* buffer, size_t length)
#In Utility
#Line # initializes from buffer ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path, copy;
path.lineTo(6.f / 7, 2.f / 3);
size_t size = path.writeToMemory(nullptr);
SkTDArray storage;
storage.setCount(size);
path.writeToMemory(storage.begin());
size_t wrongSize = size - 4;
size_t bytesRead = copy.readFromMemory(storage.begin(), wrongSize);
SkDebugf("length = %u; returned by readFromMemory = %u\n", wrongSize, bytesRead);
size_t largerSize = size + 4;
bytesRead = copy.readFromMemory(storage.begin(), largerSize);
SkDebugf("length = %u; returned by readFromMemory = %u\n", largerSize, bytesRead);
}
#StdOut
length = 32; returned by readFromMemory = 0
length = 40; returned by readFromMemory = 36
##
##
#SeeAlso writeToMemory
##
# ------------------------------------------------------------------------------
#Subtopic Generation_ID
#Alias Generation_IDs ##
#Line # value reflecting contents change ##
Generation_ID provides a quick way to check if Verb_Array, Point_Array, or
Conic_Weight has changed. Generation_ID is not a hash; identical Paths will
not necessarily have matching Generation_IDs.
Empty Paths have a Generation_ID of one.
#Method uint32_t getGenerationID() const
#In Generation_ID
#Line # returns unique ID ##
#Populate
#Example
SkPath path;
SkDebugf("empty genID = %u\n", path.getGenerationID());
path.lineTo(1, 2);
SkDebugf("1st lineTo genID = %u\n", path.getGenerationID());
path.rewind();
SkDebugf("empty genID = %u\n", path.getGenerationID());
path.lineTo(1, 2);
SkDebugf("2nd lineTo genID = %u\n", path.getGenerationID());
#StdOut
empty genID = 1
1st lineTo genID = 2
empty genID = 1
2nd lineTo genID = 3
##
##
#SeeAlso operator==(const SkPath& a, const SkPath& b)
##
#Subtopic ##
# ------------------------------------------------------------------------------
#Method bool isValid() const
#In Property
#In Utility
#Line # returns if data is internally consistent ##
#Populate
#NoExample
##
##
#Method bool pathRefIsValid() const
#Deprecated soon
##
# ------------------------------------------------------------------------------
#Class Iter
#Line # data iterator ##
#Code
#Populate
##
Iterates through Verb_Array, and associated Point_Array and Conic_Weight.
Provides options to treat open Contours as closed, and to ignore
degenerate data.
#Example
#Height 128
#Description
Ignoring the actual Verbs and replacing them with Quads rounds the
path of the glyph.
##
void draw(SkCanvas* canvas) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setTextSize(256);
SkPath asterisk, path;
paint.getTextPath("*", 1, 50, 192, &asterisk);
SkPath::Iter iter(asterisk, true);
SkPoint start[4], pts[4];
iter.next(start); // skip moveTo
iter.next(start); // first quadTo
path.moveTo((start[0] + start[1]) * 0.5f);
while (SkPath::kClose_Verb != iter.next(pts)) {
path.quadTo(pts[0], (pts[0] + pts[1]) * 0.5f);
}
path.quadTo(start[0], (start[0] + start[1]) * 0.5f);
canvas->drawPath(path, paint);
}
##
#SeeAlso RawIter
#Method Iter()
#Line # constructs Path iterator ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath::Iter iter;
SkPoint points[4];
SkDebugf("iter is " "%s" "done\n", SkPath::kDone_Verb == iter.next(points) ? "" : "not ");
SkPath path;
iter.setPath(path, false);
SkDebugf("iter is " "%s" "done\n", SkPath::kDone_Verb == iter.next(points) ? "" : "not ");
}
#StdOut
iter is done
iter is done
##
##
#SeeAlso setPath
##
#Method Iter(const SkPath& path, bool forceClose)
#Line # constructs Path iterator ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, SkPath::Iter& iter) -> void {
SkDebugf("%s:\n", prefix);
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("k%s_Verb ", verbStr[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%g, %g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
SkDebugf("\n");
};
SkPath path;
path.quadTo(10, 20, 30, 40);
SkPath::Iter openIter(path, false);
debugster("open", openIter);
SkPath::Iter closedIter(path, true);
debugster("closed", closedIter);
}
#StdOut
open:
kMove_Verb {0, 0},
kQuad_Verb {0, 0}, {10, 20}, {30, 40},
kDone_Verb
closed:
kMove_Verb {0, 0},
kQuad_Verb {0, 0}, {10, 20}, {30, 40},
kLine_Verb {30, 40}, {0, 0},
kClose_Verb {0, 0},
kDone_Verb
##
##
#SeeAlso setPath
##
#Method void setPath(const SkPath& path, bool forceClose)
#Line # resets Iter to Path ##
#Populate
#Example
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, SkPath::Iter& iter) -> void {
SkDebugf("%s:\n", prefix);
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("k%s_Verb ", verbStr[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%g, %g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
SkDebugf("\n");
};
SkPath path;
path.quadTo(10, 20, 30, 40);
SkPath::Iter iter(path, false);
debugster("quad open", iter);
SkPath path2;
path2.conicTo(1, 2, 3, 4, .5f);
iter.setPath(path2, true);
debugster("conic closed", iter);
}
#StdOut
quad open:
kMove_Verb {0, 0},
kQuad_Verb {0, 0}, {10, 20}, {30, 40},
kDone_Verb
conic closed:
kMove_Verb {0, 0},
kConic_Verb {0, 0}, {1, 2}, {3, 4}, weight = 0.5
kLine_Verb {3, 4}, {0, 0},
kClose_Verb {0, 0},
kDone_Verb
##
##
#SeeAlso Iter(const SkPath& path, bool forceClose)
##
#Method Verb next(SkPoint pts[4], bool doConsumeDegenerates = true, bool exact = false)
#Line # returns next Verb ##
#Populate
#Example
#Description
skip degenerate skips the first in a kMove_Verb pair, the kMove_Verb
followed by the kClose_Verb, the zero length Line and the very small Line.
skip degenerate if exact skips the same as skip degenerate, but shows
the very small Line.
skip none shows all of the Verbs and Points in Path.
##
void draw(SkCanvas* canvas) {
auto debugster = [](const char* prefix, const SkPath& path, bool degen, bool exact) -> void {
SkPath::Iter iter(path, false);
SkDebugf("%s:\n", prefix);
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points, degen, exact);
SkDebugf("k%s_Verb ", verbStr[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%1.8g, %1.8g}, ", points[i].fX, points[i].fY);
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
SkDebugf("\n");
};
SkPath path;
path.moveTo(10, 10);
path.moveTo(20, 20);
path.quadTo(10, 20, 30, 40);
path.moveTo(1, 1);
path.close();
path.moveTo(30, 30);
path.lineTo(30, 30);
path.moveTo(30, 30);
path.lineTo(30.00001f, 30);
debugster("skip degenerate", path, true, false);
debugster("skip degenerate if exact", path, true, true);
debugster("skip none", path, false, false);
}
#StdOut
skip degenerate:
kMove_Verb {20, 20},
kQuad_Verb {20, 20}, {10, 20}, {30, 40},
kDone_Verb
skip degenerate if exact:
kMove_Verb {20, 20},
kQuad_Verb {20, 20}, {10, 20}, {30, 40},
kMove_Verb {30, 30},
kLine_Verb {30, 30}, {30.00001, 30},
kDone_Verb
skip none:
kMove_Verb {10, 10},
kMove_Verb {20, 20},
kQuad_Verb {20, 20}, {10, 20}, {30, 40},
kMove_Verb {1, 1},
kClose_Verb {1, 1},
kMove_Verb {30, 30},
kLine_Verb {30, 30}, {30, 30},
kMove_Verb {30, 30},
kLine_Verb {30, 30}, {30.00001, 30},
kDone_Verb
##
##
#SeeAlso Verb IsLineDegenerate IsCubicDegenerate IsQuadDegenerate
##
#Method SkScalar conicWeight() const
#Line # returns Conic_Weight ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.conicTo(1, 2, 3, 4, .5f);
SkPath::Iter iter(path, false);
SkPoint p[4];
SkDebugf("first verb is " "%s" "move\n", SkPath::kMove_Verb == iter.next(p) ? "" : "not ");
SkDebugf("next verb is " "%s" "conic\n", SkPath::kConic_Verb == iter.next(p) ? "" : "not ");
SkDebugf("conic points: {%g,%g}, {%g,%g}, {%g,%g}\n", p[0].fX, p[0].fY, p[1].fX, p[1].fY,
p[2].fX, p[2].fY);
SkDebugf("conic weight: %g\n", iter.conicWeight());
}
#StdOut
first verb is move
next verb is conic
conic points: {0,0}, {1,2}, {3,4}
conic weight: 0.5
##
##
#SeeAlso Conic_Weight
##
#Method bool isCloseLine() const
#Line # returns if Line was generated by kClose_Verb ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.moveTo(6, 7);
path.conicTo(1, 2, 3, 4, .5f);
path.close();
SkPath::Iter iter(path, false);
SkPoint p[4];
SkDebugf("1st verb is " "%s" "move\n", SkPath::kMove_Verb == iter.next(p) ? "" : "not ");
SkDebugf("moveTo point: {%g,%g}\n", p[0].fX, p[0].fY);
SkDebugf("2nd verb is " "%s" "conic\n", SkPath::kConic_Verb == iter.next(p) ? "" : "not ");
SkDebugf("3rd verb is " "%s" "line\n", SkPath::kLine_Verb == iter.next(p) ? "" : "not ");
SkDebugf("line points: {%g,%g}, {%g,%g}\n", p[0].fX, p[0].fY, p[1].fX, p[1].fY);
SkDebugf("line " "%s" "generated by close\n", iter.isCloseLine() ? "" : "not ");
SkDebugf("4th verb is " "%s" "close\n", SkPath::kClose_Verb == iter.next(p) ? "" : "not ");
}
#StdOut
1st verb is move
moveTo point: {6,7}
2nd verb is conic
3rd verb is line
line points: {3,4}, {6,7}
line generated by close
4th verb is close
##
##
#SeeAlso close()
##
#Method bool isClosedContour() const
#Line # returns if Contour has kClose_Verb ##
#Populate
#Example
void draw(SkCanvas* canvas) {
for (bool forceClose : { false, true } ) {
SkPath path;
path.conicTo(1, 2, 3, 4, .5f);
SkPath::Iter iter(path, forceClose);
SkDebugf("without close(), forceClose is %s: isClosedContour returns %s\n",
forceClose ? "true " : "false", iter.isClosedContour() ? "true" : "false");
path.close();
iter.setPath(path, forceClose);
SkDebugf("with close(), forceClose is %s: isClosedContour returns %s\n",
forceClose ? "true " : "false", iter.isClosedContour() ? "true" : "false");
}
}
#StdOut
without close(), forceClose is false: isClosedContour returns false
with close(), forceClose is false: isClosedContour returns true
without close(), forceClose is true : isClosedContour returns true
with close(), forceClose is true : isClosedContour returns true
##
##
#SeeAlso Iter(const SkPath& path, bool forceClose)
##
#Class Iter ##
#Class RawIter
#Line # raw data iterator ##
#Code
#Populate
##
Iterates through Verb_Array, and associated Point_Array and Conic_Weight.
Verb_Array, Point_Array, and Conic_Weight are returned unaltered.
#Method RawIter()
#Line # constructs empty Path iterator ##
#Populate
#NoExample
##
##
#Method RawIter(const SkPath& path)
#Line # constructs with Path to iterate over ##
#Populate
#NoExample
##
##
#Method void setPath(const SkPath& path)
#Line # sets Path to iterate over ##
#Populate
#NoExample
##
##
#Method Verb next(SkPoint pts[4])
#Line # returns next Verb and associated Points ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.moveTo(50, 60);
path.quadTo(10, 20, 30, 40);
path.close();
path.lineTo(30, 30);
path.conicTo(1, 2, 3, 4, .5f);
path.cubicTo(-1, -2, -3, -4, -5, -6);
SkPath::RawIter iter(path);
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done" };
const int pointCount[] = { 1 , 2 , 3 , 3 , 4 , 1 , 0 };
SkPath::Verb verb;
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("k%s_Verb ", verbStr[(int) verb]);
for (int i = 0; i < pointCount[(int) verb]; ++i) {
SkDebugf("{%1.8g, %1.8g}, ", points[i].fX, points[i].fY);
}
if (SkPath::kConic_Verb == verb) {
SkDebugf("weight = %g", iter.conicWeight());
}
SkDebugf("\n");
} while (SkPath::kDone_Verb != verb);
}
#StdOut
kMove_Verb {50, 60},
kQuad_Verb {50, 60}, {10, 20}, {30, 40},
kClose_Verb {50, 60},
kMove_Verb {50, 60},
kLine_Verb {50, 60}, {30, 30},
kConic_Verb {30, 30}, {1, 2}, {3, 4}, weight = 0.5
kCubic_Verb {3, 4}, {-1, -2}, {-3, -4}, {-5, -6},
kDone_Verb
##
##
#SeeAlso peek()
##
#Method Verb peek() const
#Line # returns next Verb ##
#Populate
#Example
SkPath path;
path.quadTo(10, 20, 30, 40);
path.conicTo(1, 2, 3, 4, .5f);
path.cubicTo(1, 2, 3, 4, .5, 6);
SkPath::RawIter iter(path);
SkPath::Verb verb, peek = iter.peek();
const char* verbStr[] = { "Move", "Line", "Quad", "Conic", "Cubic", "Close", "Done" };
do {
SkPoint points[4];
verb = iter.next(points);
SkDebugf("peek %s %c= verb %s\n", verbStr[peek], peek == verb ? '=' : '!', verbStr[verb]);
peek = iter.peek();
} while (SkPath::kDone_Verb != verb);
SkDebugf("peek %s %c= verb %s\n", verbStr[peek], peek == verb ? '=' : '!', verbStr[verb]);
#StdOut
#Volatile
peek Move == verb Move
peek Quad == verb Quad
peek Conic == verb Conic
peek Cubic == verb Cubic
peek Done == verb Done
peek Done == verb Done
##
##
#Bug 6832
# StdOut is not really volatile, it just produces the wrong result.
# A simple fix changes the output of hairlines and needs to be
# investigated to see if the change is correct or not.
# see change 21340 (abandoned for now)
#SeeAlso next
##
#Method SkScalar conicWeight() const
#Line # returns Conic_Weight ##
#Populate
#Example
void draw(SkCanvas* canvas) {
SkPath path;
path.conicTo(1, 2, 3, 4, .5f);
SkPath::RawIter iter(path);
SkPoint p[4];
SkDebugf("first verb is " "%s" "move\n", SkPath::kMove_Verb == iter.next(p) ? "" : "not ");
SkDebugf("next verb is " "%s" "conic\n", SkPath::kConic_Verb == iter.next(p) ? "" : "not ");
SkDebugf("conic points: {%g,%g}, {%g,%g}, {%g,%g}\n", p[0].fX, p[0].fY, p[1].fX, p[1].fY,
p[2].fX, p[2].fY);
SkDebugf("conic weight: %g\n", iter.conicWeight());
}
#StdOut
first verb is move
next verb is conic
conic points: {0,0}, {1,2}, {3,4}
conic weight: 0.5
##
##
#SeeAlso Conic_Weight
##
#Class RawIter ##
#Class SkPath ##
#Topic Path ##