Skip to content

Commit ea01dba

Browse files
Earcut: Upgrade to version 2.2.4. (mrdoob#24760)
* Earcut: Upgrade version to fix infinite loop. * fix: redefine var parameters * fix: this testcase had many faulty triangles * fix: this testcase had many faulty triangles * fix: restore the modified jpg file * chore: test * chore: revent microoptimizations * chore: test * chore: test * fix: filter collinear points around the cuts * fix: simplify hole elimination * chore: format the code * Update Earcut.js Clean up. Co-authored-by: Michael Herzog <[email protected]>
1 parent d75f4d1 commit ea01dba

File tree

2 files changed

+67
-67
lines changed

2 files changed

+67
-67
lines changed
Loading

src/extras/Earcut.js

+67-67
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/**
2-
* Port from https://github.com/mapbox/earcut (v2.2.2)
2+
* Port from https://github.com/mapbox/earcut (v2.2.4)
33
*/
44

55
const Earcut = {
@@ -36,11 +36,11 @@ const Earcut = {
3636

3737
// minX, minY and invSize are later used to transform coords into integers for z-order calculation
3838
invSize = Math.max( maxX - minX, maxY - minY );
39-
invSize = invSize !== 0 ? 1 / invSize : 0;
39+
invSize = invSize !== 0 ? 32767 / invSize : 0;
4040

4141
}
4242

43-
earcutLinked( outerNode, triangles, dim, minX, minY, invSize );
43+
earcutLinked( outerNode, triangles, dim, minX, minY, invSize, 0 );
4444

4545
return triangles;
4646

@@ -125,9 +125,9 @@ function earcutLinked( ear, triangles, dim, minX, minY, invSize, pass ) {
125125
if ( invSize ? isEarHashed( ear, minX, minY, invSize ) : isEar( ear ) ) {
126126

127127
// 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 );
131131

132132
removeNode( ear );
133133

@@ -182,11 +182,19 @@ function isEar( ear ) {
182182
if ( area( a, b, c ) >= 0 ) return false; // reflex, can't be an ear
183183

184184
// 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;
186186

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 ) {
188195

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 ) &&
190198
area( p.prev, p, p.next ) >= 0 ) return false;
191199
p = p.next;
192200

@@ -204,50 +212,48 @@ function isEarHashed( ear, minX, minY, invSize ) {
204212

205213
if ( area( a, b, c ) >= 0 ) return false; // reflex, can't be an ear
206214

215+
const ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;
216+
207217
// 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 );
212222

213223
// 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 );
216226

217227
let p = ear.prevZ,
218228
n = ear.nextZ;
219229

220230
// look for points inside the triangle in both directions
221231
while ( p && p.z >= minZ && n && n.z <= maxZ ) {
222232

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;
226235
p = p.prevZ;
227236

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;
231239
n = n.nextZ;
232240

233241
}
234242

235243
// look for remaining points in decreasing z-order
236244
while ( p && p.z >= minZ ) {
237245

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;
241248
p = p.prevZ;
242249

243250
}
244251

245252
// look for remaining points in increasing z-order
246253
while ( n && n.z <= maxZ ) {
247254

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;
251257
n = n.nextZ;
252258

253259
}
@@ -267,9 +273,9 @@ function cureLocalIntersections( start, triangles, dim ) {
267273

268274
if ( ! equals( a, b ) && intersects( a, p, p.next, b ) && locallyInside( a, b ) && locallyInside( b, a ) ) {
269275

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 );
273279

274280
// remove two nodes involved
275281
removeNode( p );
@@ -307,8 +313,8 @@ function splitEarcut( start, triangles, dim, minX, minY, invSize ) {
307313
c = filterPoints( c, c.next );
308314

309315
// 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 );
312318
return;
313319

314320
}
@@ -344,8 +350,7 @@ function eliminateHoles( data, holeIndices, outerNode, dim ) {
344350
// process holes from left to right
345351
for ( i = 0; i < queue.length; i ++ ) {
346352

347-
eliminateHole( queue[ i ], outerNode );
348-
outerNode = filterPoints( outerNode, outerNode.next );
353+
outerNode = eliminateHole( queue[ i ], outerNode );
349354

350355
}
351356

@@ -362,26 +367,29 @@ function compareX( a, b ) {
362367
// find a bridge between vertices that connects hole with an outer ring and link it
363368
function eliminateHole( hole, outerNode ) {
364369

365-
outerNode = findHoleBridge( hole, outerNode );
366-
if ( outerNode ) {
367-
368-
const b = splitPolygon( outerNode, hole );
370+
const bridge = findHoleBridge( hole, outerNode );
371+
if ( ! bridge ) {
369372

370-
// filter collinear points around the cuts
371-
filterPoints( outerNode, outerNode.next );
372-
filterPoints( b, b.next );
373+
return outerNode;
373374

374375
}
375376

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+
376383
}
377384

378385
// David Eberly's algorithm for finding a bridge between hole and outer polygon
379386
function findHoleBridge( hole, outerNode ) {
380387

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;
385393

386394
// find a segment intersected by a ray from the hole's leftmost point to the left;
387395
// segment's endpoint with lesser x will be potential connection point
@@ -393,14 +401,8 @@ function findHoleBridge( hole, outerNode ) {
393401
if ( x <= hx && x > qx ) {
394402

395403
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-
403404
m = p.x < p.next.x ? p : p.next;
405+
if ( x === hx ) return m; // hole touches outer segment; pick leftmost endpoint
404406

405407
}
406408

@@ -412,8 +414,6 @@ function findHoleBridge( hole, outerNode ) {
412414

413415
if ( ! m ) return null;
414416

415-
if ( hx === qx ) return m; // hole touches outer segment; pick leftmost endpoint
416-
417417
// look for points inside the triangle of hole point, segment intersection and endpoint;
418418
// if there are no points found, we have a valid connection;
419419
// otherwise choose the point of the minimum angle with the ray as connection point
@@ -462,7 +462,7 @@ function indexCurve( start, minX, minY, invSize ) {
462462
let p = start;
463463
do {
464464

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 );
466466
p.prevZ = p.prev;
467467
p.nextZ = p.next;
468468
p = p.next;
@@ -546,8 +546,8 @@ function sortLinked( list ) {
546546
function zOrder( x, y, minX, minY, invSize ) {
547547

548548
// 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;
551551

552552
x = ( x | ( x << 8 ) ) & 0x00FF00FF;
553553
x = ( x | ( x << 4 ) ) & 0x0F0F0F0F;
@@ -582,19 +582,19 @@ function getLeftmost( start ) {
582582
// check if a point lies within a convex triangle
583583
function pointInTriangle( ax, ay, bx, by, cx, cy, px, py ) {
584584

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 );
588588

589589
}
590590

591591
// check if a diagonal between two polygon nodes is valid (lies in polygon interior)
592592
function isValidDiagonal( a, b ) {
593593

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
598598

599599
}
600600

@@ -651,7 +651,7 @@ function intersectsPolygon( a, b ) {
651651
do {
652652

653653
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;
655655
p = p.next;
656656

657657
} while ( p !== a );
@@ -679,7 +679,7 @@ function middleInside( a, b ) {
679679
do {
680680

681681
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 ) )
683683
inside = ! inside;
684684
p = p.next;
685685

@@ -761,7 +761,7 @@ function Node( i, x, y ) {
761761
this.next = null;
762762

763763
// z-order curve value
764-
this.z = null;
764+
this.z = 0;
765765

766766
// previous and next nodes in z-order
767767
this.prevZ = null;

0 commit comments

Comments
 (0)