1
1
/**
2
- * Port from https://github.com/mapbox/earcut (v2.2.2 )
2
+ * Port from https://github.com/mapbox/earcut (v2.2.4 )
3
3
*/
4
4
5
5
const Earcut = {
@@ -36,11 +36,11 @@ const Earcut = {
36
36
37
37
// minX, minY and invSize are later used to transform coords into integers for z-order calculation
38
38
invSize = Math . max ( maxX - minX , maxY - minY ) ;
39
- invSize = invSize !== 0 ? 1 / invSize : 0 ;
39
+ invSize = invSize !== 0 ? 32767 / invSize : 0 ;
40
40
41
41
}
42
42
43
- earcutLinked ( outerNode , triangles , dim , minX , minY , invSize ) ;
43
+ earcutLinked ( outerNode , triangles , dim , minX , minY , invSize , 0 ) ;
44
44
45
45
return triangles ;
46
46
@@ -125,9 +125,9 @@ function earcutLinked( ear, triangles, dim, minX, minY, invSize, pass ) {
125
125
if ( invSize ? isEarHashed ( ear , minX , minY , invSize ) : isEar ( ear ) ) {
126
126
127
127
// cut off the triangle
128
- triangles . push ( prev . i / dim ) ;
129
- triangles . push ( ear . i / dim ) ;
130
- triangles . push ( next . i / dim ) ;
128
+ triangles . push ( prev . i / dim | 0 ) ;
129
+ triangles . push ( ear . i / dim | 0 ) ;
130
+ triangles . push ( next . i / dim | 0 ) ;
131
131
132
132
removeNode ( ear ) ;
133
133
@@ -182,11 +182,19 @@ function isEar( ear ) {
182
182
if ( area ( a , b , c ) >= 0 ) return false ; // reflex, can't be an ear
183
183
184
184
// now make sure we don't have other points inside the potential ear
185
- let p = ear . next . next ;
185
+ const ax = a . x , bx = b . x , cx = c . x , ay = a . y , by = b . y , cy = c . y ;
186
186
187
- while ( p !== ear . prev ) {
187
+ // triangle bbox; min & max are calculated like this for speed
188
+ const x0 = ax < bx ? ( ax < cx ? ax : cx ) : ( bx < cx ? bx : cx ) ,
189
+ y0 = ay < by ? ( ay < cy ? ay : cy ) : ( by < cy ? by : cy ) ,
190
+ x1 = ax > bx ? ( ax > cx ? ax : cx ) : ( bx > cx ? bx : cx ) ,
191
+ y1 = ay > by ? ( ay > cy ? ay : cy ) : ( by > cy ? by : cy ) ;
192
+
193
+ let p = c . next ;
194
+ while ( p !== a ) {
188
195
189
- if ( pointInTriangle ( a . x , a . y , b . x , b . y , c . x , c . y , p . x , p . y ) &&
196
+ if ( p . x >= x0 && p . x <= x1 && p . y >= y0 && p . y <= y1 &&
197
+ pointInTriangle ( ax , ay , bx , by , cx , cy , p . x , p . y ) &&
190
198
area ( p . prev , p , p . next ) >= 0 ) return false ;
191
199
p = p . next ;
192
200
@@ -204,50 +212,48 @@ function isEarHashed( ear, minX, minY, invSize ) {
204
212
205
213
if ( area ( a , b , c ) >= 0 ) return false ; // reflex, can't be an ear
206
214
215
+ const ax = a . x , bx = b . x , cx = c . x , ay = a . y , by = b . y , cy = c . y ;
216
+
207
217
// triangle bbox; min & max are calculated like this for speed
208
- const minTX = a . x < b . x ? ( a . x < c . x ? a . x : c . x ) : ( b . x < c . x ? b . x : c . x ) ,
209
- minTY = a . y < b . y ? ( a . y < c . y ? a . y : c . y ) : ( b . y < c . y ? b . y : c . y ) ,
210
- maxTX = a . x > b . x ? ( a . x > c . x ? a . x : c . x ) : ( b . x > c . x ? b . x : c . x ) ,
211
- maxTY = a . y > b . y ? ( a . y > c . y ? a . y : c . y ) : ( b . y > c . y ? b . y : c . y ) ;
218
+ const x0 = ax < bx ? ( ax < cx ? ax : cx ) : ( bx < cx ? bx : cx ) ,
219
+ y0 = ay < by ? ( ay < cy ? ay : cy ) : ( by < cy ? by : cy ) ,
220
+ x1 = ax > bx ? ( ax > cx ? ax : cx ) : ( bx > cx ? bx : cx ) ,
221
+ y1 = ay > by ? ( ay > cy ? ay : cy ) : ( by > cy ? by : cy ) ;
212
222
213
223
// z-order range for the current triangle bbox;
214
- const minZ = zOrder ( minTX , minTY , minX , minY , invSize ) ,
215
- maxZ = zOrder ( maxTX , maxTY , minX , minY , invSize ) ;
224
+ const minZ = zOrder ( x0 , y0 , minX , minY , invSize ) ,
225
+ maxZ = zOrder ( x1 , y1 , minX , minY , invSize ) ;
216
226
217
227
let p = ear . prevZ ,
218
228
n = ear . nextZ ;
219
229
220
230
// look for points inside the triangle in both directions
221
231
while ( p && p . z >= minZ && n && n . z <= maxZ ) {
222
232
223
- if ( p !== ear . prev && p !== ear . next &&
224
- pointInTriangle ( a . x , a . y , b . x , b . y , c . x , c . y , p . x , p . y ) &&
225
- area ( p . prev , p , p . next ) >= 0 ) return false ;
233
+ if ( p . x >= x0 && p . x <= x1 && p . y >= y0 && p . y <= y1 && p !== a && p !== c &&
234
+ pointInTriangle ( ax , ay , bx , by , cx , cy , p . x , p . y ) && area ( p . prev , p , p . next ) >= 0 ) return false ;
226
235
p = p . prevZ ;
227
236
228
- if ( n !== ear . prev && n !== ear . next &&
229
- pointInTriangle ( a . x , a . y , b . x , b . y , c . x , c . y , n . x , n . y ) &&
230
- area ( n . prev , n , n . next ) >= 0 ) return false ;
237
+ if ( n . x >= x0 && n . x <= x1 && n . y >= y0 && n . y <= y1 && n !== a && n !== c &&
238
+ pointInTriangle ( ax , ay , bx , by , cx , cy , n . x , n . y ) && area ( n . prev , n , n . next ) >= 0 ) return false ;
231
239
n = n . nextZ ;
232
240
233
241
}
234
242
235
243
// look for remaining points in decreasing z-order
236
244
while ( p && p . z >= minZ ) {
237
245
238
- if ( p !== ear . prev && p !== ear . next &&
239
- pointInTriangle ( a . x , a . y , b . x , b . y , c . x , c . y , p . x , p . y ) &&
240
- area ( p . prev , p , p . next ) >= 0 ) return false ;
246
+ if ( p . x >= x0 && p . x <= x1 && p . y >= y0 && p . y <= y1 && p !== a && p !== c &&
247
+ pointInTriangle ( ax , ay , bx , by , cx , cy , p . x , p . y ) && area ( p . prev , p , p . next ) >= 0 ) return false ;
241
248
p = p . prevZ ;
242
249
243
250
}
244
251
245
252
// look for remaining points in increasing z-order
246
253
while ( n && n . z <= maxZ ) {
247
254
248
- if ( n !== ear . prev && n !== ear . next &&
249
- pointInTriangle ( a . x , a . y , b . x , b . y , c . x , c . y , n . x , n . y ) &&
250
- area ( n . prev , n , n . next ) >= 0 ) return false ;
255
+ if ( n . x >= x0 && n . x <= x1 && n . y >= y0 && n . y <= y1 && n !== a && n !== c &&
256
+ pointInTriangle ( ax , ay , bx , by , cx , cy , n . x , n . y ) && area ( n . prev , n , n . next ) >= 0 ) return false ;
251
257
n = n . nextZ ;
252
258
253
259
}
@@ -267,9 +273,9 @@ function cureLocalIntersections( start, triangles, dim ) {
267
273
268
274
if ( ! equals ( a , b ) && intersects ( a , p , p . next , b ) && locallyInside ( a , b ) && locallyInside ( b , a ) ) {
269
275
270
- triangles . push ( a . i / dim ) ;
271
- triangles . push ( p . i / dim ) ;
272
- triangles . push ( b . i / dim ) ;
276
+ triangles . push ( a . i / dim | 0 ) ;
277
+ triangles . push ( p . i / dim | 0 ) ;
278
+ triangles . push ( b . i / dim | 0 ) ;
273
279
274
280
// remove two nodes involved
275
281
removeNode ( p ) ;
@@ -307,8 +313,8 @@ function splitEarcut( start, triangles, dim, minX, minY, invSize ) {
307
313
c = filterPoints ( c , c . next ) ;
308
314
309
315
// run earcut on each half
310
- earcutLinked ( a , triangles , dim , minX , minY , invSize ) ;
311
- earcutLinked ( c , triangles , dim , minX , minY , invSize ) ;
316
+ earcutLinked ( a , triangles , dim , minX , minY , invSize , 0 ) ;
317
+ earcutLinked ( c , triangles , dim , minX , minY , invSize , 0 ) ;
312
318
return ;
313
319
314
320
}
@@ -344,8 +350,7 @@ function eliminateHoles( data, holeIndices, outerNode, dim ) {
344
350
// process holes from left to right
345
351
for ( i = 0 ; i < queue . length ; i ++ ) {
346
352
347
- eliminateHole ( queue [ i ] , outerNode ) ;
348
- outerNode = filterPoints ( outerNode , outerNode . next ) ;
353
+ outerNode = eliminateHole ( queue [ i ] , outerNode ) ;
349
354
350
355
}
351
356
@@ -362,26 +367,29 @@ function compareX( a, b ) {
362
367
// find a bridge between vertices that connects hole with an outer ring and link it
363
368
function eliminateHole ( hole , outerNode ) {
364
369
365
- outerNode = findHoleBridge ( hole , outerNode ) ;
366
- if ( outerNode ) {
367
-
368
- const b = splitPolygon ( outerNode , hole ) ;
370
+ const bridge = findHoleBridge ( hole , outerNode ) ;
371
+ if ( ! bridge ) {
369
372
370
- // filter collinear points around the cuts
371
- filterPoints ( outerNode , outerNode . next ) ;
372
- filterPoints ( b , b . next ) ;
373
+ return outerNode ;
373
374
374
375
}
375
376
377
+ const bridgeReverse = splitPolygon ( bridge , hole ) ;
378
+
379
+ // filter collinear points around the cuts
380
+ filterPoints ( bridgeReverse , bridgeReverse . next ) ;
381
+ return filterPoints ( bridge , bridge . next ) ;
382
+
376
383
}
377
384
378
385
// David Eberly's algorithm for finding a bridge between hole and outer polygon
379
386
function findHoleBridge ( hole , outerNode ) {
380
387
381
- let p = outerNode ;
382
- const hx = hole . x ;
383
- const hy = hole . y ;
384
- let qx = - Infinity , m ;
388
+ let p = outerNode ,
389
+ qx = - Infinity ,
390
+ m ;
391
+
392
+ const hx = hole . x , hy = hole . y ;
385
393
386
394
// find a segment intersected by a ray from the hole's leftmost point to the left;
387
395
// segment's endpoint with lesser x will be potential connection point
@@ -393,14 +401,8 @@ function findHoleBridge( hole, outerNode ) {
393
401
if ( x <= hx && x > qx ) {
394
402
395
403
qx = x ;
396
- if ( x === hx ) {
397
-
398
- if ( hy === p . y ) return p ;
399
- if ( hy === p . next . y ) return p . next ;
400
-
401
- }
402
-
403
404
m = p . x < p . next . x ? p : p . next ;
405
+ if ( x === hx ) return m ; // hole touches outer segment; pick leftmost endpoint
404
406
405
407
}
406
408
@@ -412,8 +414,6 @@ function findHoleBridge( hole, outerNode ) {
412
414
413
415
if ( ! m ) return null ;
414
416
415
- if ( hx === qx ) return m ; // hole touches outer segment; pick leftmost endpoint
416
-
417
417
// look for points inside the triangle of hole point, segment intersection and endpoint;
418
418
// if there are no points found, we have a valid connection;
419
419
// otherwise choose the point of the minimum angle with the ray as connection point
@@ -462,7 +462,7 @@ function indexCurve( start, minX, minY, invSize ) {
462
462
let p = start ;
463
463
do {
464
464
465
- if ( p . z === null ) p . z = zOrder ( p . x , p . y , minX , minY , invSize ) ;
465
+ if ( p . z === 0 ) p . z = zOrder ( p . x , p . y , minX , minY , invSize ) ;
466
466
p . prevZ = p . prev ;
467
467
p . nextZ = p . next ;
468
468
p = p . next ;
@@ -546,8 +546,8 @@ function sortLinked( list ) {
546
546
function zOrder ( x , y , minX , minY , invSize ) {
547
547
548
548
// coords are transformed into non-negative 15-bit integer range
549
- x = 32767 * ( x - minX ) * invSize ;
550
- y = 32767 * ( y - minY ) * invSize ;
549
+ x = ( x - minX ) * invSize | 0 ;
550
+ y = ( y - minY ) * invSize | 0 ;
551
551
552
552
x = ( x | ( x << 8 ) ) & 0x00FF00FF ;
553
553
x = ( x | ( x << 4 ) ) & 0x0F0F0F0F ;
@@ -582,19 +582,19 @@ function getLeftmost( start ) {
582
582
// check if a point lies within a convex triangle
583
583
function pointInTriangle ( ax , ay , bx , by , cx , cy , px , py ) {
584
584
585
- return ( cx - px ) * ( ay - py ) - ( ax - px ) * ( cy - py ) >= 0 &&
586
- ( ax - px ) * ( by - py ) - ( bx - px ) * ( ay - py ) >= 0 &&
587
- ( bx - px ) * ( cy - py ) - ( cx - px ) * ( by - py ) >= 0 ;
585
+ return ( cx - px ) * ( ay - py ) >= ( ax - px ) * ( cy - py ) &&
586
+ ( ax - px ) * ( by - py ) >= ( bx - px ) * ( ay - py ) &&
587
+ ( bx - px ) * ( cy - py ) >= ( cx - px ) * ( by - py ) ;
588
588
589
589
}
590
590
591
591
// check if a diagonal between two polygon nodes is valid (lies in polygon interior)
592
592
function isValidDiagonal ( a , b ) {
593
593
594
- return a . next . i !== b . i && a . prev . i !== b . i && ! intersectsPolygon ( a , b ) && // doesn 't intersect other edges
595
- ( locallyInside ( a , b ) && locallyInside ( b , a ) && middleInside ( a , b ) && // locally visible
596
- ( area ( a . prev , a , b . prev ) || area ( a , b . prev , b ) ) || // does not create opposite-facing sectors
597
- equals ( a , b ) && area ( a . prev , a , a . next ) > 0 && area ( b . prev , b , b . next ) > 0 ) ; // special zero-length case
594
+ return a . next . i !== b . i && a . prev . i !== b . i && ! intersectsPolygon ( a , b ) && // dones 't intersect other edges
595
+ ( locallyInside ( a , b ) && locallyInside ( b , a ) && middleInside ( a , b ) && // locally visible
596
+ ( area ( a . prev , a , b . prev ) || area ( a , b . prev , b ) ) || // does not create opposite-facing sectors
597
+ equals ( a , b ) && area ( a . prev , a , a . next ) > 0 && area ( b . prev , b , b . next ) > 0 ) ; // special zero-length case
598
598
599
599
}
600
600
@@ -651,7 +651,7 @@ function intersectsPolygon( a, b ) {
651
651
do {
652
652
653
653
if ( p . i !== a . i && p . next . i !== a . i && p . i !== b . i && p . next . i !== b . i &&
654
- intersects ( p , p . next , a , b ) ) return true ;
654
+ intersects ( p , p . next , a , b ) ) return true ;
655
655
p = p . next ;
656
656
657
657
} while ( p !== a ) ;
@@ -679,7 +679,7 @@ function middleInside( a, b ) {
679
679
do {
680
680
681
681
if ( ( ( p . y > py ) !== ( p . next . y > py ) ) && p . next . y !== p . y &&
682
- ( px < ( p . next . x - p . x ) * ( py - p . y ) / ( p . next . y - p . y ) + p . x ) )
682
+ ( px < ( p . next . x - p . x ) * ( py - p . y ) / ( p . next . y - p . y ) + p . x ) )
683
683
inside = ! inside ;
684
684
p = p . next ;
685
685
@@ -761,7 +761,7 @@ function Node( i, x, y ) {
761
761
this . next = null ;
762
762
763
763
// z-order curve value
764
- this . z = null ;
764
+ this . z = 0 ;
765
765
766
766
// previous and next nodes in z-order
767
767
this . prevZ = null ;
0 commit comments