1 /*
2  * Copyright (C) 2012 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #define LOG_NDEBUG 1
17 
18 #define VERTEX_DEBUG 0
19 
20 #if VERTEX_DEBUG
21 #define DEBUG_DUMP_ALPHA_BUFFER() \
22     for (unsigned int i = 0; i < vertexBuffer.getSize(); i++) { \
23         ALOGD("point %d at %f %f, alpha %f", \
24         i, buffer[i].x, buffer[i].y, buffer[i].alpha); \
25     }
26 #define DEBUG_DUMP_BUFFER() \
27     for (unsigned int i = 0; i < vertexBuffer.getSize(); i++) { \
28         ALOGD("point %d at %f %f", i, buffer[i].x, buffer[i].y); \
29     }
30 #else
31 #define DEBUG_DUMP_ALPHA_BUFFER()
32 #define DEBUG_DUMP_BUFFER()
33 #endif
34 
35 #include "PathTessellator.h"
36 
37 #include "Matrix.h"
38 #include "Vector.h"
39 #include "Vertex.h"
40 #include "utils/MathUtils.h"
41 
42 #include <algorithm>
43 
44 #include <SkPath.h>
45 #include <SkPaint.h>
46 #include <SkPoint.h>
47 #include <SkGeometry.h> // WARNING: Internal Skia Header
48 
49 #include <stdlib.h>
50 #include <stdint.h>
51 #include <sys/types.h>
52 
53 #include <utils/Log.h>
54 #include <utils/Trace.h>
55 
56 namespace android {
57 namespace uirenderer {
58 
59 #define OUTLINE_REFINE_THRESHOLD 0.5f
60 #define ROUND_CAP_THRESH 0.25f
61 #define PI 3.1415926535897932f
62 #define MAX_DEPTH 15
63 
64 /**
65  * Extracts the x and y scale from the transform as positive values, and clamps them
66  */
extractTessellationScales(const Matrix4 & transform,float * scaleX,float * scaleY)67 void PathTessellator::extractTessellationScales(const Matrix4& transform,
68         float* scaleX, float* scaleY) {
69     if (CC_LIKELY(transform.isPureTranslate())) {
70         *scaleX = 1.0f;
71         *scaleY = 1.0f;
72     } else {
73         float m00 = transform.data[Matrix4::kScaleX];
74         float m01 = transform.data[Matrix4::kSkewY];
75         float m10 = transform.data[Matrix4::kSkewX];
76         float m11 = transform.data[Matrix4::kScaleY];
77         *scaleX = MathUtils::clampTessellationScale(sqrt(m00 * m00 + m01 * m01));
78         *scaleY = MathUtils::clampTessellationScale(sqrt(m10 * m10 + m11 * m11));
79     }
80 }
81 
82 /**
83  * Produces a pseudo-normal for a vertex, given the normals of the two incoming lines. If the offset
84  * from each vertex in a perimeter is calculated, the resultant lines connecting the offset vertices
85  * will be offset by 1.0
86  *
87  * Note that we can't add and normalize the two vectors, that would result in a rectangle having an
88  * offset of (sqrt(2)/2, sqrt(2)/2) at each corner, instead of (1, 1)
89  *
90  * NOTE: assumes angles between normals 90 degrees or less
91  */
totalOffsetFromNormals(const Vector2 & normalA,const Vector2 & normalB)92 inline static Vector2 totalOffsetFromNormals(const Vector2& normalA, const Vector2& normalB) {
93     return (normalA + normalB) / (1 + fabs(normalA.dot(normalB)));
94 }
95 
96 /**
97  * Structure used for storing useful information about the SkPaint and scale used for tessellating
98  */
99 struct PaintInfo {
100 public:
PaintInfoandroid::uirenderer::PaintInfo101     PaintInfo(const SkPaint* paint, const mat4& transform) :
102             style(paint->getStyle()), cap(paint->getStrokeCap()), isAA(paint->isAntiAlias()),
103             halfStrokeWidth(paint->getStrokeWidth() * 0.5f), maxAlpha(1.0f) {
104         // compute inverse scales
105         if (CC_LIKELY(transform.isPureTranslate())) {
106             inverseScaleX = 1.0f;
107             inverseScaleY = 1.0f;
108         } else {
109             float scaleX, scaleY;
110             PathTessellator::extractTessellationScales(transform, &scaleX, &scaleY);
111             inverseScaleX = 1.0f / scaleX;
112             inverseScaleY = 1.0f / scaleY;
113         }
114 
115         if (isAA && halfStrokeWidth != 0 && inverseScaleX == inverseScaleY &&
116                 2 * halfStrokeWidth < inverseScaleX) {
117             // AA, with non-hairline stroke, width < 1 pixel. Scale alpha and treat as hairline.
118             maxAlpha *= (2 * halfStrokeWidth) / inverseScaleX;
119             halfStrokeWidth = 0.0f;
120         }
121     }
122 
123     SkPaint::Style style;
124     SkPaint::Cap cap;
125     bool isAA;
126     float inverseScaleX;
127     float inverseScaleY;
128     float halfStrokeWidth;
129     float maxAlpha;
130 
scaleOffsetForStrokeWidthandroid::uirenderer::PaintInfo131     inline void scaleOffsetForStrokeWidth(Vector2& offset) const {
132         if (halfStrokeWidth == 0.0f) {
133             // hairline - compensate for scale
134             offset.x *= 0.5f * inverseScaleX;
135             offset.y *= 0.5f * inverseScaleY;
136         } else {
137             offset *= halfStrokeWidth;
138         }
139     }
140 
141     /**
142      * NOTE: the input will not always be a normal, especially for sharp edges - it should be the
143      * result of totalOffsetFromNormals (see documentation there)
144      */
deriveAAOffsetandroid::uirenderer::PaintInfo145     inline Vector2 deriveAAOffset(const Vector2& offset) const {
146         return (Vector2){offset.x * 0.5f * inverseScaleX, offset.y * 0.5f * inverseScaleY};
147     }
148 
149     /**
150      * Returns the number of cap divisions beyond the minimum 2 (kButt_Cap/kSquareCap will return 0)
151      * Should only be used when stroking and drawing caps
152      */
capExtraDivisionsandroid::uirenderer::PaintInfo153     inline int capExtraDivisions() const {
154         if (cap == SkPaint::kRound_Cap) {
155             // always use 2 points for hairline
156             if (halfStrokeWidth == 0.0f) return 2;
157 
158             float threshold = std::min(inverseScaleX, inverseScaleY) * ROUND_CAP_THRESH;
159             return MathUtils::divisionsNeededToApproximateArc(halfStrokeWidth, PI, threshold);
160         }
161         return 0;
162     }
163 
164     /**
165      * Outset the bounds of point data (for line endpoints or points) to account for stroke
166      * geometry.
167      *
168      * bounds are in pre-scaled space.
169      */
expandBoundsForStrokeandroid::uirenderer::PaintInfo170     void expandBoundsForStroke(Rect* bounds) const {
171         if (halfStrokeWidth == 0) {
172             // hairline, outset by (0.5f + fudge factor) in post-scaling space
173             bounds->outset(fabs(inverseScaleX) * (0.5f + Vertex::GeometryFudgeFactor()),
174                     fabs(inverseScaleY) * (0.5f + Vertex::GeometryFudgeFactor()));
175         } else {
176             // non hairline, outset by half stroke width pre-scaled, and fudge factor post scaled
177             bounds->outset(halfStrokeWidth + fabs(inverseScaleX) * Vertex::GeometryFudgeFactor(),
178                     halfStrokeWidth + fabs(inverseScaleY) * Vertex::GeometryFudgeFactor());
179         }
180     }
181 };
182 
getFillVerticesFromPerimeter(const std::vector<Vertex> & perimeter,VertexBuffer & vertexBuffer)183 void getFillVerticesFromPerimeter(const std::vector<Vertex>& perimeter,
184         VertexBuffer& vertexBuffer) {
185     Vertex* buffer = vertexBuffer.alloc<Vertex>(perimeter.size());
186 
187     int currentIndex = 0;
188     // zig zag between all previous points on the inside of the hull to create a
189     // triangle strip that fills the hull
190     int srcAindex = 0;
191     int srcBindex = perimeter.size() - 1;
192     while (srcAindex <= srcBindex) {
193         buffer[currentIndex++] = perimeter[srcAindex];
194         if (srcAindex == srcBindex) break;
195         buffer[currentIndex++] = perimeter[srcBindex];
196         srcAindex++;
197         srcBindex--;
198     }
199 }
200 
201 /*
202  * Fills a vertexBuffer with non-alpha vertices, zig-zagging at each perimeter point to create a
203  * tri-strip as wide as the stroke.
204  *
205  * Uses an additional 2 vertices at the end to wrap around, closing the tri-strip
206  * (for a total of perimeter.size() * 2 + 2 vertices)
207  */
getStrokeVerticesFromPerimeter(const PaintInfo & paintInfo,const std::vector<Vertex> & perimeter,VertexBuffer & vertexBuffer)208 void getStrokeVerticesFromPerimeter(const PaintInfo& paintInfo,
209         const std::vector<Vertex>& perimeter, VertexBuffer& vertexBuffer) {
210     Vertex* buffer = vertexBuffer.alloc<Vertex>(perimeter.size() * 2 + 2);
211 
212     int currentIndex = 0;
213     const Vertex* last = &(perimeter[perimeter.size() - 1]);
214     const Vertex* current = &(perimeter[0]);
215     Vector2 lastNormal = {current->y - last->y, last->x - current->x};
216     lastNormal.normalize();
217     for (unsigned int i = 0; i < perimeter.size(); i++) {
218         const Vertex* next = &(perimeter[i + 1 >= perimeter.size() ? 0 : i + 1]);
219         Vector2 nextNormal = {next->y - current->y, current->x - next->x};
220         nextNormal.normalize();
221 
222         Vector2 totalOffset = totalOffsetFromNormals(lastNormal, nextNormal);
223         paintInfo.scaleOffsetForStrokeWidth(totalOffset);
224 
225         Vertex::set(&buffer[currentIndex++],
226                 current->x + totalOffset.x,
227                 current->y + totalOffset.y);
228 
229         Vertex::set(&buffer[currentIndex++],
230                 current->x - totalOffset.x,
231                 current->y - totalOffset.y);
232 
233         current = next;
234         lastNormal = nextNormal;
235     }
236 
237     // wrap around to beginning
238     buffer[currentIndex++] = buffer[0];
239     buffer[currentIndex++] = buffer[1];
240 
241     DEBUG_DUMP_BUFFER();
242 }
243 
storeBeginEnd(const PaintInfo & paintInfo,const Vertex & center,const Vector2 & normal,Vertex * buffer,int & currentIndex,bool begin)244 static inline void storeBeginEnd(const PaintInfo& paintInfo, const Vertex& center,
245         const Vector2& normal, Vertex* buffer, int& currentIndex, bool begin) {
246     Vector2 strokeOffset = normal;
247     paintInfo.scaleOffsetForStrokeWidth(strokeOffset);
248 
249     Vector2 referencePoint = {center.x, center.y};
250     if (paintInfo.cap == SkPaint::kSquare_Cap) {
251         Vector2 rotated = {-strokeOffset.y, strokeOffset.x};
252         referencePoint += rotated * (begin ? -1 : 1);
253     }
254 
255     Vertex::set(&buffer[currentIndex++], referencePoint + strokeOffset);
256     Vertex::set(&buffer[currentIndex++], referencePoint - strokeOffset);
257 }
258 
259 /**
260  * Fills a vertexBuffer with non-alpha vertices similar to getStrokeVerticesFromPerimeter, except:
261  *
262  * 1 - Doesn't need to wrap around, since the input vertices are unclosed
263  *
264  * 2 - can zig-zag across 'extra' vertices at either end, to create round caps
265  */
getStrokeVerticesFromUnclosedVertices(const PaintInfo & paintInfo,const std::vector<Vertex> & vertices,VertexBuffer & vertexBuffer)266 void getStrokeVerticesFromUnclosedVertices(const PaintInfo& paintInfo,
267         const std::vector<Vertex>& vertices, VertexBuffer& vertexBuffer) {
268     const int extra = paintInfo.capExtraDivisions();
269     const int allocSize = (vertices.size() + extra) * 2;
270     Vertex* buffer = vertexBuffer.alloc<Vertex>(allocSize);
271 
272     const int lastIndex = vertices.size() - 1;
273     if (extra > 0) {
274         // tessellate both round caps
275         float beginTheta = atan2(
276                     - (vertices[0].x - vertices[1].x),
277                     vertices[0].y - vertices[1].y);
278         float endTheta = atan2(
279                     - (vertices[lastIndex].x - vertices[lastIndex - 1].x),
280                     vertices[lastIndex].y - vertices[lastIndex - 1].y);
281         const float dTheta = PI / (extra + 1);
282 
283         int capOffset;
284         for (int i = 0; i < extra; i++) {
285             if (i < extra / 2) {
286                 capOffset = extra - 2 * i - 1;
287             } else {
288                 capOffset = 2 * i - extra;
289             }
290 
291             beginTheta += dTheta;
292             Vector2 beginRadialOffset = {cosf(beginTheta), sinf(beginTheta)};
293             paintInfo.scaleOffsetForStrokeWidth(beginRadialOffset);
294             Vertex::set(&buffer[capOffset],
295                     vertices[0].x + beginRadialOffset.x,
296                     vertices[0].y + beginRadialOffset.y);
297 
298             endTheta += dTheta;
299             Vector2 endRadialOffset = {cosf(endTheta), sinf(endTheta)};
300             paintInfo.scaleOffsetForStrokeWidth(endRadialOffset);
301             Vertex::set(&buffer[allocSize - 1 - capOffset],
302                     vertices[lastIndex].x + endRadialOffset.x,
303                     vertices[lastIndex].y + endRadialOffset.y);
304         }
305     }
306 
307     int currentIndex = extra;
308     const Vertex* last = &(vertices[0]);
309     const Vertex* current = &(vertices[1]);
310     Vector2 lastNormal = {current->y - last->y, last->x - current->x};
311     lastNormal.normalize();
312 
313     storeBeginEnd(paintInfo, vertices[0], lastNormal, buffer, currentIndex, true);
314 
315     for (unsigned int i = 1; i < vertices.size() - 1; i++) {
316         const Vertex* next = &(vertices[i + 1]);
317         Vector2 nextNormal = {next->y - current->y, current->x - next->x};
318         nextNormal.normalize();
319 
320         Vector2 strokeOffset  = totalOffsetFromNormals(lastNormal, nextNormal);
321         paintInfo.scaleOffsetForStrokeWidth(strokeOffset);
322 
323         Vector2 center = {current->x, current->y};
324         Vertex::set(&buffer[currentIndex++], center + strokeOffset);
325         Vertex::set(&buffer[currentIndex++], center - strokeOffset);
326 
327         current = next;
328         lastNormal = nextNormal;
329     }
330 
331     storeBeginEnd(paintInfo, vertices[lastIndex], lastNormal, buffer, currentIndex, false);
332 
333     DEBUG_DUMP_BUFFER();
334 }
335 
336 /**
337  * Populates a vertexBuffer with AlphaVertices to create an anti-aliased fill shape tessellation
338  *
339  * 1 - create the AA perimeter of unit width, by zig-zagging at each point around the perimeter of
340  * the shape (using 2 * perimeter.size() vertices)
341  *
342  * 2 - wrap around to the beginning to complete the perimeter (2 vertices)
343  *
344  * 3 - zig zag back and forth inside the shape to fill it (using perimeter.size() vertices)
345  */
getFillVerticesFromPerimeterAA(const PaintInfo & paintInfo,const std::vector<Vertex> & perimeter,VertexBuffer & vertexBuffer,float maxAlpha=1.0f)346 void getFillVerticesFromPerimeterAA(const PaintInfo& paintInfo,
347         const std::vector<Vertex>& perimeter, VertexBuffer& vertexBuffer,
348         float maxAlpha = 1.0f) {
349     AlphaVertex* buffer = vertexBuffer.alloc<AlphaVertex>(perimeter.size() * 3 + 2);
350 
351     // generate alpha points - fill Alpha vertex gaps in between each point with
352     // alpha 0 vertex, offset by a scaled normal.
353     int currentIndex = 0;
354     const Vertex* last = &(perimeter[perimeter.size() - 1]);
355     const Vertex* current = &(perimeter[0]);
356     Vector2 lastNormal = {current->y - last->y, last->x - current->x};
357     lastNormal.normalize();
358     for (unsigned int i = 0; i < perimeter.size(); i++) {
359         const Vertex* next = &(perimeter[i + 1 >= perimeter.size() ? 0 : i + 1]);
360         Vector2 nextNormal = {next->y - current->y, current->x - next->x};
361         nextNormal.normalize();
362 
363         // AA point offset from original point is that point's normal, such that each side is offset
364         // by .5 pixels
365         Vector2 totalOffset = paintInfo.deriveAAOffset(totalOffsetFromNormals(lastNormal, nextNormal));
366 
367         AlphaVertex::set(&buffer[currentIndex++],
368                 current->x + totalOffset.x,
369                 current->y + totalOffset.y,
370                 0.0f);
371         AlphaVertex::set(&buffer[currentIndex++],
372                 current->x - totalOffset.x,
373                 current->y - totalOffset.y,
374                 maxAlpha);
375 
376         current = next;
377         lastNormal = nextNormal;
378     }
379 
380     // wrap around to beginning
381     buffer[currentIndex++] = buffer[0];
382     buffer[currentIndex++] = buffer[1];
383 
384     // zig zag between all previous points on the inside of the hull to create a
385     // triangle strip that fills the hull, repeating the first inner point to
386     // create degenerate tris to start inside path
387     int srcAindex = 0;
388     int srcBindex = perimeter.size() - 1;
389     while (srcAindex <= srcBindex) {
390         buffer[currentIndex++] = buffer[srcAindex * 2 + 1];
391         if (srcAindex == srcBindex) break;
392         buffer[currentIndex++] = buffer[srcBindex * 2 + 1];
393         srcAindex++;
394         srcBindex--;
395     }
396 
397     DEBUG_DUMP_BUFFER();
398 }
399 
400 /**
401  * Stores geometry for a single, AA-perimeter (potentially rounded) cap
402  *
403  * For explanation of constants and general methodoloyg, see comments for
404  * getStrokeVerticesFromUnclosedVerticesAA() below.
405  */
storeCapAA(const PaintInfo & paintInfo,const std::vector<Vertex> & vertices,AlphaVertex * buffer,bool isFirst,Vector2 normal,int offset)406 inline static void storeCapAA(const PaintInfo& paintInfo, const std::vector<Vertex>& vertices,
407         AlphaVertex* buffer, bool isFirst, Vector2 normal, int offset) {
408     const int extra = paintInfo.capExtraDivisions();
409     const int extraOffset = (extra + 1) / 2;
410     const int capIndex = isFirst
411             ? 2 * offset + 6 + 2 * (extra + extraOffset)
412             : offset + 2 + 2 * extraOffset;
413     if (isFirst) normal *= -1;
414 
415     // TODO: this normal should be scaled by radialScale if extra != 0, see totalOffsetFromNormals()
416     Vector2 AAOffset = paintInfo.deriveAAOffset(normal);
417 
418     Vector2 strokeOffset = normal;
419     paintInfo.scaleOffsetForStrokeWidth(strokeOffset);
420     Vector2 outerOffset = strokeOffset + AAOffset;
421     Vector2 innerOffset = strokeOffset - AAOffset;
422 
423     Vector2 capAAOffset = {0, 0};
424     if (paintInfo.cap != SkPaint::kRound_Cap) {
425         // if the cap is square or butt, the inside primary cap vertices will be inset in two
426         // directions - both normal to the stroke, and parallel to it.
427         capAAOffset = (Vector2){-AAOffset.y, AAOffset.x};
428     }
429 
430     // determine referencePoint, the center point for the 4 primary cap vertices
431     const Vertex& point = isFirst ? vertices.front() : vertices.back();
432     Vector2 referencePoint = {point.x, point.y};
433     if (paintInfo.cap == SkPaint::kSquare_Cap) {
434         // To account for square cap, move the primary cap vertices (that create the AA edge) by the
435         // stroke offset vector (rotated to be parallel to the stroke)
436         Vector2 rotated = {-strokeOffset.y, strokeOffset.x};
437         referencePoint += rotated;
438     }
439 
440     AlphaVertex::set(&buffer[capIndex + 0],
441             referencePoint.x + outerOffset.x + capAAOffset.x,
442             referencePoint.y + outerOffset.y + capAAOffset.y,
443             0.0f);
444     AlphaVertex::set(&buffer[capIndex + 1],
445             referencePoint.x + innerOffset.x - capAAOffset.x,
446             referencePoint.y + innerOffset.y - capAAOffset.y,
447             paintInfo.maxAlpha);
448 
449     bool isRound = paintInfo.cap == SkPaint::kRound_Cap;
450 
451     const int postCapIndex = (isRound && isFirst) ? (2 * extraOffset - 2) : capIndex + (2 * extra);
452     AlphaVertex::set(&buffer[postCapIndex + 2],
453             referencePoint.x - outerOffset.x + capAAOffset.x,
454             referencePoint.y - outerOffset.y + capAAOffset.y,
455             0.0f);
456     AlphaVertex::set(&buffer[postCapIndex + 3],
457             referencePoint.x - innerOffset.x - capAAOffset.x,
458             referencePoint.y - innerOffset.y - capAAOffset.y,
459             paintInfo.maxAlpha);
460 
461     if (isRound) {
462         const float dTheta = PI / (extra + 1);
463         const float radialScale = 2.0f / (1 + cos(dTheta));
464         float theta = atan2(normal.y, normal.x);
465         int capPerimIndex = capIndex + 2;
466 
467         for (int i = 0; i < extra; i++) {
468             theta += dTheta;
469 
470             Vector2 radialOffset = {cosf(theta), sinf(theta)};
471 
472             // scale to compensate for pinching at sharp angles, see totalOffsetFromNormals()
473             radialOffset *= radialScale;
474 
475             AAOffset = paintInfo.deriveAAOffset(radialOffset);
476             paintInfo.scaleOffsetForStrokeWidth(radialOffset);
477             AlphaVertex::set(&buffer[capPerimIndex++],
478                     referencePoint.x + radialOffset.x + AAOffset.x,
479                     referencePoint.y + radialOffset.y + AAOffset.y,
480                     0.0f);
481             AlphaVertex::set(&buffer[capPerimIndex++],
482                     referencePoint.x + radialOffset.x - AAOffset.x,
483                     referencePoint.y + radialOffset.y - AAOffset.y,
484                     paintInfo.maxAlpha);
485 
486             if (isFirst && i == extra - extraOffset) {
487                 //copy most recent two points to first two points
488                 buffer[0] = buffer[capPerimIndex - 2];
489                 buffer[1] = buffer[capPerimIndex - 1];
490 
491                 capPerimIndex = 2; // start writing the rest of the round cap at index 2
492             }
493         }
494 
495         if (isFirst) {
496             const int startCapFillIndex = capIndex + 2 * (extra - extraOffset) + 4;
497             int capFillIndex = startCapFillIndex;
498             for (int i = 0; i < extra + 2; i += 2) {
499                 buffer[capFillIndex++] = buffer[1 + i];
500                 // TODO: to support odd numbers of divisions, break here on the last iteration
501                 buffer[capFillIndex++] = buffer[startCapFillIndex - 3 - i];
502             }
503         } else {
504             int capFillIndex = 6 * vertices.size() + 2 + 6 * extra - (extra + 2);
505             for (int i = 0; i < extra + 2; i += 2) {
506                 buffer[capFillIndex++] = buffer[capIndex + 1 + i];
507                 // TODO: to support odd numbers of divisions, break here on the last iteration
508                 buffer[capFillIndex++] = buffer[capIndex + 3 + 2 * extra - i];
509             }
510         }
511         return;
512     }
513     if (isFirst) {
514         buffer[0] = buffer[postCapIndex + 2];
515         buffer[1] = buffer[postCapIndex + 3];
516         buffer[postCapIndex + 4] = buffer[1]; // degenerate tris (the only two!)
517         buffer[postCapIndex + 5] = buffer[postCapIndex + 1];
518     } else {
519         buffer[6 * vertices.size()] = buffer[postCapIndex + 1];
520         buffer[6 * vertices.size() + 1] = buffer[postCapIndex + 3];
521     }
522 }
523 
524 /*
525 the geometry for an aa, capped stroke consists of the following:
526 
527        # vertices       |    function
528 ----------------------------------------------------------------------
529 a) 2                    | Start AA perimeter
530 b) 2, 2 * roundDivOff   | First half of begin cap's perimeter
531                         |
532    2 * middlePts        | 'Outer' or 'Top' AA perimeter half (between caps)
533                         |
534 a) 4                    | End cap's
535 b) 2, 2 * roundDivs, 2  |    AA perimeter
536                         |
537    2 * middlePts        | 'Inner' or 'bottom' AA perimeter half
538                         |
539 a) 6                    | Begin cap's perimeter
540 b) 2, 2*(rD - rDO + 1), | Last half of begin cap's perimeter
541        roundDivs, 2     |
542                         |
543    2 * middlePts        | Stroke's full opacity center strip
544                         |
545 a) 2                    | end stroke
546 b) 2, roundDivs         |    (and end cap fill, for round)
547 
548 Notes:
549 * rows starting with 'a)' denote the Butt or Square cap vertex use, 'b)' denote Round
550 
551 * 'middlePts' is (number of points in the unclosed input vertex list, minus 2) times two
552 
553 * 'roundDivs' or 'rD' is the number of extra vertices (beyond the minimum of 2) that define the
554         round cap's shape, and is at least two. This will increase with cap size to sufficiently
555         define the cap's level of tessellation.
556 
557 * 'roundDivOffset' or 'rDO' is the point about halfway along the start cap's round perimeter, where
558         the stream of vertices for the AA perimeter starts. By starting and ending the perimeter at
559         this offset, the fill of the stroke is drawn from this point with minimal extra vertices.
560 
561 This means the outer perimeter starts at:
562     outerIndex = (2) OR (2 + 2 * roundDivOff)
563 the inner perimeter (since it is filled in reverse) starts at:
564     innerIndex = outerIndex + (4 * middlePts) + ((4) OR (4 + 2 * roundDivs)) - 1
565 the stroke starts at:
566     strokeIndex = innerIndex + 1 + ((6) OR (6 + 3 * roundDivs - 2 * roundDivOffset))
567 
568 The total needed allocated space is either:
569     2 + 4 + 6 + 2 + 3 * (2 * middlePts) = 14 + 6 * middlePts = 2 + 6 * pts
570 or, for rounded caps:
571     (2 + 2 * rDO) + (4 + 2 * rD) + (2 * (rD - rDO + 1)
572             + roundDivs + 4) + (2 + roundDivs) + 3 * (2 * middlePts)
573     = 14 + 6 * middlePts + 6 * roundDivs
574     = 2 + 6 * pts + 6 * roundDivs
575  */
getStrokeVerticesFromUnclosedVerticesAA(const PaintInfo & paintInfo,const std::vector<Vertex> & vertices,VertexBuffer & vertexBuffer)576 void getStrokeVerticesFromUnclosedVerticesAA(const PaintInfo& paintInfo,
577         const std::vector<Vertex>& vertices, VertexBuffer& vertexBuffer) {
578 
579     const int extra = paintInfo.capExtraDivisions();
580     const int allocSize = 6 * vertices.size() + 2 + 6 * extra;
581 
582     AlphaVertex* buffer = vertexBuffer.alloc<AlphaVertex>(allocSize);
583 
584     const int extraOffset = (extra + 1) / 2;
585     int offset = 2 * (vertices.size() - 2);
586     // there is no outer/inner here, using them for consistency with below approach
587     int currentAAOuterIndex = 2 + 2 * extraOffset;
588     int currentAAInnerIndex = currentAAOuterIndex + (2 * offset) + 3 + (2 * extra);
589     int currentStrokeIndex = currentAAInnerIndex + 7 + (3 * extra - 2 * extraOffset);
590 
591     const Vertex* last = &(vertices[0]);
592     const Vertex* current = &(vertices[1]);
593     Vector2 lastNormal = {current->y - last->y, last->x - current->x};
594     lastNormal.normalize();
595 
596     // TODO: use normal from bezier traversal for cap, instead of from vertices
597     storeCapAA(paintInfo, vertices, buffer, true, lastNormal, offset);
598 
599     for (unsigned int i = 1; i < vertices.size() - 1; i++) {
600         const Vertex* next = &(vertices[i + 1]);
601         Vector2 nextNormal = {next->y - current->y, current->x - next->x};
602         nextNormal.normalize();
603 
604         Vector2 totalOffset = totalOffsetFromNormals(lastNormal, nextNormal);
605         Vector2 AAOffset = paintInfo.deriveAAOffset(totalOffset);
606 
607         Vector2 innerOffset = totalOffset;
608         paintInfo.scaleOffsetForStrokeWidth(innerOffset);
609         Vector2 outerOffset = innerOffset + AAOffset;
610         innerOffset -= AAOffset;
611 
612         AlphaVertex::set(&buffer[currentAAOuterIndex++],
613                 current->x + outerOffset.x,
614                 current->y + outerOffset.y,
615                 0.0f);
616         AlphaVertex::set(&buffer[currentAAOuterIndex++],
617                 current->x + innerOffset.x,
618                 current->y + innerOffset.y,
619                 paintInfo.maxAlpha);
620 
621         AlphaVertex::set(&buffer[currentStrokeIndex++],
622                 current->x + innerOffset.x,
623                 current->y + innerOffset.y,
624                 paintInfo.maxAlpha);
625         AlphaVertex::set(&buffer[currentStrokeIndex++],
626                 current->x - innerOffset.x,
627                 current->y - innerOffset.y,
628                 paintInfo.maxAlpha);
629 
630         AlphaVertex::set(&buffer[currentAAInnerIndex--],
631                 current->x - innerOffset.x,
632                 current->y - innerOffset.y,
633                 paintInfo.maxAlpha);
634         AlphaVertex::set(&buffer[currentAAInnerIndex--],
635                 current->x - outerOffset.x,
636                 current->y - outerOffset.y,
637                 0.0f);
638 
639         current = next;
640         lastNormal = nextNormal;
641     }
642 
643     // TODO: use normal from bezier traversal for cap, instead of from vertices
644     storeCapAA(paintInfo, vertices, buffer, false, lastNormal, offset);
645 
646     DEBUG_DUMP_ALPHA_BUFFER();
647 }
648 
649 
getStrokeVerticesFromPerimeterAA(const PaintInfo & paintInfo,const std::vector<Vertex> & perimeter,VertexBuffer & vertexBuffer)650 void getStrokeVerticesFromPerimeterAA(const PaintInfo& paintInfo,
651         const std::vector<Vertex>& perimeter, VertexBuffer& vertexBuffer) {
652     AlphaVertex* buffer = vertexBuffer.alloc<AlphaVertex>(6 * perimeter.size() + 8);
653 
654     int offset = 2 * perimeter.size() + 3;
655     int currentAAOuterIndex = 0;
656     int currentStrokeIndex = offset;
657     int currentAAInnerIndex = offset * 2;
658 
659     const Vertex* last = &(perimeter[perimeter.size() - 1]);
660     const Vertex* current = &(perimeter[0]);
661     Vector2 lastNormal = {current->y - last->y, last->x - current->x};
662     lastNormal.normalize();
663     for (unsigned int i = 0; i < perimeter.size(); i++) {
664         const Vertex* next = &(perimeter[i + 1 >= perimeter.size() ? 0 : i + 1]);
665         Vector2 nextNormal = {next->y - current->y, current->x - next->x};
666         nextNormal.normalize();
667 
668         Vector2 totalOffset = totalOffsetFromNormals(lastNormal, nextNormal);
669         Vector2 AAOffset = paintInfo.deriveAAOffset(totalOffset);
670 
671         Vector2 innerOffset = totalOffset;
672         paintInfo.scaleOffsetForStrokeWidth(innerOffset);
673         Vector2 outerOffset = innerOffset + AAOffset;
674         innerOffset -= AAOffset;
675 
676         AlphaVertex::set(&buffer[currentAAOuterIndex++],
677                 current->x + outerOffset.x,
678                 current->y + outerOffset.y,
679                 0.0f);
680         AlphaVertex::set(&buffer[currentAAOuterIndex++],
681                 current->x + innerOffset.x,
682                 current->y + innerOffset.y,
683                 paintInfo.maxAlpha);
684 
685         AlphaVertex::set(&buffer[currentStrokeIndex++],
686                 current->x + innerOffset.x,
687                 current->y + innerOffset.y,
688                 paintInfo.maxAlpha);
689         AlphaVertex::set(&buffer[currentStrokeIndex++],
690                 current->x - innerOffset.x,
691                 current->y - innerOffset.y,
692                 paintInfo.maxAlpha);
693 
694         AlphaVertex::set(&buffer[currentAAInnerIndex++],
695                 current->x - innerOffset.x,
696                 current->y - innerOffset.y,
697                 paintInfo.maxAlpha);
698         AlphaVertex::set(&buffer[currentAAInnerIndex++],
699                 current->x - outerOffset.x,
700                 current->y - outerOffset.y,
701                 0.0f);
702 
703         current = next;
704         lastNormal = nextNormal;
705     }
706 
707     // wrap each strip around to beginning, creating degenerate tris to bridge strips
708     buffer[currentAAOuterIndex++] = buffer[0];
709     buffer[currentAAOuterIndex++] = buffer[1];
710     buffer[currentAAOuterIndex++] = buffer[1];
711 
712     buffer[currentStrokeIndex++] = buffer[offset];
713     buffer[currentStrokeIndex++] = buffer[offset + 1];
714     buffer[currentStrokeIndex++] = buffer[offset + 1];
715 
716     buffer[currentAAInnerIndex++] = buffer[2 * offset];
717     buffer[currentAAInnerIndex++] = buffer[2 * offset + 1];
718     // don't need to create last degenerate tri
719 
720     DEBUG_DUMP_ALPHA_BUFFER();
721 }
722 
tessellatePath(const SkPath & path,const SkPaint * paint,const mat4 & transform,VertexBuffer & vertexBuffer)723 void PathTessellator::tessellatePath(const SkPath &path, const SkPaint* paint,
724         const mat4& transform, VertexBuffer& vertexBuffer) {
725     ATRACE_CALL();
726 
727     const PaintInfo paintInfo(paint, transform);
728 
729     std::vector<Vertex> tempVertices;
730     float threshInvScaleX = paintInfo.inverseScaleX;
731     float threshInvScaleY = paintInfo.inverseScaleY;
732     if (paintInfo.style == SkPaint::kStroke_Style) {
733         // alter the bezier recursion threshold values we calculate in order to compensate for
734         // expansion done after the path vertices are found
735         SkRect bounds = path.getBounds();
736         if (!bounds.isEmpty()) {
737             threshInvScaleX *= bounds.width() / (bounds.width() + paint->getStrokeWidth());
738             threshInvScaleY *= bounds.height() / (bounds.height() + paint->getStrokeWidth());
739         }
740     }
741 
742     // force close if we're filling the path, since fill path expects closed perimeter.
743     bool forceClose = paintInfo.style != SkPaint::kStroke_Style;
744     PathApproximationInfo approximationInfo(threshInvScaleX, threshInvScaleY,
745             OUTLINE_REFINE_THRESHOLD);
746     bool wasClosed = approximatePathOutlineVertices(path, forceClose,
747             approximationInfo, tempVertices);
748 
749     if (!tempVertices.size()) {
750         // path was empty, return without allocating vertex buffer
751         return;
752     }
753 
754 #if VERTEX_DEBUG
755     for (unsigned int i = 0; i < tempVertices.size(); i++) {
756         ALOGD("orig path: point at %f %f",
757                 tempVertices[i].x, tempVertices[i].y);
758     }
759 #endif
760 
761     if (paintInfo.style == SkPaint::kStroke_Style) {
762         if (!paintInfo.isAA) {
763             if (wasClosed) {
764                 getStrokeVerticesFromPerimeter(paintInfo, tempVertices, vertexBuffer);
765             } else {
766                 getStrokeVerticesFromUnclosedVertices(paintInfo, tempVertices, vertexBuffer);
767             }
768 
769         } else {
770             if (wasClosed) {
771                 getStrokeVerticesFromPerimeterAA(paintInfo, tempVertices, vertexBuffer);
772             } else {
773                 getStrokeVerticesFromUnclosedVerticesAA(paintInfo, tempVertices, vertexBuffer);
774             }
775         }
776     } else {
777         // For kStrokeAndFill style, the path should be adjusted externally.
778         // It will be treated as a fill here.
779         if (!paintInfo.isAA) {
780             getFillVerticesFromPerimeter(tempVertices, vertexBuffer);
781         } else {
782             getFillVerticesFromPerimeterAA(paintInfo, tempVertices, vertexBuffer);
783         }
784     }
785 
786     Rect bounds(path.getBounds());
787     paintInfo.expandBoundsForStroke(&bounds);
788     vertexBuffer.setBounds(bounds);
789     vertexBuffer.setMeshFeatureFlags(paintInfo.isAA ? VertexBuffer::kAlpha : VertexBuffer::kNone);
790 }
791 
792 template <class TYPE>
instanceVertices(VertexBuffer & srcBuffer,VertexBuffer & dstBuffer,const float * points,int count,Rect & bounds)793 static void instanceVertices(VertexBuffer& srcBuffer, VertexBuffer& dstBuffer,
794         const float* points, int count, Rect& bounds) {
795     bounds.set(points[0], points[1], points[0], points[1]);
796 
797     int numPoints = count / 2;
798     int verticesPerPoint = srcBuffer.getVertexCount();
799     dstBuffer.alloc<TYPE>(numPoints * verticesPerPoint + (numPoints - 1) * 2);
800 
801     for (int i = 0; i < count; i += 2) {
802         bounds.expandToCover(points[i + 0], points[i + 1]);
803         dstBuffer.copyInto<TYPE>(srcBuffer, points[i + 0], points[i + 1]);
804     }
805     dstBuffer.createDegenerateSeparators<TYPE>(verticesPerPoint);
806 }
807 
tessellatePoints(const float * points,int count,const SkPaint * paint,const mat4 & transform,VertexBuffer & vertexBuffer)808 void PathTessellator::tessellatePoints(const float* points, int count, const SkPaint* paint,
809         const mat4& transform, VertexBuffer& vertexBuffer) {
810     const PaintInfo paintInfo(paint, transform);
811 
812     // determine point shape
813     SkPath path;
814     float radius = paintInfo.halfStrokeWidth;
815     if (radius == 0.0f) radius = 0.5f;
816 
817     if (paintInfo.cap == SkPaint::kRound_Cap) {
818         path.addCircle(0, 0, radius);
819     } else {
820         path.addRect(-radius, -radius, radius, radius);
821     }
822 
823     // calculate outline
824     std::vector<Vertex> outlineVertices;
825     PathApproximationInfo approximationInfo(paintInfo.inverseScaleX, paintInfo.inverseScaleY,
826             OUTLINE_REFINE_THRESHOLD);
827     approximatePathOutlineVertices(path, true, approximationInfo, outlineVertices);
828 
829     if (!outlineVertices.size()) return;
830 
831     Rect bounds;
832     // tessellate, then duplicate outline across points
833     VertexBuffer tempBuffer;
834     if (!paintInfo.isAA) {
835         getFillVerticesFromPerimeter(outlineVertices, tempBuffer);
836         instanceVertices<Vertex>(tempBuffer, vertexBuffer, points, count, bounds);
837     } else {
838         // note: pass maxAlpha directly, since we want fill to be alpha modulated
839         getFillVerticesFromPerimeterAA(paintInfo, outlineVertices, tempBuffer, paintInfo.maxAlpha);
840         instanceVertices<AlphaVertex>(tempBuffer, vertexBuffer, points, count, bounds);
841     }
842 
843     // expand bounds from vertex coords to pixel data
844     paintInfo.expandBoundsForStroke(&bounds);
845     vertexBuffer.setBounds(bounds);
846     vertexBuffer.setMeshFeatureFlags(paintInfo.isAA ? VertexBuffer::kAlpha : VertexBuffer::kNone);
847 }
848 
tessellateLines(const float * points,int count,const SkPaint * paint,const mat4 & transform,VertexBuffer & vertexBuffer)849 void PathTessellator::tessellateLines(const float* points, int count, const SkPaint* paint,
850         const mat4& transform, VertexBuffer& vertexBuffer) {
851     ATRACE_CALL();
852     const PaintInfo paintInfo(paint, transform);
853 
854     const int extra = paintInfo.capExtraDivisions();
855     int numLines = count / 4;
856     int lineAllocSize;
857     // pre-allocate space for lines in the buffer, and degenerate tris in between
858     if (paintInfo.isAA) {
859         lineAllocSize = 6 * (2) + 2 + 6 * extra;
860         vertexBuffer.alloc<AlphaVertex>(numLines * lineAllocSize + (numLines - 1) * 2);
861     } else {
862         lineAllocSize = 2 * ((2) + extra);
863         vertexBuffer.alloc<Vertex>(numLines * lineAllocSize + (numLines - 1) * 2);
864     }
865 
866     std::vector<Vertex> tempVertices(2);
867     Vertex* tempVerticesData = &tempVertices.front();
868     Rect bounds;
869     bounds.set(points[0], points[1], points[0], points[1]);
870     for (int i = 0; i < count; i += 4) {
871         Vertex::set(&(tempVerticesData[0]), points[i + 0], points[i + 1]);
872         Vertex::set(&(tempVerticesData[1]), points[i + 2], points[i + 3]);
873 
874         if (paintInfo.isAA) {
875             getStrokeVerticesFromUnclosedVerticesAA(paintInfo, tempVertices, vertexBuffer);
876         } else {
877             getStrokeVerticesFromUnclosedVertices(paintInfo, tempVertices, vertexBuffer);
878         }
879 
880         // calculate bounds
881         bounds.expandToCover(tempVerticesData[0].x, tempVerticesData[0].y);
882         bounds.expandToCover(tempVerticesData[1].x, tempVerticesData[1].y);
883     }
884 
885     // since multiple objects tessellated into buffer, separate them with degen tris
886     if (paintInfo.isAA) {
887         vertexBuffer.createDegenerateSeparators<AlphaVertex>(lineAllocSize);
888     } else {
889         vertexBuffer.createDegenerateSeparators<Vertex>(lineAllocSize);
890     }
891 
892     // expand bounds from vertex coords to pixel data
893     paintInfo.expandBoundsForStroke(&bounds);
894     vertexBuffer.setBounds(bounds);
895     vertexBuffer.setMeshFeatureFlags(paintInfo.isAA ? VertexBuffer::kAlpha : VertexBuffer::kNone);
896 }
897 
898 ///////////////////////////////////////////////////////////////////////////////
899 // Simple path line approximation
900 ///////////////////////////////////////////////////////////////////////////////
901 
approximatePathOutlineVertices(const SkPath & path,float threshold,std::vector<Vertex> & outputVertices)902 bool PathTessellator::approximatePathOutlineVertices(const SkPath& path, float threshold,
903         std::vector<Vertex>& outputVertices) {
904     PathApproximationInfo approximationInfo(1.0f, 1.0f, threshold);
905     return approximatePathOutlineVertices(path, true, approximationInfo, outputVertices);
906 }
907 
908 class ClockwiseEnforcer {
909 public:
addPoint(const SkPoint & point)910     void addPoint(const SkPoint& point) {
911         double x = point.x();
912         double y = point.y();
913 
914         if (initialized) {
915             sum += (x + lastX) * (y - lastY);
916         } else {
917             initialized = true;
918         }
919 
920         lastX = x;
921         lastY = y;
922     }
reverseVectorIfNotClockwise(std::vector<Vertex> & vertices)923     void reverseVectorIfNotClockwise(std::vector<Vertex>& vertices) {
924         if (sum < 0) {
925             // negative sum implies CounterClockwise
926             const int size = vertices.size();
927             for (int i = 0; i < size / 2; i++) {
928                 Vertex tmp = vertices[i];
929                 int k = size - 1 - i;
930                 vertices[i] = vertices[k];
931                 vertices[k] = tmp;
932             }
933         }
934     }
935 private:
936     bool initialized = false;
937     double lastX = 0;
938     double lastY = 0;
939     double sum = 0;
940 };
941 
approximatePathOutlineVertices(const SkPath & path,bool forceClose,const PathApproximationInfo & approximationInfo,std::vector<Vertex> & outputVertices)942 bool PathTessellator::approximatePathOutlineVertices(const SkPath& path, bool forceClose,
943         const PathApproximationInfo& approximationInfo, std::vector<Vertex>& outputVertices) {
944     ATRACE_CALL();
945 
946     // TODO: to support joins other than sharp miter, join vertices should be labelled in the
947     // perimeter, or resolved into more vertices. Reconsider forceClose-ing in that case.
948     SkPath::Iter iter(path, forceClose);
949     SkPoint pts[4];
950     SkPath::Verb v;
951     ClockwiseEnforcer clockwiseEnforcer;
952     while (SkPath::kDone_Verb != (v = iter.next(pts))) {
953             switch (v) {
954             case SkPath::kMove_Verb:
955                 outputVertices.push_back(Vertex{pts[0].x(), pts[0].y()});
956                 ALOGV("Move to pos %f %f", pts[0].x(), pts[0].y());
957                 clockwiseEnforcer.addPoint(pts[0]);
958                 break;
959             case SkPath::kClose_Verb:
960                 ALOGV("Close at pos %f %f", pts[0].x(), pts[0].y());
961                 clockwiseEnforcer.addPoint(pts[0]);
962                 break;
963             case SkPath::kLine_Verb:
964                 ALOGV("kLine_Verb %f %f -> %f %f", pts[0].x(), pts[0].y(), pts[1].x(), pts[1].y());
965                 outputVertices.push_back(Vertex{pts[1].x(), pts[1].y()});
966                 clockwiseEnforcer.addPoint(pts[1]);
967                 break;
968             case SkPath::kQuad_Verb:
969                 ALOGV("kQuad_Verb");
970                 recursiveQuadraticBezierVertices(
971                         pts[0].x(), pts[0].y(),
972                         pts[2].x(), pts[2].y(),
973                         pts[1].x(), pts[1].y(),
974                         approximationInfo, outputVertices);
975                 clockwiseEnforcer.addPoint(pts[1]);
976                 clockwiseEnforcer.addPoint(pts[2]);
977                 break;
978             case SkPath::kCubic_Verb:
979                 ALOGV("kCubic_Verb");
980                 recursiveCubicBezierVertices(
981                         pts[0].x(), pts[0].y(),
982                         pts[1].x(), pts[1].y(),
983                         pts[3].x(), pts[3].y(),
984                         pts[2].x(), pts[2].y(),
985                         approximationInfo, outputVertices);
986                 clockwiseEnforcer.addPoint(pts[1]);
987                 clockwiseEnforcer.addPoint(pts[2]);
988                 clockwiseEnforcer.addPoint(pts[3]);
989                 break;
990             case SkPath::kConic_Verb: {
991                 ALOGV("kConic_Verb");
992                 SkAutoConicToQuads converter;
993                 const SkPoint* quads = converter.computeQuads(pts, iter.conicWeight(),
994                         approximationInfo.thresholdForConicQuads);
995                 for (int i = 0; i < converter.countQuads(); ++i) {
996                     const int offset = 2 * i;
997                     recursiveQuadraticBezierVertices(
998                             quads[offset].x(), quads[offset].y(),
999                             quads[offset+2].x(), quads[offset+2].y(),
1000                             quads[offset+1].x(), quads[offset+1].y(),
1001                             approximationInfo, outputVertices);
1002                 }
1003                 clockwiseEnforcer.addPoint(pts[1]);
1004                 clockwiseEnforcer.addPoint(pts[2]);
1005                 break;
1006             }
1007             default:
1008                 static_assert(SkPath::kMove_Verb == 0
1009                                 && SkPath::kLine_Verb == 1
1010                                 && SkPath::kQuad_Verb == 2
1011                                 && SkPath::kConic_Verb == 3
1012                                 && SkPath::kCubic_Verb == 4
1013                                 && SkPath::kClose_Verb == 5
1014                                 && SkPath::kDone_Verb == 6,
1015                         "Path enum changed, new types may have been added");
1016                 break;
1017             }
1018     }
1019 
1020     bool wasClosed = false;
1021     int size = outputVertices.size();
1022     if (size >= 2 && outputVertices[0].x == outputVertices[size - 1].x &&
1023             outputVertices[0].y == outputVertices[size - 1].y) {
1024         outputVertices.pop_back();
1025         wasClosed = true;
1026     }
1027 
1028     // ensure output vector is clockwise
1029     clockwiseEnforcer.reverseVectorIfNotClockwise(outputVertices);
1030     return wasClosed;
1031 }
1032 
1033 ///////////////////////////////////////////////////////////////////////////////
1034 // Bezier approximation
1035 //
1036 // All the inputs and outputs here are in path coordinates.
1037 // We convert the error threshold from screen coordinates into path coordinates.
1038 ///////////////////////////////////////////////////////////////////////////////
1039 
1040 // Get a threshold in path coordinates, by scaling the thresholdSquared from screen coordinates.
1041 // TODO: Document the math behind this algorithm.
getThreshold(const PathApproximationInfo & info,float dx,float dy)1042 static inline float getThreshold(const PathApproximationInfo& info, float dx, float dy) {
1043     // multiplying by sqrInvScaleY/X equivalent to multiplying in dimensional scale factors
1044     float scale = (dx * dx * info.sqrInvScaleY + dy * dy * info.sqrInvScaleX);
1045     return info.thresholdSquared * scale;
1046 }
1047 
recursiveCubicBezierVertices(float p1x,float p1y,float c1x,float c1y,float p2x,float p2y,float c2x,float c2y,const PathApproximationInfo & approximationInfo,std::vector<Vertex> & outputVertices,int depth)1048 void PathTessellator::recursiveCubicBezierVertices(
1049         float p1x, float p1y, float c1x, float c1y,
1050         float p2x, float p2y, float c2x, float c2y,
1051         const PathApproximationInfo& approximationInfo,
1052         std::vector<Vertex>& outputVertices, int depth) {
1053     float dx = p2x - p1x;
1054     float dy = p2y - p1y;
1055     float d1 = fabs((c1x - p2x) * dy - (c1y - p2y) * dx);
1056     float d2 = fabs((c2x - p2x) * dy - (c2y - p2y) * dx);
1057     float d = d1 + d2;
1058 
1059     if (depth >= MAX_DEPTH
1060             || d * d <= getThreshold(approximationInfo, dx, dy)) {
1061         // below thresh, draw line by adding endpoint
1062         outputVertices.push_back(Vertex{p2x, p2y});
1063     } else {
1064         float p1c1x = (p1x + c1x) * 0.5f;
1065         float p1c1y = (p1y + c1y) * 0.5f;
1066         float p2c2x = (p2x + c2x) * 0.5f;
1067         float p2c2y = (p2y + c2y) * 0.5f;
1068 
1069         float c1c2x = (c1x + c2x) * 0.5f;
1070         float c1c2y = (c1y + c2y) * 0.5f;
1071 
1072         float p1c1c2x = (p1c1x + c1c2x) * 0.5f;
1073         float p1c1c2y = (p1c1y + c1c2y) * 0.5f;
1074 
1075         float p2c1c2x = (p2c2x + c1c2x) * 0.5f;
1076         float p2c1c2y = (p2c2y + c1c2y) * 0.5f;
1077 
1078         float mx = (p1c1c2x + p2c1c2x) * 0.5f;
1079         float my = (p1c1c2y + p2c1c2y) * 0.5f;
1080 
1081         recursiveCubicBezierVertices(
1082                 p1x, p1y, p1c1x, p1c1y,
1083                 mx, my, p1c1c2x, p1c1c2y,
1084                 approximationInfo, outputVertices, depth + 1);
1085         recursiveCubicBezierVertices(
1086                 mx, my, p2c1c2x, p2c1c2y,
1087                 p2x, p2y, p2c2x, p2c2y,
1088                 approximationInfo, outputVertices, depth + 1);
1089     }
1090 }
1091 
recursiveQuadraticBezierVertices(float ax,float ay,float bx,float by,float cx,float cy,const PathApproximationInfo & approximationInfo,std::vector<Vertex> & outputVertices,int depth)1092 void PathTessellator::recursiveQuadraticBezierVertices(
1093         float ax, float ay,
1094         float bx, float by,
1095         float cx, float cy,
1096         const PathApproximationInfo& approximationInfo,
1097         std::vector<Vertex>& outputVertices, int depth) {
1098     float dx = bx - ax;
1099     float dy = by - ay;
1100     // d is the cross product of vector (B-A) and (C-B).
1101     float d = (cx - bx) * dy - (cy - by) * dx;
1102 
1103     if (depth >= MAX_DEPTH
1104             || d * d <= getThreshold(approximationInfo, dx, dy)) {
1105         // below thresh, draw line by adding endpoint
1106         outputVertices.push_back(Vertex{bx, by});
1107     } else {
1108         float acx = (ax + cx) * 0.5f;
1109         float bcx = (bx + cx) * 0.5f;
1110         float acy = (ay + cy) * 0.5f;
1111         float bcy = (by + cy) * 0.5f;
1112 
1113         // midpoint
1114         float mx = (acx + bcx) * 0.5f;
1115         float my = (acy + bcy) * 0.5f;
1116 
1117         recursiveQuadraticBezierVertices(ax, ay, mx, my, acx, acy,
1118                 approximationInfo, outputVertices, depth + 1);
1119         recursiveQuadraticBezierVertices(mx, my, bx, by, bcx, bcy,
1120                 approximationInfo, outputVertices, depth + 1);
1121     }
1122 }
1123 
1124 }; // namespace uirenderer
1125 }; // namespace android
1126