1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2012 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPaint.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPathUtils.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPoint.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkScalar.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkStrokeRec.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkFloatBits.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkPathPriv.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
18*c8dee2aaSAndroid Build Coastguard Worker
19*c8dee2aaSAndroid Build Coastguard Worker #include <array>
20*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
21*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
22*c8dee2aaSAndroid Build Coastguard Worker
equal(const SkRect & a,const SkRect & b)23*c8dee2aaSAndroid Build Coastguard Worker static bool equal(const SkRect& a, const SkRect& b) {
24*c8dee2aaSAndroid Build Coastguard Worker return SkScalarNearlyEqual(a.left(), b.left()) &&
25*c8dee2aaSAndroid Build Coastguard Worker SkScalarNearlyEqual(a.top(), b.top()) &&
26*c8dee2aaSAndroid Build Coastguard Worker SkScalarNearlyEqual(a.right(), b.right()) &&
27*c8dee2aaSAndroid Build Coastguard Worker SkScalarNearlyEqual(a.bottom(), b.bottom());
28*c8dee2aaSAndroid Build Coastguard Worker }
29*c8dee2aaSAndroid Build Coastguard Worker
test_strokecubic(skiatest::Reporter * reporter)30*c8dee2aaSAndroid Build Coastguard Worker static void test_strokecubic(skiatest::Reporter* reporter) {
31*c8dee2aaSAndroid Build Coastguard Worker uint32_t hexCubicVals[] = {
32*c8dee2aaSAndroid Build Coastguard Worker 0x424c1086, 0x44bcf0cb, // fX=51.0161362 fY=1511.52478
33*c8dee2aaSAndroid Build Coastguard Worker 0x424c107c, 0x44bcf0cb, // fX=51.0160980 fY=1511.52478
34*c8dee2aaSAndroid Build Coastguard Worker 0x424c10c2, 0x44bcf0cb, // fX=51.0163651 fY=1511.52478
35*c8dee2aaSAndroid Build Coastguard Worker 0x424c1119, 0x44bcf0ca, // fX=51.0166969 fY=1511.52466
36*c8dee2aaSAndroid Build Coastguard Worker };
37*c8dee2aaSAndroid Build Coastguard Worker SkPoint cubicVals[] = {
38*c8dee2aaSAndroid Build Coastguard Worker {51.0161362f, 1511.52478f },
39*c8dee2aaSAndroid Build Coastguard Worker {51.0160980f, 1511.52478f },
40*c8dee2aaSAndroid Build Coastguard Worker {51.0163651f, 1511.52478f },
41*c8dee2aaSAndroid Build Coastguard Worker {51.0166969f, 1511.52466f },
42*c8dee2aaSAndroid Build Coastguard Worker };
43*c8dee2aaSAndroid Build Coastguard Worker SkPaint paint;
44*c8dee2aaSAndroid Build Coastguard Worker
45*c8dee2aaSAndroid Build Coastguard Worker paint.setStyle(SkPaint::kStroke_Style);
46*c8dee2aaSAndroid Build Coastguard Worker paint.setStrokeWidth(0.394537568f);
47*c8dee2aaSAndroid Build Coastguard Worker SkPath path, fillPath;
48*c8dee2aaSAndroid Build Coastguard Worker path.moveTo(cubicVals[0]);
49*c8dee2aaSAndroid Build Coastguard Worker path.cubicTo(cubicVals[1], cubicVals[2], cubicVals[3]);
50*c8dee2aaSAndroid Build Coastguard Worker skpathutils::FillPathWithPaint(path, paint, &fillPath);
51*c8dee2aaSAndroid Build Coastguard Worker path.reset();
52*c8dee2aaSAndroid Build Coastguard Worker path.moveTo(SkBits2Float(hexCubicVals[0]), SkBits2Float(hexCubicVals[1]));
53*c8dee2aaSAndroid Build Coastguard Worker path.cubicTo(SkBits2Float(hexCubicVals[2]), SkBits2Float(hexCubicVals[3]),
54*c8dee2aaSAndroid Build Coastguard Worker SkBits2Float(hexCubicVals[4]), SkBits2Float(hexCubicVals[5]),
55*c8dee2aaSAndroid Build Coastguard Worker SkBits2Float(hexCubicVals[6]), SkBits2Float(hexCubicVals[7]));
56*c8dee2aaSAndroid Build Coastguard Worker skpathutils::FillPathWithPaint(path, paint, &fillPath);
57*c8dee2aaSAndroid Build Coastguard Worker }
58*c8dee2aaSAndroid Build Coastguard Worker
test_strokerect(skiatest::Reporter * reporter)59*c8dee2aaSAndroid Build Coastguard Worker static void test_strokerect(skiatest::Reporter* reporter) {
60*c8dee2aaSAndroid Build Coastguard Worker const SkScalar width = SkIntToScalar(10);
61*c8dee2aaSAndroid Build Coastguard Worker SkPaint paint;
62*c8dee2aaSAndroid Build Coastguard Worker
63*c8dee2aaSAndroid Build Coastguard Worker paint.setStyle(SkPaint::kStroke_Style);
64*c8dee2aaSAndroid Build Coastguard Worker paint.setStrokeWidth(width);
65*c8dee2aaSAndroid Build Coastguard Worker
66*c8dee2aaSAndroid Build Coastguard Worker SkRect r = { 0, 0, SkIntToScalar(200), SkIntToScalar(100) };
67*c8dee2aaSAndroid Build Coastguard Worker
68*c8dee2aaSAndroid Build Coastguard Worker SkRect outer(r);
69*c8dee2aaSAndroid Build Coastguard Worker outer.outset(width/2, width/2);
70*c8dee2aaSAndroid Build Coastguard Worker
71*c8dee2aaSAndroid Build Coastguard Worker static const SkPaint::Join joins[] = {
72*c8dee2aaSAndroid Build Coastguard Worker SkPaint::kMiter_Join, SkPaint::kRound_Join, SkPaint::kBevel_Join
73*c8dee2aaSAndroid Build Coastguard Worker };
74*c8dee2aaSAndroid Build Coastguard Worker
75*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(joins); ++i) {
76*c8dee2aaSAndroid Build Coastguard Worker paint.setStrokeJoin(joins[i]);
77*c8dee2aaSAndroid Build Coastguard Worker
78*c8dee2aaSAndroid Build Coastguard Worker SkPath path, fillPath;
79*c8dee2aaSAndroid Build Coastguard Worker path.addRect(r);
80*c8dee2aaSAndroid Build Coastguard Worker skpathutils::FillPathWithPaint(path, paint, &fillPath);
81*c8dee2aaSAndroid Build Coastguard Worker
82*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equal(outer, fillPath.getBounds()));
83*c8dee2aaSAndroid Build Coastguard Worker
84*c8dee2aaSAndroid Build Coastguard Worker bool isMiter = SkPaint::kMiter_Join == joins[i];
85*c8dee2aaSAndroid Build Coastguard Worker SkRect nested[2];
86*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, SkPathPriv::IsNestedFillRects(fillPath, nested) == isMiter);
87*c8dee2aaSAndroid Build Coastguard Worker if (isMiter) {
88*c8dee2aaSAndroid Build Coastguard Worker SkRect inner(r);
89*c8dee2aaSAndroid Build Coastguard Worker inner.inset(width/2, width/2);
90*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equal(nested[0], outer));
91*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equal(nested[1], inner));
92*c8dee2aaSAndroid Build Coastguard Worker }
93*c8dee2aaSAndroid Build Coastguard Worker }
94*c8dee2aaSAndroid Build Coastguard Worker }
95*c8dee2aaSAndroid Build Coastguard Worker
test_strokerec_equality(skiatest::Reporter * reporter)96*c8dee2aaSAndroid Build Coastguard Worker static void test_strokerec_equality(skiatest::Reporter* reporter) {
97*c8dee2aaSAndroid Build Coastguard Worker {
98*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s1(SkStrokeRec::kFill_InitStyle);
99*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s2(SkStrokeRec::kFill_InitStyle);
100*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
101*c8dee2aaSAndroid Build Coastguard Worker
102*c8dee2aaSAndroid Build Coastguard Worker // Test that style mismatch is detected.
103*c8dee2aaSAndroid Build Coastguard Worker s2.setHairlineStyle();
104*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !s1.hasEqualEffect(s2));
105*c8dee2aaSAndroid Build Coastguard Worker
106*c8dee2aaSAndroid Build Coastguard Worker s1.setHairlineStyle();
107*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
108*c8dee2aaSAndroid Build Coastguard Worker
109*c8dee2aaSAndroid Build Coastguard Worker // ResScale is not part of equality.
110*c8dee2aaSAndroid Build Coastguard Worker s1.setResScale(2.1f);
111*c8dee2aaSAndroid Build Coastguard Worker s2.setResScale(1.2f);
112*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
113*c8dee2aaSAndroid Build Coastguard Worker s1.setFillStyle();
114*c8dee2aaSAndroid Build Coastguard Worker s2.setFillStyle();
115*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
116*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeStyle(1.0f, false);
117*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeStyle(1.0f, false);
118*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.9f);
119*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.9f);
120*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
121*c8dee2aaSAndroid Build Coastguard Worker }
122*c8dee2aaSAndroid Build Coastguard Worker
123*c8dee2aaSAndroid Build Coastguard Worker // Stroke parameters on fill or hairline style are not part of equality.
124*c8dee2aaSAndroid Build Coastguard Worker {
125*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s1(SkStrokeRec::kFill_InitStyle);
126*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s2(SkStrokeRec::kFill_InitStyle);
127*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < 2; ++i) {
128*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.9f);
129*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.1f);
130*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
131*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kBevel_Join, 2.9f);
132*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
133*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kRound_Cap, SkPaint::kRound_Join, 2.9f);
134*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
135*c8dee2aaSAndroid Build Coastguard Worker s1.setHairlineStyle();
136*c8dee2aaSAndroid Build Coastguard Worker s2.setHairlineStyle();
137*c8dee2aaSAndroid Build Coastguard Worker }
138*c8dee2aaSAndroid Build Coastguard Worker }
139*c8dee2aaSAndroid Build Coastguard Worker
140*c8dee2aaSAndroid Build Coastguard Worker // Stroke parameters on stroke style are part of equality.
141*c8dee2aaSAndroid Build Coastguard Worker {
142*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s1(SkStrokeRec::kFill_InitStyle);
143*c8dee2aaSAndroid Build Coastguard Worker SkStrokeRec s2(SkStrokeRec::kFill_InitStyle);
144*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.9f);
145*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.9f);
146*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeStyle(1.0f, false);
147*c8dee2aaSAndroid Build Coastguard Worker
148*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeStyle(1.0f, true);
149*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !s1.hasEqualEffect(s2));
150*c8dee2aaSAndroid Build Coastguard Worker
151*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeStyle(2.1f, false);
152*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !s1.hasEqualEffect(s2));
153*c8dee2aaSAndroid Build Coastguard Worker
154*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeStyle(1.0f, false);
155*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
156*c8dee2aaSAndroid Build Coastguard Worker
157*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kRound_Join, 2.1f);
158*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2)); // Miter limit not relevant to butt caps.
159*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kButt_Cap, SkPaint::kBevel_Join, 2.9f);
160*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !s1.hasEqualEffect(s2));
161*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeParams(SkPaint::kRound_Cap, SkPaint::kRound_Join, 2.9f);
162*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !s1.hasEqualEffect(s2));
163*c8dee2aaSAndroid Build Coastguard Worker
164*c8dee2aaSAndroid Build Coastguard Worker // Sets fill.
165*c8dee2aaSAndroid Build Coastguard Worker s1.setStrokeStyle(0.0f, true);
166*c8dee2aaSAndroid Build Coastguard Worker s2.setStrokeStyle(0.0f, true);
167*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, s1.hasEqualEffect(s2));
168*c8dee2aaSAndroid Build Coastguard Worker }
169*c8dee2aaSAndroid Build Coastguard Worker }
170*c8dee2aaSAndroid Build Coastguard Worker
171*c8dee2aaSAndroid Build Coastguard Worker // From skbug.com/6491. The large stroke width can cause numerical instabilities.
test_big_stroke(skiatest::Reporter * reporter)172*c8dee2aaSAndroid Build Coastguard Worker static void test_big_stroke(skiatest::Reporter* reporter) {
173*c8dee2aaSAndroid Build Coastguard Worker SkPaint paint;
174*c8dee2aaSAndroid Build Coastguard Worker paint.setStyle(SkPaint::kStrokeAndFill_Style);
175*c8dee2aaSAndroid Build Coastguard Worker paint.setStrokeWidth(1.49679073e+10f);
176*c8dee2aaSAndroid Build Coastguard Worker
177*c8dee2aaSAndroid Build Coastguard Worker SkPath path;
178*c8dee2aaSAndroid Build Coastguard Worker path.moveTo(SkBits2Float(0x46380000), SkBits2Float(0xc6380000)); // 11776, -11776
179*c8dee2aaSAndroid Build Coastguard Worker path.lineTo(SkBits2Float(0x46a00000), SkBits2Float(0xc6a00000)); // 20480, -20480
180*c8dee2aaSAndroid Build Coastguard Worker path.lineTo(SkBits2Float(0x468c0000), SkBits2Float(0xc68c0000)); // 17920, -17920
181*c8dee2aaSAndroid Build Coastguard Worker path.lineTo(SkBits2Float(0x46100000), SkBits2Float(0xc6100000)); // 9216, -9216
182*c8dee2aaSAndroid Build Coastguard Worker path.lineTo(SkBits2Float(0x46380000), SkBits2Float(0xc6380000)); // 11776, -11776
183*c8dee2aaSAndroid Build Coastguard Worker path.close();
184*c8dee2aaSAndroid Build Coastguard Worker
185*c8dee2aaSAndroid Build Coastguard Worker SkPath strokeAndFillPath;
186*c8dee2aaSAndroid Build Coastguard Worker skpathutils::FillPathWithPaint(path, paint, &strokeAndFillPath);
187*c8dee2aaSAndroid Build Coastguard Worker }
188*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Stroke,reporter)189*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Stroke, reporter) {
190*c8dee2aaSAndroid Build Coastguard Worker test_strokecubic(reporter);
191*c8dee2aaSAndroid Build Coastguard Worker test_strokerect(reporter);
192*c8dee2aaSAndroid Build Coastguard Worker test_strokerec_equality(reporter);
193*c8dee2aaSAndroid Build Coastguard Worker test_big_stroke(reporter);
194*c8dee2aaSAndroid Build Coastguard Worker }
195