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