OLD | NEW |
(Empty) | |
| 1 /* |
| 2 * Copyright 2010, Google Inc. |
| 3 * All rights reserved. |
| 4 * |
| 5 * Redistribution and use in source and binary forms, with or without |
| 6 * modification, are permitted provided that the following conditions are |
| 7 * met: |
| 8 * |
| 9 * * Redistributions of source code must retain the above copyright |
| 10 * notice, this list of conditions and the following disclaimer. |
| 11 * * Redistributions in binary form must reproduce the above |
| 12 * copyright notice, this list of conditions and the following disclaimer |
| 13 * in the documentation and/or other materials provided with the |
| 14 * distribution. |
| 15 * * Neither the name of Google Inc. nor the names of its |
| 16 * contributors may be used to endorse or promote products derived from |
| 17 * this software without specific prior written permission. |
| 18 * |
| 19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 20 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 21 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 22 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 23 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 26 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 27 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 28 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 29 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 30 */ |
| 31 |
| 32 #include "core/cross/gpu2d/path_processor.h" |
| 33 |
| 34 #include <algorithm> |
| 35 |
| 36 #include "base/logging.h" |
| 37 #include "core/cross/gpu2d/arena.h" |
| 38 #include "core/cross/gpu2d/cubic_classifier.h" |
| 39 #include "core/cross/gpu2d/cubic_math_utils.h" |
| 40 #include "core/cross/gpu2d/cubic_texture_coords.h" |
| 41 #include "core/cross/gpu2d/interval_tree.h" |
| 42 #include "core/cross/gpu2d/local_triangulator.h" |
| 43 #include "core/cross/gpu2d/path_cache.h" |
| 44 #include "third_party/skia/include/core/SkGeometry.h" |
| 45 #include "third_party/skia/include/core/SkPath.h" |
| 46 #include "third_party/skia/include/core/SkScalar.h" |
| 47 #include "third_party/glu/internal_glu.h" |
| 48 |
| 49 namespace o3d { |
| 50 namespace gpu2d { |
| 51 |
| 52 class Contour; |
| 53 |
| 54 //---------------------------------------------------------------------- |
| 55 // min / max helpers |
| 56 // |
| 57 |
| 58 template <typename T> |
| 59 T min2(const T& v1, const T& v2) { |
| 60 return std::min(v1, v2); |
| 61 } |
| 62 |
| 63 template <typename T> |
| 64 T max2(const T& v1, const T& v2) { |
| 65 return std::max(v1, v2); |
| 66 } |
| 67 |
| 68 template <typename T> |
| 69 T min3(const T& v1, const T& v2, const T& v3) { |
| 70 return min2(min2(v1, v2), v3); |
| 71 } |
| 72 |
| 73 template <typename T> |
| 74 T max3(const T& v1, const T& v2, const T& v3) { |
| 75 return max2(max2(v1, v2), v3); |
| 76 } |
| 77 |
| 78 template <typename T> |
| 79 T min4(const T& v1, const T& v2, const T& v3, const T& v4) { |
| 80 return min2(min2(v1, v2), min2(v3, v4)); |
| 81 } |
| 82 |
| 83 template <typename T> |
| 84 T max4(const T& v1, const T& v2, const T& v3, const T& v4) { |
| 85 return max2(max2(v1, v2), max2(v3, v4)); |
| 86 } |
| 87 |
| 88 //---------------------------------------------------------------------- |
| 89 // BBox |
| 90 // |
| 91 |
| 92 // Extremely simple bounding box class for Segments. |
| 93 class BBox { |
| 94 public: |
| 95 BBox() |
| 96 : min_x_(0), |
| 97 min_y_(0), |
| 98 max_x_(0), |
| 99 max_y_(0) { |
| 100 } |
| 101 |
| 102 // Initializes the parameters of the bounding box. |
| 103 void Setup(float min_x, |
| 104 float min_y, |
| 105 float max_x, |
| 106 float max_y) { |
| 107 min_x_ = min_x; |
| 108 min_y_ = min_y; |
| 109 max_x_ = max_x; |
| 110 max_y_ = max_y; |
| 111 } |
| 112 |
| 113 // Initializes the bounding box to surround the given triangle. |
| 114 void Setup(LocalTriangulator::Triangle* triangle) { |
| 115 Setup(min3(triangle->get_vertex(0)->x(), |
| 116 triangle->get_vertex(1)->x(), |
| 117 triangle->get_vertex(2)->x()), |
| 118 min3(triangle->get_vertex(0)->y(), |
| 119 triangle->get_vertex(1)->y(), |
| 120 triangle->get_vertex(2)->y()), |
| 121 max3(triangle->get_vertex(0)->x(), |
| 122 triangle->get_vertex(1)->x(), |
| 123 triangle->get_vertex(2)->x()), |
| 124 max3(triangle->get_vertex(0)->y(), |
| 125 triangle->get_vertex(1)->y(), |
| 126 triangle->get_vertex(2)->y())); |
| 127 } |
| 128 |
| 129 float min_x() const { return min_x_; } |
| 130 float min_y() const { return min_y_; } |
| 131 float max_x() const { return max_x_; } |
| 132 float max_y() const { return max_y_; } |
| 133 |
| 134 private: |
| 135 float min_x_; |
| 136 float min_y_; |
| 137 float max_x_; |
| 138 float max_y_; |
| 139 DISALLOW_COPY_AND_ASSIGN(BBox); |
| 140 }; |
| 141 |
| 142 //---------------------------------------------------------------------- |
| 143 // Segment |
| 144 // |
| 145 |
| 146 // Describes a segment of the path: either a cubic or a line segment. |
| 147 // These are stored in a doubly linked list to speed up curve |
| 148 // subdivision, which occurs due to either rendering artifacts in the |
| 149 // loop case or due to overlapping triangles. |
| 150 class Segment { |
| 151 public: |
| 152 // The kind of segment: cubic or line. |
| 153 enum Kind { |
| 154 kCubic, |
| 155 kLine |
| 156 }; |
| 157 |
| 158 // No-argument constructor allows construction by the Arena class. |
| 159 Segment() |
| 160 : arena_(NULL), |
| 161 kind_(kCubic), |
| 162 prev_(NULL), |
| 163 next_(NULL), |
| 164 contour_(NULL), |
| 165 triangulator_(NULL), |
| 166 marked_for_subdivision_(false) { |
| 167 } |
| 168 |
| 169 // Initializer for cubic curve segments. |
| 170 void Setup(Arena* arena, |
| 171 Contour* contour, |
| 172 SkPoint cp0, |
| 173 SkPoint cp1, |
| 174 SkPoint cp2, |
| 175 SkPoint cp3) { |
| 176 arena_ = arena; |
| 177 contour_ = contour; |
| 178 kind_ = kCubic; |
| 179 points_[0] = cp0; |
| 180 points_[1] = cp1; |
| 181 points_[2] = cp2; |
| 182 points_[3] = cp3; |
| 183 ComputeBoundingBox(); |
| 184 } |
| 185 |
| 186 // Initializer for line segments. |
| 187 void Setup(Arena* arena, |
| 188 Contour* contour, |
| 189 SkPoint p0, |
| 190 SkPoint p1) { |
| 191 arena_ = arena; |
| 192 contour_ = contour; |
| 193 kind_ = kLine; |
| 194 points_[0] = p0; |
| 195 points_[1] = p1; |
| 196 ComputeBoundingBox(); |
| 197 } |
| 198 |
| 199 // Returns the kind of segment. |
| 200 Kind kind() const { |
| 201 return kind_; |
| 202 } |
| 203 |
| 204 // Returns the i'th control point, 0 <= i < 4. |
| 205 const SkPoint& get_point(int i) { |
| 206 DCHECK(i >= 0 && i < 4); |
| 207 return points_[i]; |
| 208 } |
| 209 |
| 210 // Returns the next segment in the contour. |
| 211 Segment* next() const { return next_; } |
| 212 |
| 213 // Returns the previous segment in the contour. |
| 214 Segment* prev() const { return prev_; } |
| 215 |
| 216 // Sets the next segment in the contour. |
| 217 void set_next(Segment* next) { next_ = next; } |
| 218 |
| 219 // Sets the previous segment in the contour. |
| 220 void set_prev(Segment* prev) { prev_ = prev; } |
| 221 |
| 222 // The contour this segment belongs to. |
| 223 Contour* contour() const { return contour_; } |
| 224 |
| 225 // Subdivides the current segment at the given parameter value (0 <= |
| 226 // t <= 1) and replaces it with the two newly created Segments in |
| 227 // the linked list, if possible. Returns a pointer to the leftmost |
| 228 // Segment. |
| 229 Segment* Subdivide(float param) { |
| 230 SkPoint dst[7]; |
| 231 SkChopCubicAt(points_, dst, param); |
| 232 Segment* left = arena_->Alloc<Segment>(); |
| 233 Segment* right = arena_->Alloc<Segment>(); |
| 234 left->Setup(arena_, contour_, dst[0], dst[1], dst[2], dst[3]); |
| 235 right->Setup(arena_, contour_, dst[3], dst[4], dst[5], dst[6]); |
| 236 left->set_next(right); |
| 237 right->set_prev(left); |
| 238 // Try to set up a link between "this->prev()" and "left". |
| 239 if (prev() != NULL) { |
| 240 left->set_prev(prev()); |
| 241 prev()->set_next(left); |
| 242 } |
| 243 // Try to set up a link between "this->next()" and "right". |
| 244 Segment* n = next(); |
| 245 if (n != NULL) { |
| 246 right->set_next(n); |
| 247 n->set_prev(right); |
| 248 } |
| 249 // Set up a link between "this" and "left"; this is only to |
| 250 // provide a certain amount of continuity during forward iteration. |
| 251 set_next(left); |
| 252 return left; |
| 253 } |
| 254 |
| 255 // Subdivides the current segment at the halfway point and replaces |
| 256 // it with the two newly created Segments in the linked list, if |
| 257 // possible. Returns a pointer to the leftmost Segment. |
| 258 Segment* Subdivide() { |
| 259 return Subdivide(0.5f); |
| 260 } |
| 261 |
| 262 // Returns the bounding box of this segment. |
| 263 const BBox& bbox() const { |
| 264 return bbox_; |
| 265 } |
| 266 |
| 267 // Computes the number of times a query line starting at the given |
| 268 // point and extending to x=+infinity crosses this segment. |
| 269 int NumCrossingsForXRay(SkPoint pt) const { |
| 270 if (kind_ == kCubic) { |
| 271 // Should consider caching the monotonic cubics. |
| 272 return SkNumXRayCrossingsForCubic(pt, points_); |
| 273 } else { |
| 274 return SkXRayCrossesLine(pt, points_) ? 1 : 0; |
| 275 } |
| 276 } |
| 277 |
| 278 // Performs a local triangulation of the control points in this |
| 279 // segment. This operation only makes sense for cubic type segments. |
| 280 void Triangulate(bool compute_inside_edges, |
| 281 CubicTextureCoords::Result* tex_coords); |
| 282 |
| 283 // Returns the number of control point triangles associated with |
| 284 // this segment. |
| 285 int num_triangles() const { |
| 286 if (!triangulator_) |
| 287 return 0; |
| 288 return triangulator_->num_triangles(); |
| 289 } |
| 290 |
| 291 // Fetches the given control point triangle for this segment. |
| 292 LocalTriangulator::Triangle* get_triangle(int index) { |
| 293 DCHECK(triangulator_); |
| 294 return triangulator_->get_triangle(index); |
| 295 } |
| 296 |
| 297 // Number of vertices along the inside edge of this segment. This |
| 298 // can be called either for line or cubic type segments. |
| 299 int num_interior_vertices() const { |
| 300 if (kind_ == kCubic) { |
| 301 if (triangulator_) { |
| 302 return triangulator_->num_interior_vertices(); |
| 303 } else { |
| 304 return 0; |
| 305 } |
| 306 } |
| 307 |
| 308 return 2; |
| 309 } |
| 310 |
| 311 // Returns the given interior vertex, 0 <= index < num_interior_vertices(). |
| 312 SkPoint get_interior_vertex(int index) const { |
| 313 DCHECK(index >= 0 && index < num_interior_vertices()); |
| 314 if (kind_ == kCubic) { |
| 315 SkPoint res; |
| 316 if (triangulator_) { |
| 317 LocalTriangulator::Vertex* vertex = |
| 318 triangulator_->get_interior_vertex(index); |
| 319 if (vertex) |
| 320 res.set(SkFloatToScalar(vertex->x()), |
| 321 SkFloatToScalar(vertex->y())); |
| 322 } |
| 323 return res; |
| 324 } |
| 325 |
| 326 return points_[index]; |
| 327 } |
| 328 |
| 329 // State to assist with curve subdivision. |
| 330 bool marked_for_subdivision() { |
| 331 return marked_for_subdivision_; |
| 332 } |
| 333 |
| 334 // State to assist with curve subdivision. |
| 335 void set_marked_for_subdivision(bool marked_for_subdivision) { |
| 336 marked_for_subdivision_ = marked_for_subdivision; |
| 337 } |
| 338 |
| 339 private: |
| 340 // Computes the bounding box of this Segment. |
| 341 void ComputeBoundingBox() { |
| 342 switch (kind_) { |
| 343 case kCubic: |
| 344 bbox_.Setup(SkScalarToFloat(min4(points_[0].fX, |
| 345 points_[1].fX, |
| 346 points_[2].fX, |
| 347 points_[3].fX)), |
| 348 SkScalarToFloat(min4(points_[0].fY, |
| 349 points_[1].fY, |
| 350 points_[2].fY, |
| 351 points_[3].fY)), |
| 352 SkScalarToFloat(max4(points_[0].fX, |
| 353 points_[1].fX, |
| 354 points_[2].fX, |
| 355 points_[3].fX)), |
| 356 SkScalarToFloat(max4(points_[0].fY, |
| 357 points_[1].fY, |
| 358 points_[2].fY, |
| 359 points_[3].fY))); |
| 360 break; |
| 361 |
| 362 case kLine: |
| 363 bbox_.Setup(SkScalarToFloat(min2(points_[0].fX, |
| 364 points_[1].fX)), |
| 365 SkScalarToFloat(min2(points_[0].fY, |
| 366 points_[1].fY)), |
| 367 SkScalarToFloat(max2(points_[0].fX, |
| 368 points_[1].fX)), |
| 369 SkScalarToFloat(max2(points_[0].fY, |
| 370 points_[1].fY))); |
| 371 break; |
| 372 |
| 373 default: |
| 374 NOTREACHED(); |
| 375 break; |
| 376 } |
| 377 } |
| 378 |
| 379 Arena* arena_; |
| 380 Kind kind_; |
| 381 SkPoint points_[4]; |
| 382 Segment* prev_; |
| 383 Segment* next_; |
| 384 Contour* contour_; |
| 385 BBox bbox_; |
| 386 LocalTriangulator* triangulator_; |
| 387 bool marked_for_subdivision_; |
| 388 |
| 389 DISALLOW_COPY_AND_ASSIGN(Segment); |
| 390 }; |
| 391 |
| 392 //---------------------------------------------------------------------- |
| 393 // Contour |
| 394 // |
| 395 |
| 396 // Describes a closed contour of the path. |
| 397 class Contour { |
| 398 public: |
| 399 Contour() { |
| 400 first_ = &sentinel_; |
| 401 first_->set_next(first_); |
| 402 first_->set_prev(first_); |
| 403 ccw_ = true; |
| 404 fill_right_side_ = true; |
| 405 } |
| 406 |
| 407 // Adds a segment to this contour. |
| 408 void Add(Segment* segment) { |
| 409 if (first_ == &sentinel_) { |
| 410 // First element is the sentinel. Replace it with the incoming |
| 411 // segment. |
| 412 segment->set_next(first_); |
| 413 segment->set_prev(first_); |
| 414 first_->set_next(segment); |
| 415 first_->set_prev(segment); |
| 416 first_ = segment; |
| 417 } else { |
| 418 // first_->prev() is the sentinel. |
| 419 Segment* sentinel = first_->prev(); |
| 420 Segment* last = sentinel->prev(); |
| 421 last->set_next(segment); |
| 422 segment->set_prev(last); |
| 423 segment->set_next(sentinel); |
| 424 sentinel->set_prev(segment); |
| 425 } |
| 426 } |
| 427 |
| 428 // Subdivides the given segment at the given parametric value. |
| 429 // Returns a pointer to the first of the two portions of the |
| 430 // subdivided segment. |
| 431 Segment* Subdivide(Segment* segment, float param) { |
| 432 Segment* left = segment->Subdivide(param); |
| 433 if (first_ == segment) |
| 434 first_ = left; |
| 435 return left; |
| 436 } |
| 437 |
| 438 // Subdivides the given segment at the halfway point. Returns a |
| 439 // pointer to the first of the two portions of the subdivided |
| 440 // segment. |
| 441 Segment* Subdivide(Segment* segment) { |
| 442 Segment* left = segment->Subdivide(); |
| 443 if (first_ == segment) |
| 444 first_ = left; |
| 445 return left; |
| 446 } |
| 447 |
| 448 // Returns the first segment in the contour for iteration. |
| 449 Segment* begin() const { |
| 450 return first_; |
| 451 } |
| 452 |
| 453 // Returns the last segment in the contour for iteration. Callers |
| 454 // should not iterate over this segment. In other words: |
| 455 // for (Segment* cur = contour->begin(); |
| 456 // cur != contour->end(); |
| 457 // cur = cur->next()) { |
| 458 // // .. process cur ... |
| 459 // } |
| 460 Segment* end() const { |
| 461 return first_->prev(); |
| 462 } |
| 463 |
| 464 // Returns whether this contour is oriented counterclockwise. |
| 465 bool ccw() const { |
| 466 return ccw_; |
| 467 } |
| 468 |
| 469 void set_ccw(bool ccw) { |
| 470 ccw_ = ccw; |
| 471 } |
| 472 |
| 473 // Returns whether the right side of this contour is filled. |
| 474 bool fill_right_side() const { return fill_right_side_; } |
| 475 |
| 476 void set_fill_right_side(bool fill_right_side) { |
| 477 fill_right_side_ = fill_right_side; |
| 478 } |
| 479 |
| 480 private: |
| 481 // The start of the segment chain. The segments are kept in a |
| 482 // circular doubly linked list for rapid access to the beginning and |
| 483 // end. |
| 484 Segment* first_; |
| 485 |
| 486 // The sentinel element at the end of the chain, needed for |
| 487 // reasonable iteration semantics. |
| 488 Segment sentinel_; |
| 489 |
| 490 // Whether this contour is oriented counterclockwise. |
| 491 bool ccw_; |
| 492 |
| 493 // Whether we should fill the right (or left) side of this contour. |
| 494 bool fill_right_side_; |
| 495 |
| 496 DISALLOW_COPY_AND_ASSIGN(Contour); |
| 497 }; |
| 498 |
| 499 //---------------------------------------------------------------------- |
| 500 // Segment |
| 501 // |
| 502 |
| 503 // Definition of Segment::Triangulate(), which must come after |
| 504 // declaration of Contour. |
| 505 void Segment::Triangulate(bool compute_inside_edges, |
| 506 CubicTextureCoords::Result* tex_coords) { |
| 507 DCHECK(kind_ == kCubic); |
| 508 if (triangulator_ == NULL) { |
| 509 triangulator_ = arena_->Alloc<LocalTriangulator>(); |
| 510 } |
| 511 triangulator_->Reset(); |
| 512 for (int i = 0; i < 4; i++) { |
| 513 LocalTriangulator::Vertex* vertex = triangulator_->get_vertex(i); |
| 514 if (tex_coords) { |
| 515 vertex->Set(get_point(i).fX, |
| 516 get_point(i).fY, |
| 517 tex_coords->coords[i].getX(), |
| 518 tex_coords->coords[i].getY(), |
| 519 tex_coords->coords[i].getZ()); |
| 520 } else { |
| 521 vertex->Set(get_point(i).fX, |
| 522 get_point(i).fY, |
| 523 // No texture coordinates yet |
| 524 0, 0, 0); |
| 525 } |
| 526 } |
| 527 triangulator_->Triangulate(compute_inside_edges, |
| 528 contour()->fill_right_side()); |
| 529 } |
| 530 |
| 531 //---------------------------------------------------------------------- |
| 532 // PathProcessor |
| 533 // |
| 534 |
| 535 PathProcessor::PathProcessor() |
| 536 : arena_(new Arena()), |
| 537 should_delete_arena_(true), |
| 538 verbose_logging_(false) { |
| 539 } |
| 540 |
| 541 PathProcessor::PathProcessor(Arena* arena) |
| 542 : arena_(arena), |
| 543 should_delete_arena_(false), |
| 544 verbose_logging_(false) { |
| 545 } |
| 546 |
| 547 PathProcessor::~PathProcessor() { |
| 548 if (should_delete_arena_) { |
| 549 delete arena_; |
| 550 } |
| 551 } |
| 552 |
| 553 void PathProcessor::Process(const SkPath& path, PathCache* cache) { |
| 554 BuildContours(path); |
| 555 |
| 556 // Run plane-sweep algorithm to determine overlaps of control point |
| 557 // curves and subdivide curves appropriately. |
| 558 SubdivideCurves(); |
| 559 |
| 560 // Determine orientations of countours. Based on orientation and the |
| 561 // number of curve crossings at a random point on the contour, |
| 562 // determine whether to fill the left or right side of the contour. |
| 563 DetermineSidesToFill(); |
| 564 |
| 565 // Classify curves, compute texture coordinates and subdivide as |
| 566 // necessary to eliminate rendering artifacts. Do the final |
| 567 // triangulation of the curve segments, determining the path along |
| 568 // the interior of the shape. |
| 569 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 570 iter != contours_.end(); |
| 571 iter++) { |
| 572 Contour* cur = *iter; |
| 573 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 574 if (seg->kind() == Segment::kCubic) { |
| 575 CubicClassifier::Result classification = |
| 576 CubicClassifier::Classify(seg->get_point(0).fX, |
| 577 seg->get_point(0).fY, |
| 578 seg->get_point(1).fX, |
| 579 seg->get_point(1).fY, |
| 580 seg->get_point(2).fX, |
| 581 seg->get_point(2).fY, |
| 582 seg->get_point(3).fX, |
| 583 seg->get_point(3).fY); |
| 584 DLOG_IF(INFO, verbose_logging_) << "Classification:" |
| 585 << classification.curve_type(); |
| 586 CubicTextureCoords::Result tex_coords; |
| 587 CubicTextureCoords::Compute(classification, |
| 588 cur->fill_right_side(), |
| 589 &tex_coords); |
| 590 if (tex_coords.has_rendering_artifact) { |
| 591 // TODO(kbr): split at the subdivision parameter value |
| 592 cur->Subdivide(seg); |
| 593 // Next iteration will handle the newly subdivided halves |
| 594 } else { |
| 595 if (!tex_coords.is_line_or_point) { |
| 596 seg->Triangulate(true, &tex_coords); |
| 597 for (int i = 0; i < seg->num_triangles(); i++) { |
| 598 LocalTriangulator::Triangle* triangle = |
| 599 seg->get_triangle(i); |
| 600 for (int j = 0; j < 3; j++) { |
| 601 LocalTriangulator::Vertex* vert = |
| 602 triangle->get_vertex(j); |
| 603 cache->AddVertex(vert->x(), |
| 604 vert->y(), |
| 605 vert->k(), |
| 606 vert->l(), |
| 607 vert->m()); |
| 608 } |
| 609 } |
| 610 #ifdef O3D_CORE_CROSS_GPU2D_PATH_CACHE_DEBUG_INTERIOR_EDGES |
| 611 // Show the end user the interior edges as well |
| 612 for (int i = 1; i < seg->num_interior_vertices(); i++) { |
| 613 SkPoint vert = seg->get_interior_vertex(i); |
| 614 // Duplicate previous vertex to be able to draw GL_LINES |
| 615 SkPoint prev = seg->get_interior_vertex(i - 1); |
| 616 cache->AddInteriorEdgeVertex(prev.fX, prev.fY); |
| 617 cache->AddInteriorEdgeVertex(vert.fX, vert.fY); |
| 618 } |
| 619 #endif // O3D_CORE_CROSS_GPU2D_PATH_CACHE_DEBUG_INTERIOR_EDGES |
| 620 } |
| 621 } |
| 622 } |
| 623 } |
| 624 } |
| 625 |
| 626 // Run the interior paths through a tessellation algorithm |
| 627 // supporting multiple contours. |
| 628 TessellateInterior(cache); |
| 629 } |
| 630 |
| 631 void PathProcessor::BuildContours(const SkPath& path) { |
| 632 // Clear out the contours |
| 633 contours_.clear(); |
| 634 SkPath::Iter iter(path, false); |
| 635 SkPoint pts[4]; |
| 636 SkPath::Verb verb; |
| 637 Contour* contour = NULL; |
| 638 SkPoint cur_pt; |
| 639 SkPoint move_to_pt; |
| 640 do { |
| 641 verb = iter.next(pts); |
| 642 if (verb != SkPath::kMove_Verb) { |
| 643 if (contour == NULL) { |
| 644 contour = arena_->Alloc<Contour>(); |
| 645 contours_.push_back(contour); |
| 646 } |
| 647 } |
| 648 switch (verb) { |
| 649 case SkPath::kMove_Verb: { |
| 650 contour = arena_->Alloc<Contour>(); |
| 651 contours_.push_back(contour); |
| 652 cur_pt = pts[0]; |
| 653 move_to_pt = pts[0]; |
| 654 DLOG_IF(INFO, verbose_logging_) << "MoveTo (" << pts[0].fX |
| 655 << ", " << pts[0].fY << ")"; |
| 656 break; |
| 657 } |
| 658 case SkPath::kLine_Verb: { |
| 659 Segment* segment = arena_->Alloc<Segment>(); |
| 660 if (iter.isCloseLine()) { |
| 661 segment->Setup(arena_, contour, cur_pt, pts[1]); |
| 662 DLOG_IF(INFO, verbose_logging_) << "CloseLineTo (" << cur_pt.fX |
| 663 << ", " << cur_pt.fY |
| 664 << "), (" << pts[1].fX |
| 665 << ", " << pts[1].fY << ")"; |
| 666 contour->Add(segment); |
| 667 contour = NULL; |
| 668 } else { |
| 669 segment->Setup(arena_, contour, pts[0], pts[1]); |
| 670 DLOG_IF(INFO, verbose_logging_) << "LineTo (" << pts[0].fX |
| 671 << ", " << pts[0].fY |
| 672 << "), (" << pts[1].fX |
| 673 << ", " << pts[1].fY << ")"; |
| 674 contour->Add(segment); |
| 675 cur_pt = pts[1]; |
| 676 } |
| 677 break; |
| 678 } |
| 679 case SkPath::kQuad_Verb: { |
| 680 // Need to degree elevate the quadratic into a cubic |
| 681 SkPoint cubic[4]; |
| 682 SkConvertQuadToCubic(pts, cubic); |
| 683 Segment* segment = arena_->Alloc<Segment>(); |
| 684 segment->Setup(arena_, contour, |
| 685 cubic[0], cubic[1], cubic[2], cubic[3]); |
| 686 DLOG_IF(INFO, verbose_logging_) << "Quad->CubicTo (" << cubic[0].fX |
| 687 << ", " << cubic[0].fY |
| 688 << "), (" << cubic[1].fX |
| 689 << ", " << cubic[1].fY |
| 690 << "), (" << cubic[2].fX |
| 691 << ", " << cubic[2].fY |
| 692 << "), (" << cubic[3].fX |
| 693 << ", " << cubic[3].fY << ")"; |
| 694 contour->Add(segment); |
| 695 cur_pt = cubic[3]; |
| 696 break; |
| 697 } |
| 698 case SkPath::kCubic_Verb: { |
| 699 Segment* segment = arena_->Alloc<Segment>(); |
| 700 segment->Setup(arena_, contour, pts[0], pts[1], pts[2], pts[3]); |
| 701 DLOG_IF(INFO, verbose_logging_) << "CubicTo (" << pts[0].fX |
| 702 << ", " << pts[0].fY |
| 703 << "), (" << pts[1].fX |
| 704 << ", " << pts[1].fY |
| 705 << "), (" << pts[2].fX |
| 706 << ", " << pts[2].fY |
| 707 << "), (" << pts[3].fX |
| 708 << ", " << pts[3].fY << ")"; |
| 709 contour->Add(segment); |
| 710 cur_pt = pts[3]; |
| 711 break; |
| 712 } |
| 713 case SkPath::kClose_Verb: { |
| 714 Segment* segment = arena_->Alloc<Segment>(); |
| 715 segment->Setup(arena_, contour, cur_pt, move_to_pt); |
| 716 DLOG_IF(INFO, verbose_logging_) << "Close (" << cur_pt.fX |
| 717 << ", " << cur_pt.fY |
| 718 << ") -> (" << move_to_pt.fX |
| 719 << ", " << move_to_pt.fY << ")"; |
| 720 contour->Add(segment); |
| 721 contour = NULL; |
| 722 } |
| 723 default: |
| 724 break; |
| 725 } |
| 726 } while (verb != SkPath::kDone_Verb); |
| 727 } |
| 728 |
| 729 std::vector<Segment*> PathProcessor::AllSegmentsOverlappingY(float y) { |
| 730 std::vector<Segment*> res; |
| 731 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 732 iter != contours_.end(); |
| 733 iter++) { |
| 734 Contour* cur = *iter; |
| 735 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 736 const BBox& bbox = seg->bbox(); |
| 737 if (bbox.min_y() <= y && y <= bbox.max_y()) { |
| 738 res.push_back(seg); |
| 739 } |
| 740 } |
| 741 } |
| 742 return res; |
| 743 } |
| 744 |
| 745 // Uncomment this to debug the orientation computation |
| 746 // #define O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION |
| 747 |
| 748 void PathProcessor::DetermineSidesToFill() { |
| 749 // Loop and Blinn's algorithm can only easily emulate the even/odd |
| 750 // fill rule, and only for non-intersecting curves. We can determine |
| 751 // which side of each curve segment to fill based on its |
| 752 // clockwise/counterclockwise orientation and how many other |
| 753 // contours surround it. |
| 754 |
| 755 // To optimize the query of all curve segments intersecting a |
| 756 // horizontal line going to x=+infinity, we build up an interval |
| 757 // tree whose keys are the y extents of the segments. |
| 758 IntervalTree<float, Segment*> tree(arena_); |
| 759 typedef IntervalTree<float, Segment*>::IntervalType IntervalType; |
| 760 |
| 761 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 762 iter != contours_.end(); |
| 763 iter++) { |
| 764 Contour* cur = *iter; |
| 765 DetermineOrientation(cur); |
| 766 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 767 const BBox& bbox = seg->bbox(); |
| 768 tree.Insert(tree.MakeInterval(bbox.min_y(), bbox.max_y(), seg)); |
| 769 } |
| 770 } |
| 771 |
| 772 // Now iterate through the contours and pick a random segment (in |
| 773 // this case we use the first) and a random point on that segment. |
| 774 // Find all segments from other contours which intersect this one |
| 775 // and count the number of crossings a horizontal line to |
| 776 // x=+infinity makes with those contours. This combined with the |
| 777 // orientation of the curve tells us which side to fill -- again, |
| 778 // assuming an even/odd fill rule, which is all we can easily |
| 779 // handle. |
| 780 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 781 iter != contours_.end(); |
| 782 iter++) { |
| 783 Contour* cur = *iter; |
| 784 Segment* seg = cur->begin(); |
| 785 |
| 786 int num_crossings = 0; |
| 787 |
| 788 // We use a zero-sized vertical interval for the query |
| 789 std::vector<IntervalType> overlaps = |
| 790 tree.AllOverlaps(IntervalType(SkScalarToFloat(seg->get_point(0).fY), |
| 791 SkScalarToFloat(seg->get_point(0).fY), |
| 792 NULL)); |
| 793 #if !defined(O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION) |
| 794 for (std::vector<IntervalType>::iterator iter = overlaps.begin(); |
| 795 iter != overlaps.end(); |
| 796 iter++) { |
| 797 const IntervalType& interval = *iter; |
| 798 Segment* query_seg = interval.data(); |
| 799 // Ignore segments coming from the same contour |
| 800 if (query_seg->contour() != cur) { |
| 801 num_crossings += query_seg->NumCrossingsForXRay(seg->get_point(0)); |
| 802 } |
| 803 } |
| 804 #endif // !defined(O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION) |
| 805 |
| 806 #ifdef O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION |
| 807 // For debugging |
| 808 std::vector<Segment*> slow_overlaps = |
| 809 AllSegmentsOverlappingY(seg->get_point(0).fY); |
| 810 DCHECK(overlaps.size() == slow_overlaps.size()); |
| 811 for (std::vector<Segment*>::iterator iter = slow_overlaps.begin(); |
| 812 iter != slow_overlaps.end(); |
| 813 iter++) { |
| 814 Segment* query_seg = *iter; |
| 815 // Ignore segments coming from the same contour |
| 816 if (query_seg->contour() != cur) { |
| 817 num_crossings += query_seg->NumCrossingsForXRay(seg->get_point(0)); |
| 818 } |
| 819 } |
| 820 #endif // O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION |
| 821 |
| 822 if (cur->ccw()) { |
| 823 if (num_crossings & 1) { |
| 824 cur->set_fill_right_side(true); |
| 825 } else { |
| 826 cur->set_fill_right_side(false); |
| 827 } |
| 828 } else { |
| 829 if (num_crossings & 1) { |
| 830 cur->set_fill_right_side(false); |
| 831 } else { |
| 832 cur->set_fill_right_side(true); |
| 833 } |
| 834 } |
| 835 } |
| 836 } |
| 837 |
| 838 void PathProcessor::DetermineOrientation(Contour* contour) { |
| 839 // Determine signed area of the polygon represented by the points |
| 840 // along the segments. Consider this an approximation to the true |
| 841 // orientation of the polygon; it probably won't handle |
| 842 // self-intersecting curves correctly. |
| 843 // |
| 844 // There is also a pretty basic assumption here that the contour is |
| 845 // closed. |
| 846 float signed_area = 0; |
| 847 for (Segment* seg = contour->begin(); |
| 848 seg != contour->end(); |
| 849 seg = seg->next()) { |
| 850 int limit = (seg->kind() == Segment::kCubic) ? 4 : 2; |
| 851 for (int i = 1; i < limit; i++) { |
| 852 const SkPoint& prev_point = seg->get_point(i - 1); |
| 853 const SkPoint& point = seg->get_point(i); |
| 854 float cur_area = prev_point.fX * point.fY - prev_point.fY * point.fX; |
| 855 DLOG_IF(INFO, verbose_logging_) << "Adding to signed area (" |
| 856 << prev_point.fX << ", " |
| 857 << prev_point.fY << ") -> (" |
| 858 << point.fX << ", " |
| 859 << point.fY << ") = " |
| 860 << cur_area; |
| 861 signed_area += cur_area; |
| 862 } |
| 863 } |
| 864 |
| 865 if (signed_area > 0) |
| 866 contour->set_ccw(true); |
| 867 else |
| 868 contour->set_ccw(false); |
| 869 } |
| 870 |
| 871 //---------------------------------------------------------------------- |
| 872 // Classes and typedefs needed for curve subdivision. |
| 873 // Unfortunately it appears we can't scope these within the |
| 874 // SubdivideCurves() method itself, because templates then fail to |
| 875 // instantiate. |
| 876 |
| 877 // The user data which is placed in the IntervalTree. |
| 878 struct SweepData { |
| 879 SweepData() |
| 880 : triangle(NULL), |
| 881 segment(NULL) { |
| 882 } |
| 883 |
| 884 // The triangle this interval is associated with |
| 885 LocalTriangulator::Triangle* triangle; |
| 886 // The segment the triangle is associated with |
| 887 Segment* segment; |
| 888 }; |
| 889 |
| 890 typedef IntervalTree<float, SweepData*> SweepTree; |
| 891 typedef SweepTree::IntervalType SweepInterval; |
| 892 |
| 893 // The entry / exit events which occur at the minimum and maximum x |
| 894 // coordinates of the control point triangles' bounding boxes. |
| 895 // |
| 896 // Note that this class requires its copy constructor and assignment |
| 897 // operator since it needs to be stored in a std::vector. |
| 898 class SweepEvent { |
| 899 public: |
| 900 SweepEvent() |
| 901 : x_(0), |
| 902 entry_(false), |
| 903 interval_(0, 0, NULL) { |
| 904 } |
| 905 |
| 906 // Initializes the SweepEvent. |
| 907 void Setup(float x, bool entry, SweepInterval interval) { |
| 908 x_ = x; |
| 909 entry_ = entry; |
| 910 interval_ = interval; |
| 911 } |
| 912 |
| 913 float x() const { return x_; } |
| 914 bool entry() const { return entry_; } |
| 915 const SweepInterval& interval() const { return interval_; } |
| 916 |
| 917 bool operator<(const SweepEvent& other) const { |
| 918 return x_ < other.x_; |
| 919 } |
| 920 |
| 921 private: |
| 922 float x_; |
| 923 bool entry_; |
| 924 SweepInterval interval_; |
| 925 }; |
| 926 |
| 927 namespace { |
| 928 |
| 929 bool TrianglesOverlap(LocalTriangulator::Triangle* t0, |
| 930 LocalTriangulator::Triangle* t1) { |
| 931 LocalTriangulator::Vertex* t0v0 = t0->get_vertex(0); |
| 932 LocalTriangulator::Vertex* t0v1 = t0->get_vertex(1); |
| 933 LocalTriangulator::Vertex* t0v2 = t0->get_vertex(2); |
| 934 LocalTriangulator::Vertex* t1v0 = t1->get_vertex(0); |
| 935 LocalTriangulator::Vertex* t1v1 = t1->get_vertex(1); |
| 936 LocalTriangulator::Vertex* t1v2 = t1->get_vertex(2); |
| 937 return cubic::TrianglesOverlap(t0v0->x(), t0v0->y(), |
| 938 t0v1->x(), t0v1->y(), |
| 939 t0v2->x(), t0v2->y(), |
| 940 t1v0->x(), t1v0->y(), |
| 941 t1v1->x(), t1v1->y(), |
| 942 t1v2->x(), t1v2->y()); |
| 943 } |
| 944 |
| 945 } // anonymous namespace |
| 946 |
| 947 void PathProcessor::SubdivideCurves() { |
| 948 // We need to determine all overlaps of all control point triangles |
| 949 // (from different segments, not the same segment) and, if any |
| 950 // exist, subdivide the associated curves. |
| 951 // |
| 952 // The plane-sweep algorithm determines all overlaps of a set of |
| 953 // rectangles in the 2D plane. Our problem maps very well to this |
| 954 // algorithm and significantly reduces the complexity compared to a |
| 955 // naive implementation. |
| 956 // |
| 957 // Each bounding box of a control point triangle is converted into |
| 958 // an "entry" event at its smallest X coordinate and an "exit" event |
| 959 // at its largest X coordinate. Each event has an associated |
| 960 // one-dimensional interval representing the Y span of the bounding |
| 961 // box. We sort these events by increasing X coordinate. We then |
| 962 // iterate through them. For each entry event we add the interval to |
| 963 // a side interval tree, and query this tree for overlapping |
| 964 // intervals. Any overlapping interval corresponds to an overlapping |
| 965 // bounding box. For each exit event we remove the associated |
| 966 // interval from the interval tree. |
| 967 |
| 968 std::vector<Segment*> cur_segments; |
| 969 std::vector<Segment*> next_segments; |
| 970 |
| 971 // Start things off by considering all of the segments |
| 972 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 973 iter != contours_.end(); |
| 974 iter++) { |
| 975 Contour* cur = *iter; |
| 976 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 977 if (seg->kind() == Segment::kCubic) { |
| 978 seg->Triangulate(false, NULL); |
| 979 cur_segments.push_back(seg); |
| 980 } |
| 981 } |
| 982 } |
| 983 |
| 984 // Subdivide curves at most this many times |
| 985 const int kMaxIter = 5; |
| 986 std::vector<SweepInterval> overlaps; |
| 987 |
| 988 for (int cur_iter = 0; cur_iter < kMaxIter; ++cur_iter) { |
| 989 if (cur_segments.size() == 0) { |
| 990 // Done |
| 991 break; |
| 992 } |
| 993 |
| 994 std::vector<SweepEvent> events; |
| 995 SweepTree tree(arena_); |
| 996 for (std::vector<Segment*>::iterator iter = cur_segments.begin(); |
| 997 iter != cur_segments.end(); |
| 998 iter++) { |
| 999 Segment* seg = *iter; |
| 1000 if (seg->kind() == Segment::kCubic) { |
| 1001 for (int i = 0; i < seg->num_triangles(); i++) { |
| 1002 LocalTriangulator::Triangle* triangle = seg->get_triangle(i); |
| 1003 BBox bbox; |
| 1004 bbox.Setup(triangle); |
| 1005 // Ignore zero-width triangles to avoid issues with |
| 1006 // coincident entry and exit events for the same triangle |
| 1007 if (bbox.max_x() > bbox.min_x()) { |
| 1008 SweepData* data = arena_->Alloc<SweepData>(); |
| 1009 data->triangle = triangle; |
| 1010 data->segment = seg; |
| 1011 SweepInterval interval = |
| 1012 tree.MakeInterval(bbox.min_y(), bbox.max_y(), data); |
| 1013 // Add entry and exit events |
| 1014 SweepEvent event; |
| 1015 event.Setup(bbox.min_x(), true, interval); |
| 1016 events.push_back(event); |
| 1017 event.Setup(bbox.max_x(), false, interval); |
| 1018 events.push_back(event); |
| 1019 } |
| 1020 } |
| 1021 } |
| 1022 } |
| 1023 |
| 1024 // Sort events by increasing X coordinate |
| 1025 std::sort(events.begin(), events.end()); |
| 1026 |
| 1027 // Now iterate through the events |
| 1028 for (std::vector<SweepEvent>::iterator iter = events.begin(); |
| 1029 iter != events.end(); |
| 1030 iter++) { |
| 1031 SweepEvent event = *iter; |
| 1032 if (event.entry()) { |
| 1033 // Add this interval into the tree |
| 1034 tree.Insert(event.interval()); |
| 1035 // See whether the associated segment has been subdivided yet |
| 1036 if (!event.interval().data()->segment->marked_for_subdivision()) { |
| 1037 // Query the tree |
| 1038 overlaps.clear(); |
| 1039 tree.AllOverlaps(event.interval(), overlaps); |
| 1040 // Now see exactly which triangles overlap this one |
| 1041 for (std::vector<SweepInterval>::iterator iter = overlaps.begin(); |
| 1042 iter != overlaps.end(); |
| 1043 iter++) { |
| 1044 SweepInterval overlap = *iter; |
| 1045 // Only pay attention to overlaps from a different Segment |
| 1046 if (event.interval().data()->segment != overlap.data()->segment) { |
| 1047 // See whether the triangles actually overlap |
| 1048 if (TrianglesOverlap(event.interval().data()->triangle, |
| 1049 overlap.data()->triangle)) { |
| 1050 // Actually subdivide the segments. |
| 1051 // Each one might already have been subdivided. |
| 1052 Segment* seg = event.interval().data()->segment; |
| 1053 ConditionallySubdivide(seg, &next_segments); |
| 1054 seg = overlap.data()->segment; |
| 1055 ConditionallySubdivide(seg, &next_segments); |
| 1056 } |
| 1057 } |
| 1058 } |
| 1059 } |
| 1060 } else { |
| 1061 // Remove this interval from the tree |
| 1062 tree.Delete(event.interval()); |
| 1063 } |
| 1064 } |
| 1065 |
| 1066 cur_segments = next_segments; |
| 1067 next_segments.clear(); |
| 1068 } |
| 1069 } |
| 1070 |
| 1071 void PathProcessor::ConditionallySubdivide( |
| 1072 Segment* seg, std::vector<Segment*>* next_segments) { |
| 1073 if (!seg->marked_for_subdivision()) { |
| 1074 seg->set_marked_for_subdivision(true); |
| 1075 Segment* next = seg->contour()->Subdivide(seg); |
| 1076 // Triangulate the newly subdivided segments. |
| 1077 next->Triangulate(false, NULL); |
| 1078 next->next()->Triangulate(false, NULL); |
| 1079 // Add them for the next iteration. |
| 1080 next_segments->push_back(next); |
| 1081 next_segments->push_back(next->next()); |
| 1082 } |
| 1083 } |
| 1084 |
| 1085 void PathProcessor::SubdivideCurvesSlow() { |
| 1086 // Alternate, significantly slower algorithm for curve subdivision |
| 1087 // for use in debugging. |
| 1088 std::vector<Segment*> cur_segments; |
| 1089 std::vector<Segment*> next_segments; |
| 1090 |
| 1091 // Start things off by considering all of the segments |
| 1092 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 1093 iter != contours_.end(); |
| 1094 iter++) { |
| 1095 Contour* cur = *iter; |
| 1096 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 1097 if (seg->kind() == Segment::kCubic) { |
| 1098 seg->Triangulate(false, NULL); |
| 1099 cur_segments.push_back(seg); |
| 1100 } |
| 1101 } |
| 1102 } |
| 1103 |
| 1104 // Subdivide curves at most this many times |
| 1105 const int kMaxIter = 5; |
| 1106 |
| 1107 for (int cur_iter = 0; cur_iter < kMaxIter; ++cur_iter) { |
| 1108 if (cur_segments.size() == 0) { |
| 1109 // Done |
| 1110 break; |
| 1111 } |
| 1112 |
| 1113 for (std::vector<Segment*>::iterator iter = cur_segments.begin(); |
| 1114 iter != cur_segments.end(); |
| 1115 iter++) { |
| 1116 Segment* seg = *iter; |
| 1117 if (seg->kind() == Segment::kCubic) { |
| 1118 for (std::vector<Segment*>::iterator iter2 = cur_segments.begin(); |
| 1119 iter2 != cur_segments.end(); |
| 1120 iter2++) { |
| 1121 Segment* seg2 = *iter2; |
| 1122 if (seg2->kind() == Segment::kCubic && |
| 1123 seg != seg2) { |
| 1124 for (int i = 0; i < seg->num_triangles(); i++) { |
| 1125 LocalTriangulator::Triangle* triangle = seg->get_triangle(i); |
| 1126 for (int j = 0; j < seg2->num_triangles(); j++) { |
| 1127 LocalTriangulator::Triangle* triangle2 = seg2->get_triangle(j); |
| 1128 if (TrianglesOverlap(triangle, triangle2)) { |
| 1129 ConditionallySubdivide(seg, &next_segments); |
| 1130 ConditionallySubdivide(seg2, &next_segments); |
| 1131 } |
| 1132 } |
| 1133 } |
| 1134 } |
| 1135 } |
| 1136 } |
| 1137 } |
| 1138 |
| 1139 cur_segments = next_segments; |
| 1140 next_segments.clear(); |
| 1141 } |
| 1142 } |
| 1143 |
| 1144 //---------------------------------------------------------------------- |
| 1145 // Structures and callbacks for tessellation of the interior region of |
| 1146 // the contours. |
| 1147 |
| 1148 // The user data for the GLU tessellator. |
| 1149 struct TessellationState { |
| 1150 PathCache* cache; |
| 1151 std::vector<void*> allocated_pointers; |
| 1152 }; |
| 1153 |
| 1154 namespace { |
| 1155 |
| 1156 void VertexCallback(void* vertex_data, void* data) { |
| 1157 TessellationState* state = static_cast<TessellationState*>(data); |
| 1158 PathCache* cache = state->cache; |
| 1159 GLdouble* location = static_cast<GLdouble*>(vertex_data); |
| 1160 cache->AddInteriorVertex(static_cast<float>(location[0]), |
| 1161 static_cast<float>(location[1])); |
| 1162 } |
| 1163 |
| 1164 void CombineCallback(GLdouble coords[3], void* vertex_data[4], |
| 1165 GLfloat weight[4], void** out_data, |
| 1166 void* polygon_data) { |
| 1167 TessellationState* state = static_cast<TessellationState*>(polygon_data); |
| 1168 GLdouble* out_vertex = static_cast<GLdouble*>(malloc(3 * sizeof(GLdouble))); |
| 1169 state->allocated_pointers.push_back(out_vertex); |
| 1170 out_vertex[0] = coords[0]; |
| 1171 out_vertex[1] = coords[1]; |
| 1172 out_vertex[2] = coords[2]; |
| 1173 *out_data = out_vertex; |
| 1174 } |
| 1175 |
| 1176 void EdgeFlagCallback(GLboolean flag) { |
| 1177 // No-op just to prevent triangle strips and fans from being passed |
| 1178 // to us |
| 1179 } |
| 1180 |
| 1181 } // anonymous namespace |
| 1182 |
| 1183 void PathProcessor::TessellateInterior(PathCache* cache) { |
| 1184 // Because the GLU tessellator requires its input in |
| 1185 // double-precision format, we need to make a separate copy of the |
| 1186 // data. |
| 1187 std::vector<GLdouble> vertex_data; |
| 1188 std::vector<size_t> contour_endings; |
| 1189 // For avoiding adding coincident vertices. |
| 1190 float cur_x = 0, cur_y = 0; |
| 1191 for (std::vector<Contour*>::iterator iter = contours_.begin(); |
| 1192 iter != contours_.end(); |
| 1193 iter++) { |
| 1194 Contour* cur = *iter; |
| 1195 bool first = true; |
| 1196 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) { |
| 1197 int num_interior_vertices = seg->num_interior_vertices(); |
| 1198 for (int i = 0; i < num_interior_vertices - 1; i++) { |
| 1199 SkPoint point = seg->get_interior_vertex(i); |
| 1200 if (first) { |
| 1201 first = false; |
| 1202 vertex_data.push_back(point.fX); |
| 1203 vertex_data.push_back(point.fY); |
| 1204 vertex_data.push_back(0); |
| 1205 cur_x = point.fX; |
| 1206 cur_y = point.fY; |
| 1207 } else if (point.fX != cur_x || point.fY != cur_y) { |
| 1208 vertex_data.push_back(point.fX); |
| 1209 vertex_data.push_back(point.fY); |
| 1210 vertex_data.push_back(0); |
| 1211 cur_x = point.fX; |
| 1212 cur_y = point.fY; |
| 1213 } |
| 1214 } |
| 1215 } |
| 1216 contour_endings.push_back(vertex_data.size()); |
| 1217 } |
| 1218 // Now that we have all of the vertex data in a stable location in |
| 1219 // memory, call the tessellator. |
| 1220 GLUtesselator* tess = internal_gluNewTess(); |
| 1221 TessellationState state; |
| 1222 state.cache = cache; |
| 1223 internal_gluTessCallback(tess, GLU_TESS_VERTEX_DATA, |
| 1224 reinterpret_cast<GLvoid (*)()>(VertexCallback)); |
| 1225 internal_gluTessCallback(tess, GLU_TESS_COMBINE_DATA, |
| 1226 reinterpret_cast<GLvoid (*)()>(CombineCallback)); |
| 1227 internal_gluTessCallback(tess, GLU_TESS_EDGE_FLAG, |
| 1228 reinterpret_cast<GLvoid (*)()>(EdgeFlagCallback)); |
| 1229 internal_gluTessBeginPolygon(tess, &state); |
| 1230 internal_gluTessBeginContour(tess); |
| 1231 GLdouble* base = &vertex_data.front(); |
| 1232 int contour_idx = 0; |
| 1233 for (size_t i = 0; i < vertex_data.size(); i += 3) { |
| 1234 if (i == contour_endings[contour_idx]) { |
| 1235 internal_gluTessEndContour(tess); |
| 1236 internal_gluTessBeginContour(tess); |
| 1237 ++contour_idx; |
| 1238 } |
| 1239 internal_gluTessVertex(tess, &base[i], &base[i]); |
| 1240 } |
| 1241 internal_gluTessEndContour(tess); |
| 1242 internal_gluTessEndPolygon(tess); |
| 1243 for (size_t i = 0; i < state.allocated_pointers.size(); i++) { |
| 1244 free(state.allocated_pointers[i]); |
| 1245 } |
| 1246 internal_gluDeleteTess(tess); |
| 1247 } |
| 1248 |
| 1249 } // namespace gpu2d |
| 1250 } // namespace o3d |
| 1251 |
OLD | NEW |