caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 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 |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 16 | |
caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 17 | */ |
| 18 | |
| 19 | static const SkDConic testSet[] = { |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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}, |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 24 | {10.834027290344238, 988.964111328125}, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 25 | {163.40835571289062, 988.964111328125}}}, 0.72944212f}, |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 26 | {{{{163.40835571289062, 988.964111328125}, |
| 27 | {5, 988.964111328125}, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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 | |
caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 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}, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 42 | |
caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 43 | }; |
| 44 | |
| 45 | const int testSetCount = (int) SK_ARRAY_COUNT(testSet); |
| 46 | |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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; |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 119 | SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale }, |
| 120 | {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale }, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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[] = { |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 229 | frame0, frame1, frame2, frame3, frame4, frame5, frame6 |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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]; |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 247 | SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale }, |
| 248 | {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale }, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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]; |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 276 | SkDConic dConic = {{{ {dC.fPts[0].fX * scale, dC.fPts[0].fY * scale }, |
| 277 | {dC.fPts[1].fX * scale, dC.fPts[1].fY * scale }, |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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 | |
caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 296 | static void oneOff(skiatest::Reporter* reporter, const SkDConic& c1, const SkDConic& c2, |
| 297 | bool coin) { |
caryclark | ef784fb | 2015-10-30 12:03:06 -0700 | [diff] [blame] | 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 |
caryclark | 1049f12 | 2015-04-20 08:31:59 -0700 | [diff] [blame] | 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 | } |