xref: /aosp_15_r20/external/skia/tests/PathOpsConicIntersectionTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2015 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 #include "include/core/SkPoint.h"
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkScalar.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkGeometry.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkIntersections.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsConic.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsPoint.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsQuad.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsTypes.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "tests/PathOpsTestCommon.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 
21*c8dee2aaSAndroid Build Coastguard Worker /*
22*c8dee2aaSAndroid Build Coastguard Worker manually compute the intersection of a pair of circles and see if the conic intersection matches
23*c8dee2aaSAndroid Build Coastguard Worker   given two circles
24*c8dee2aaSAndroid Build Coastguard Worker     construct a line connecting their centers
25*c8dee2aaSAndroid Build Coastguard Worker 
26*c8dee2aaSAndroid Build Coastguard Worker  */
27*c8dee2aaSAndroid Build Coastguard Worker 
28*c8dee2aaSAndroid Build Coastguard Worker static const ConicPts testSet[] = {
29*c8dee2aaSAndroid Build Coastguard Worker     {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
30*c8dee2aaSAndroid Build Coastguard Worker     {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker     {{{{5.1114602088928223, 628.77813720703125},
33*c8dee2aaSAndroid Build Coastguard Worker         {10.834027290344238, 988.964111328125},
34*c8dee2aaSAndroid Build Coastguard Worker         {163.40835571289062, 988.964111328125}}}, 0.72944212f},
35*c8dee2aaSAndroid Build Coastguard Worker     {{{{163.40835571289062, 988.964111328125},
36*c8dee2aaSAndroid Build Coastguard Worker         {5, 988.964111328125},
37*c8dee2aaSAndroid Build Coastguard Worker         {5, 614.7423095703125}}}, 0.707106769f},
38*c8dee2aaSAndroid Build Coastguard Worker 
39*c8dee2aaSAndroid Build Coastguard Worker     {{{{11.17222976684570312, -8.103978157043457031},
40*c8dee2aaSAndroid Build Coastguard Worker         {22.91432571411132812, -10.37866020202636719},
41*c8dee2aaSAndroid Build Coastguard Worker         {23.7764129638671875, -7.725424289703369141}}}, 1.00862849f},
42*c8dee2aaSAndroid Build Coastguard Worker     {{{{-1.545085430145263672, -4.755282402038574219},
43*c8dee2aaSAndroid Build Coastguard Worker         {22.23132705688476562, -12.48070907592773438},
44*c8dee2aaSAndroid Build Coastguard Worker         {23.7764129638671875, -7.725427150726318359}}}, 0.707106769f},
45*c8dee2aaSAndroid Build Coastguard Worker 
46*c8dee2aaSAndroid Build Coastguard Worker     {{{{-4,1}, {-4,5}, {0,5}}}, 0.707106769f},
47*c8dee2aaSAndroid Build Coastguard Worker     {{{{-3,4}, {-3,1}, {0,1}}}, 0.707106769f},
48*c8dee2aaSAndroid Build Coastguard Worker 
49*c8dee2aaSAndroid Build Coastguard Worker     {{{{0, 0}, {0, 1}, {1, 1}}}, 0.5f},
50*c8dee2aaSAndroid Build Coastguard Worker     {{{{1, 0}, {0, 0}, {0, 1}}}, 0.5f},
51*c8dee2aaSAndroid Build Coastguard Worker 
52*c8dee2aaSAndroid Build Coastguard Worker };
53*c8dee2aaSAndroid Build Coastguard Worker 
54*c8dee2aaSAndroid Build Coastguard Worker const int testSetCount = (int) std::size(testSet);
55*c8dee2aaSAndroid Build Coastguard Worker 
chopCompare(const SkConic chopped[2],const SkDConic dChopped[2])56*c8dee2aaSAndroid Build Coastguard Worker static void chopCompare(const SkConic chopped[2], const SkDConic dChopped[2]) {
57*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(roughly_equal(chopped[0].fW, dChopped[0].fWeight));
58*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(roughly_equal(chopped[1].fW, dChopped[1].fWeight));
59*c8dee2aaSAndroid Build Coastguard Worker     for (int cIndex = 0; cIndex < 2; ++cIndex) {
60*c8dee2aaSAndroid Build Coastguard Worker         for (int pIndex = 0; pIndex < 3; ++pIndex) {
61*c8dee2aaSAndroid Build Coastguard Worker             SkDPoint up;
62*c8dee2aaSAndroid Build Coastguard Worker             up.set(chopped[cIndex].fPts[pIndex]);
63*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(dChopped[cIndex].fPts[pIndex].approximatelyEqual(up));
64*c8dee2aaSAndroid Build Coastguard Worker         }
65*c8dee2aaSAndroid Build Coastguard Worker     }
66*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
67*c8dee2aaSAndroid Build Coastguard Worker     dChopped[0].dump();
68*c8dee2aaSAndroid Build Coastguard Worker     dChopped[1].dump();
69*c8dee2aaSAndroid Build Coastguard Worker #endif
70*c8dee2aaSAndroid Build Coastguard Worker }
71*c8dee2aaSAndroid Build Coastguard Worker 
72*c8dee2aaSAndroid Build Coastguard Worker #define DEBUG_VISUALIZE_CONICS 0
73*c8dee2aaSAndroid Build Coastguard Worker 
74*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
75*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkBitmap.h"
76*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkCanvas.h"
77*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPaint.h"
78*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
79*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsRect.h"
80*c8dee2aaSAndroid Build Coastguard Worker 
writePng(const SkConic & c,const SkConic ch[2],const char * name)81*c8dee2aaSAndroid Build Coastguard Worker static void writePng(const SkConic& c, const SkConic ch[2], const char* name) {
82*c8dee2aaSAndroid Build Coastguard Worker     const int scale = 10;
83*c8dee2aaSAndroid Build Coastguard Worker     SkConic conic, chopped[2];
84*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < 3; ++index) {
85*c8dee2aaSAndroid Build Coastguard Worker         conic.fPts[index].fX = c.fPts[index].fX * scale;
86*c8dee2aaSAndroid Build Coastguard Worker         conic.fPts[index].fY = c.fPts[index].fY * scale;
87*c8dee2aaSAndroid Build Coastguard Worker         for (int chIndex = 0; chIndex < 2; ++chIndex) {
88*c8dee2aaSAndroid Build Coastguard Worker             chopped[chIndex].fPts[index].fX = ch[chIndex].fPts[index].fX * scale;
89*c8dee2aaSAndroid Build Coastguard Worker             chopped[chIndex].fPts[index].fY = ch[chIndex].fPts[index].fY * scale;
90*c8dee2aaSAndroid Build Coastguard Worker         }
91*c8dee2aaSAndroid Build Coastguard Worker     }
92*c8dee2aaSAndroid Build Coastguard Worker     conic.fW = c.fW;
93*c8dee2aaSAndroid Build Coastguard Worker     chopped[0].fW = ch[0].fW;
94*c8dee2aaSAndroid Build Coastguard Worker     chopped[1].fW = ch[1].fW;
95*c8dee2aaSAndroid Build Coastguard Worker     SkBitmap bitmap;
96*c8dee2aaSAndroid Build Coastguard Worker     SkRect bounds;
97*c8dee2aaSAndroid Build Coastguard Worker     conic.computeTightBounds(&bounds);
98*c8dee2aaSAndroid Build Coastguard Worker     bounds.outset(10, 10);
99*c8dee2aaSAndroid Build Coastguard Worker     bitmap.tryAllocPixels(SkImageInfo::MakeN32Premul(
100*c8dee2aaSAndroid Build Coastguard Worker           SkScalarRoundToInt(bounds.width()), SkScalarRoundToInt(bounds.height())));
101*c8dee2aaSAndroid Build Coastguard Worker     SkCanvas canvas(bitmap);
102*c8dee2aaSAndroid Build Coastguard Worker     SkPaint paint;
103*c8dee2aaSAndroid Build Coastguard Worker     paint.setAntiAlias(true);
104*c8dee2aaSAndroid Build Coastguard Worker     paint.setStyle(SkPaint::kStroke_Style);
105*c8dee2aaSAndroid Build Coastguard Worker     canvas.translate(-bounds.fLeft, -bounds.fTop);
106*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawColor(SK_ColorWHITE);
107*c8dee2aaSAndroid Build Coastguard Worker     SkPath path;
108*c8dee2aaSAndroid Build Coastguard Worker     path.moveTo(conic.fPts[0]);
109*c8dee2aaSAndroid Build Coastguard Worker     path.conicTo(conic.fPts[1], conic.fPts[2], conic.fW);
110*c8dee2aaSAndroid Build Coastguard Worker     paint.setARGB(0x80, 0xFF, 0, 0);
111*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawPath(path, paint);
112*c8dee2aaSAndroid Build Coastguard Worker     path.reset();
113*c8dee2aaSAndroid Build Coastguard Worker     path.moveTo(chopped[0].fPts[0]);
114*c8dee2aaSAndroid Build Coastguard Worker     path.conicTo(chopped[0].fPts[1], chopped[0].fPts[2], chopped[0].fW);
115*c8dee2aaSAndroid Build Coastguard Worker     path.moveTo(chopped[1].fPts[0]);
116*c8dee2aaSAndroid Build Coastguard Worker     path.conicTo(chopped[1].fPts[1], chopped[1].fPts[2], chopped[1].fW);
117*c8dee2aaSAndroid Build Coastguard Worker     paint.setARGB(0x80, 0, 0, 0xFF);
118*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawPath(path, paint);
119*c8dee2aaSAndroid Build Coastguard Worker     SkString filename("c:\\Users\\caryclark\\Documents\\");
120*c8dee2aaSAndroid Build Coastguard Worker     filename.appendf("%s.png", name);
121*c8dee2aaSAndroid Build Coastguard Worker     ToolUtils::EncodeImageToPngFile(filename.c_str(), bitmap);
122*c8dee2aaSAndroid Build Coastguard Worker }
123*c8dee2aaSAndroid Build Coastguard Worker 
writeDPng(const SkDConic & dC,const char * name)124*c8dee2aaSAndroid Build Coastguard Worker static void writeDPng(const SkDConic& dC, const char* name) {
125*c8dee2aaSAndroid Build Coastguard Worker     const int scale = 5;
126*c8dee2aaSAndroid Build Coastguard Worker     SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale },
127*c8dee2aaSAndroid Build Coastguard Worker         {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale },
128*c8dee2aaSAndroid Build Coastguard Worker         {dC.fPts[2].fX * scale, dC.fPts[2].fY * scale }}}, dC.fWeight };
129*c8dee2aaSAndroid Build Coastguard Worker     SkBitmap bitmap;
130*c8dee2aaSAndroid Build Coastguard Worker     SkDRect bounds;
131*c8dee2aaSAndroid Build Coastguard Worker     bounds.setBounds(dConic);
132*c8dee2aaSAndroid Build Coastguard Worker     bounds.fLeft -= 10;
133*c8dee2aaSAndroid Build Coastguard Worker     bounds.fTop -= 10;
134*c8dee2aaSAndroid Build Coastguard Worker     bounds.fRight += 10;
135*c8dee2aaSAndroid Build Coastguard Worker     bounds.fBottom += 10;
136*c8dee2aaSAndroid Build Coastguard Worker     bitmap.tryAllocPixels(SkImageInfo::MakeN32Premul(
137*c8dee2aaSAndroid Build Coastguard Worker           SkScalarRoundToInt(SkDoubleToScalar(bounds.width())),
138*c8dee2aaSAndroid Build Coastguard Worker           SkScalarRoundToInt(SkDoubleToScalar(bounds.height()))));
139*c8dee2aaSAndroid Build Coastguard Worker     SkCanvas canvas(bitmap);
140*c8dee2aaSAndroid Build Coastguard Worker     SkPaint paint;
141*c8dee2aaSAndroid Build Coastguard Worker     paint.setAntiAlias(true);
142*c8dee2aaSAndroid Build Coastguard Worker     paint.setStyle(SkPaint::kStroke_Style);
143*c8dee2aaSAndroid Build Coastguard Worker     canvas.translate(SkDoubleToScalar(-bounds.fLeft), SkDoubleToScalar(-bounds.fTop));
144*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawColor(SK_ColorWHITE);
145*c8dee2aaSAndroid Build Coastguard Worker     SkPath path;
146*c8dee2aaSAndroid Build Coastguard Worker     path.moveTo(dConic.fPts[0].asSkPoint());
147*c8dee2aaSAndroid Build Coastguard Worker     path.conicTo(dConic.fPts[1].asSkPoint(), dConic.fPts[2].asSkPoint(), dConic.fWeight);
148*c8dee2aaSAndroid Build Coastguard Worker     paint.setARGB(0x80, 0xFF, 0, 0);
149*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawPath(path, paint);
150*c8dee2aaSAndroid Build Coastguard Worker     path.reset();
151*c8dee2aaSAndroid Build Coastguard Worker     const int chops = 2;
152*c8dee2aaSAndroid Build Coastguard Worker     for (int tIndex = 0; tIndex < chops; ++tIndex) {
153*c8dee2aaSAndroid Build Coastguard Worker         SkDConic chopped = dConic.subDivide(tIndex / (double) chops,
154*c8dee2aaSAndroid Build Coastguard Worker                 (tIndex + 1) / (double) chops);
155*c8dee2aaSAndroid Build Coastguard Worker         path.moveTo(chopped.fPts[0].asSkPoint());
156*c8dee2aaSAndroid Build Coastguard Worker         path.conicTo(chopped.fPts[1].asSkPoint(), chopped.fPts[2].asSkPoint(), chopped.fWeight);
157*c8dee2aaSAndroid Build Coastguard Worker     }
158*c8dee2aaSAndroid Build Coastguard Worker     paint.setARGB(0x80, 0, 0, 0xFF);
159*c8dee2aaSAndroid Build Coastguard Worker     canvas.drawPath(path, paint);
160*c8dee2aaSAndroid Build Coastguard Worker     SkString filename("c:\\Users\\caryclark\\Documents\\");
161*c8dee2aaSAndroid Build Coastguard Worker     filename.appendf("%s.png", name);
162*c8dee2aaSAndroid Build Coastguard Worker     ToolUtils::EncodeImageToPngFile(filename.c_str(), bitmap);
163*c8dee2aaSAndroid Build Coastguard Worker }
164*c8dee2aaSAndroid Build Coastguard Worker #endif
165*c8dee2aaSAndroid Build Coastguard Worker 
chopBothWays(const SkDConic & dConic,double t,const char * name)166*c8dee2aaSAndroid Build Coastguard Worker static void chopBothWays(const SkDConic& dConic, double t, const char* name) {
167*c8dee2aaSAndroid Build Coastguard Worker     SkConic conic;
168*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < 3; ++index) {
169*c8dee2aaSAndroid Build Coastguard Worker         conic.fPts[index] = dConic.fPts[index].asSkPoint();
170*c8dee2aaSAndroid Build Coastguard Worker     }
171*c8dee2aaSAndroid Build Coastguard Worker     conic.fW = dConic.fWeight;
172*c8dee2aaSAndroid Build Coastguard Worker     SkConic chopped[2];
173*c8dee2aaSAndroid Build Coastguard Worker     SkDConic dChopped[2];
174*c8dee2aaSAndroid Build Coastguard Worker     if (!conic.chopAt(SkDoubleToScalar(t), chopped)) {
175*c8dee2aaSAndroid Build Coastguard Worker         return;
176*c8dee2aaSAndroid Build Coastguard Worker     }
177*c8dee2aaSAndroid Build Coastguard Worker     dChopped[0] = dConic.subDivide(0, t);
178*c8dee2aaSAndroid Build Coastguard Worker     dChopped[1] = dConic.subDivide(t, 1);
179*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
180*c8dee2aaSAndroid Build Coastguard Worker     dConic.dump();
181*c8dee2aaSAndroid Build Coastguard Worker #endif
182*c8dee2aaSAndroid Build Coastguard Worker     chopCompare(chopped, dChopped);
183*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
184*c8dee2aaSAndroid Build Coastguard Worker     writePng(conic, chopped, name);
185*c8dee2aaSAndroid Build Coastguard Worker #endif
186*c8dee2aaSAndroid Build Coastguard Worker }
187*c8dee2aaSAndroid Build Coastguard Worker 
188*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
189*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame0[] = {
190*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
191*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
192*c8dee2aaSAndroid Build Coastguard Worker };
193*c8dee2aaSAndroid Build Coastguard Worker 
194*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame1[] = {
195*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
196*c8dee2aaSAndroid Build Coastguard Worker {{{{306.58801299999999, -227.983994}, {212.46499600000001, -262.24200400000001}, {95.551200899999998, 58.976398500000002}}}, 0.707107008f},
197*c8dee2aaSAndroid Build Coastguard Worker {{{{377.21899400000001, -141.98100299999999}, {237.77799285476553, -166.56830755921084}, {134.08399674208422, -155.06258330544892}}}, 0.788580656f},
198*c8dee2aaSAndroid Build Coastguard Worker {{{{134.08399674208422, -155.06258330544892}, {30.390000629402859, -143.55685905168704}, {23.185499199999999, -102.697998}}}, 0.923879623f},
199*c8dee2aaSAndroid Build Coastguard Worker };
200*c8dee2aaSAndroid Build Coastguard Worker 
201*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame2[] = {
202*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
203*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
204*c8dee2aaSAndroid Build Coastguard Worker {{{{205.78973252799028, -158.12538713371103}, {143.97848953841861, -74.076645245042371}, {95.551200899999998, 58.976398500000002}}}, 0.923879623f},
205*c8dee2aaSAndroid Build Coastguard Worker {{{{377.21899400000001, -141.98100299999999}, {237.77799285476553, -166.56830755921084}, {134.08399674208422, -155.06258330544892}}}, 0.788580656f},
206*c8dee2aaSAndroid Build Coastguard Worker };
207*c8dee2aaSAndroid Build Coastguard Worker 
208*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame3[] = {
209*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
210*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
211*c8dee2aaSAndroid Build Coastguard Worker {{{{205.78973252799028, -158.12538713371103}, {143.97848953841861, -74.076645245042371}, {95.551200899999998, 58.976398500000002}}}, 0.923879623f},
212*c8dee2aaSAndroid Build Coastguard Worker {{{{252.08225670812539, -156.90491625851064}, {185.93099479842493, -160.81544543232982}, {134.08399674208422, -155.06258330544892}}}, 0.835816324f},
213*c8dee2aaSAndroid Build Coastguard Worker };
214*c8dee2aaSAndroid Build Coastguard Worker 
215*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame4[] = {
216*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
217*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
218*c8dee2aaSAndroid Build Coastguard Worker {{{{205.78973252799028, -158.12538713371103}, {174.88411103320448, -116.10101618937664}, {145.19509369736275, -56.857102571363754}}}, 0.871667147f},
219*c8dee2aaSAndroid Build Coastguard Worker {{{{252.08225670812539, -156.90491625851064}, {185.93099479842493, -160.81544543232982}, {134.08399674208422, -155.06258330544892}}}, 0.835816324f},
220*c8dee2aaSAndroid Build Coastguard Worker };
221*c8dee2aaSAndroid Build Coastguard Worker 
222*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame5[] = {
223*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
224*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
225*c8dee2aaSAndroid Build Coastguard Worker {{{{205.78973252799028, -158.12538713371103}, {174.88411103320448, -116.10101618937664}, {145.19509369736275, -56.857102571363754}}}, 0.871667147f},
226*c8dee2aaSAndroid Build Coastguard Worker {{{{252.08225670812539, -156.90491625851064}, {219.70109133058406, -158.81912754088933}, {190.17095392508796, -158.38373974664466}}}, 0.858306944f},
227*c8dee2aaSAndroid Build Coastguard Worker };
228*c8dee2aaSAndroid Build Coastguard Worker 
229*c8dee2aaSAndroid Build Coastguard Worker const SkDConic frame6[] = {
230*c8dee2aaSAndroid Build Coastguard Worker {{{{306.588013,-227.983994}, {212.464996,-262.242004}, {95.5512009,58.9763985}}}, 0.707107008f},
231*c8dee2aaSAndroid Build Coastguard Worker {{{{377.218994,-141.981003}, {40.578701,-201.339996}, {23.1854992,-102.697998}}}, 0.707107008f},
232*c8dee2aaSAndroid Build Coastguard Worker {{{{205.78973252799028, -158.12538713371103}, {190.33692178059735, -137.11320166154385}, {174.87004877564593, -111.2132534799228}}}, 0.858117759f},
233*c8dee2aaSAndroid Build Coastguard Worker {{{{252.08225670812539, -156.90491625851064}, {219.70109133058406, -158.81912754088933}, {190.17095392508796, -158.38373974664466}}}, 0.858306944f},
234*c8dee2aaSAndroid Build Coastguard Worker };
235*c8dee2aaSAndroid Build Coastguard Worker 
236*c8dee2aaSAndroid Build Coastguard Worker const SkDConic* frames[] = {
237*c8dee2aaSAndroid Build Coastguard Worker     frame0, frame1, frame2, frame3, frame4, frame5, frame6
238*c8dee2aaSAndroid Build Coastguard Worker };
239*c8dee2aaSAndroid Build Coastguard Worker 
240*c8dee2aaSAndroid Build Coastguard Worker const int frameSizes[] = { (int) std::size(frame0), (int) std::size(frame1),
241*c8dee2aaSAndroid Build Coastguard Worker         (int) std::size(frame2), (int) std::size(frame3),
242*c8dee2aaSAndroid Build Coastguard Worker         (int) std::size(frame4), (int) std::size(frame5),
243*c8dee2aaSAndroid Build Coastguard Worker         (int) std::size(frame6),
244*c8dee2aaSAndroid Build Coastguard Worker };
245*c8dee2aaSAndroid Build Coastguard Worker 
writeFrames()246*c8dee2aaSAndroid Build Coastguard Worker static void writeFrames() {
247*c8dee2aaSAndroid Build Coastguard Worker     const int scale = 5;
248*c8dee2aaSAndroid Build Coastguard Worker 
249*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < (int) std::size(frameSizes); ++index) {
250*c8dee2aaSAndroid Build Coastguard Worker         SkDRect bounds;
251*c8dee2aaSAndroid Build Coastguard Worker         bool boundsSet = false;
252*c8dee2aaSAndroid Build Coastguard Worker         int frameSize = frameSizes[index];
253*c8dee2aaSAndroid Build Coastguard Worker         for (int fIndex = 0; fIndex < frameSize; ++fIndex) {
254*c8dee2aaSAndroid Build Coastguard Worker             const SkDConic& dC = frames[index][fIndex];
255*c8dee2aaSAndroid Build Coastguard Worker             SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale },
256*c8dee2aaSAndroid Build Coastguard Worker                 {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale },
257*c8dee2aaSAndroid Build Coastguard Worker                 {dC.fPts[2].fX * scale, dC.fPts[2].fY * scale }}}, dC.fWeight };
258*c8dee2aaSAndroid Build Coastguard Worker             SkDRect dBounds;
259*c8dee2aaSAndroid Build Coastguard Worker             dBounds.setBounds(dConic);
260*c8dee2aaSAndroid Build Coastguard Worker             if (!boundsSet) {
261*c8dee2aaSAndroid Build Coastguard Worker                 bounds = dBounds;
262*c8dee2aaSAndroid Build Coastguard Worker                 boundsSet = true;
263*c8dee2aaSAndroid Build Coastguard Worker             } else {
264*c8dee2aaSAndroid Build Coastguard Worker                 bounds.add((SkDPoint&) dBounds.fLeft);
265*c8dee2aaSAndroid Build Coastguard Worker                 bounds.add((SkDPoint&) dBounds.fRight);
266*c8dee2aaSAndroid Build Coastguard Worker             }
267*c8dee2aaSAndroid Build Coastguard Worker         }
268*c8dee2aaSAndroid Build Coastguard Worker         bounds.fLeft -= 10;
269*c8dee2aaSAndroid Build Coastguard Worker         bounds.fTop -= 10;
270*c8dee2aaSAndroid Build Coastguard Worker         bounds.fRight += 10;
271*c8dee2aaSAndroid Build Coastguard Worker         bounds.fBottom += 10;
272*c8dee2aaSAndroid Build Coastguard Worker         SkBitmap bitmap;
273*c8dee2aaSAndroid Build Coastguard Worker         bitmap.tryAllocPixels(SkImageInfo::MakeN32Premul(
274*c8dee2aaSAndroid Build Coastguard Worker               SkScalarRoundToInt(SkDoubleToScalar(bounds.width())),
275*c8dee2aaSAndroid Build Coastguard Worker               SkScalarRoundToInt(SkDoubleToScalar(bounds.height()))));
276*c8dee2aaSAndroid Build Coastguard Worker         SkCanvas canvas(bitmap);
277*c8dee2aaSAndroid Build Coastguard Worker         SkPaint paint;
278*c8dee2aaSAndroid Build Coastguard Worker         paint.setAntiAlias(true);
279*c8dee2aaSAndroid Build Coastguard Worker         paint.setStyle(SkPaint::kStroke_Style);
280*c8dee2aaSAndroid Build Coastguard Worker         canvas.translate(SkDoubleToScalar(-bounds.fLeft), SkDoubleToScalar(-bounds.fTop));
281*c8dee2aaSAndroid Build Coastguard Worker         canvas.drawColor(SK_ColorWHITE);
282*c8dee2aaSAndroid Build Coastguard Worker         for (int fIndex = 0; fIndex < frameSize; ++fIndex) {
283*c8dee2aaSAndroid Build Coastguard Worker             const SkDConic& dC = frames[index][fIndex];
284*c8dee2aaSAndroid Build Coastguard Worker             SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale },
285*c8dee2aaSAndroid Build Coastguard Worker                 {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale },
286*c8dee2aaSAndroid Build Coastguard Worker                 {dC.fPts[2].fX * scale, dC.fPts[2].fY * scale }}}, dC.fWeight };
287*c8dee2aaSAndroid Build Coastguard Worker             SkPath path;
288*c8dee2aaSAndroid Build Coastguard Worker             path.moveTo(dConic.fPts[0].asSkPoint());
289*c8dee2aaSAndroid Build Coastguard Worker             path.conicTo(dConic.fPts[1].asSkPoint(), dConic.fPts[2].asSkPoint(), dConic.fWeight);
290*c8dee2aaSAndroid Build Coastguard Worker             if (fIndex < 2) {
291*c8dee2aaSAndroid Build Coastguard Worker                 paint.setARGB(0x80, 0xFF, 0, 0);
292*c8dee2aaSAndroid Build Coastguard Worker             } else {
293*c8dee2aaSAndroid Build Coastguard Worker                 paint.setARGB(0x80, 0, 0, 0xFF);
294*c8dee2aaSAndroid Build Coastguard Worker             }
295*c8dee2aaSAndroid Build Coastguard Worker             canvas.drawPath(path, paint);
296*c8dee2aaSAndroid Build Coastguard Worker         }
297*c8dee2aaSAndroid Build Coastguard Worker         SkString filename("c:\\Users\\caryclark\\Documents\\");
298*c8dee2aaSAndroid Build Coastguard Worker         filename.appendf("f%d.png", index);
299*c8dee2aaSAndroid Build Coastguard Worker         ToolUtils::EncodeImageToPngFile(filename.c_str(), bitmap);
300*c8dee2aaSAndroid Build Coastguard Worker     }
301*c8dee2aaSAndroid Build Coastguard Worker }
302*c8dee2aaSAndroid Build Coastguard Worker #endif
303*c8dee2aaSAndroid Build Coastguard Worker 
oneOff(skiatest::Reporter * reporter,const ConicPts & conic1,const ConicPts & conic2,bool coin)304*c8dee2aaSAndroid Build Coastguard Worker static void oneOff(skiatest::Reporter* reporter, const ConicPts& conic1, const ConicPts& conic2,
305*c8dee2aaSAndroid Build Coastguard Worker         bool coin) {
306*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
307*c8dee2aaSAndroid Build Coastguard Worker     writeFrames();
308*c8dee2aaSAndroid Build Coastguard Worker #endif
309*c8dee2aaSAndroid Build Coastguard Worker     SkDConic c1, c2;
310*c8dee2aaSAndroid Build Coastguard Worker     c1.debugSet(conic1.fPts.fPts, conic1.fWeight);
311*c8dee2aaSAndroid Build Coastguard Worker     c2.debugSet(conic2.fPts.fPts, conic2.fWeight);
312*c8dee2aaSAndroid Build Coastguard Worker     chopBothWays(c1, 0.5, "c1");
313*c8dee2aaSAndroid Build Coastguard Worker     chopBothWays(c2, 0.5, "c2");
314*c8dee2aaSAndroid Build Coastguard Worker #if DEBUG_VISUALIZE_CONICS
315*c8dee2aaSAndroid Build Coastguard Worker     writeDPng(c1, "d1");
316*c8dee2aaSAndroid Build Coastguard Worker     writeDPng(c2, "d2");
317*c8dee2aaSAndroid Build Coastguard Worker #endif
318*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(ValidConic(c1));
319*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(ValidConic(c2));
320*c8dee2aaSAndroid Build Coastguard Worker     SkIntersections intersections;
321*c8dee2aaSAndroid Build Coastguard Worker     intersections.intersect(c1, c2);
322*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, !coin || intersections.used() == 2);
323*c8dee2aaSAndroid Build Coastguard Worker     double tt1, tt2;
324*c8dee2aaSAndroid Build Coastguard Worker     SkDPoint xy1, xy2;
325*c8dee2aaSAndroid Build Coastguard Worker     for (int pt3 = 0; pt3 < intersections.used(); ++pt3) {
326*c8dee2aaSAndroid Build Coastguard Worker         tt1 = intersections[0][pt3];
327*c8dee2aaSAndroid Build Coastguard Worker         xy1 = c1.ptAtT(tt1);
328*c8dee2aaSAndroid Build Coastguard Worker         tt2 = intersections[1][pt3];
329*c8dee2aaSAndroid Build Coastguard Worker         xy2 = c2.ptAtT(tt2);
330*c8dee2aaSAndroid Build Coastguard Worker         const SkDPoint& iPt = intersections.pt(pt3);
331*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, xy1.approximatelyEqual(iPt));
332*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, xy2.approximatelyEqual(iPt));
333*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, xy1.approximatelyEqual(xy2));
334*c8dee2aaSAndroid Build Coastguard Worker     }
335*c8dee2aaSAndroid Build Coastguard Worker     reporter->bumpTestCount();
336*c8dee2aaSAndroid Build Coastguard Worker }
337*c8dee2aaSAndroid Build Coastguard Worker 
oneOff(skiatest::Reporter * reporter,int outer,int inner)338*c8dee2aaSAndroid Build Coastguard Worker static void oneOff(skiatest::Reporter* reporter, int outer, int inner) {
339*c8dee2aaSAndroid Build Coastguard Worker     const ConicPts& c1 = testSet[outer];
340*c8dee2aaSAndroid Build Coastguard Worker     const ConicPts& c2 = testSet[inner];
341*c8dee2aaSAndroid Build Coastguard Worker     oneOff(reporter, c1, c2, false);
342*c8dee2aaSAndroid Build Coastguard Worker }
343*c8dee2aaSAndroid Build Coastguard Worker 
oneOffTests(skiatest::Reporter * reporter)344*c8dee2aaSAndroid Build Coastguard Worker static void oneOffTests(skiatest::Reporter* reporter) {
345*c8dee2aaSAndroid Build Coastguard Worker     for (int outer = 0; outer < testSetCount - 1; ++outer) {
346*c8dee2aaSAndroid Build Coastguard Worker         for (int inner = outer + 1; inner < testSetCount; ++inner) {
347*c8dee2aaSAndroid Build Coastguard Worker             oneOff(reporter, outer, inner);
348*c8dee2aaSAndroid Build Coastguard Worker         }
349*c8dee2aaSAndroid Build Coastguard Worker     }
350*c8dee2aaSAndroid Build Coastguard Worker }
351*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(PathOpsConicIntersectionOneOff,reporter)352*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(PathOpsConicIntersectionOneOff, reporter) {
353*c8dee2aaSAndroid Build Coastguard Worker     oneOff(reporter, 0, 1);
354*c8dee2aaSAndroid Build Coastguard Worker }
355*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(PathOpsConicIntersection,reporter)356*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(PathOpsConicIntersection, reporter) {
357*c8dee2aaSAndroid Build Coastguard Worker     oneOffTests(reporter);
358*c8dee2aaSAndroid Build Coastguard Worker }
359