#include <assert.h>
#include <stddef.h>
#include <setjmp.h>
#include "mesh.h"
#include "geom.h"
#include "tess.h"
#include "dict.h"
#include "priorityq.h"
#include "bucketalloc.h"
#include "sweep.h"
#define TRUE 1
#define FALSE 0
#ifdef FOR_TRITE_TEST_PROGRAM
extern void DebugEvent( TESStesselator *tess );
#else
#define DebugEvent( tess )
#endif
#define MAX(x,y) ((x) >= (y) ? (x) : (y))
#define MIN(x,y) ((x) <= (y) ? (x) : (y))
#define AddWinding(eDst,eSrc) (eDst->winding += eSrc->winding, \
eDst->Sym->winding += eSrc->Sym->winding)
static void SweepEvent( TESStesselator *tess, TESSvertex *vEvent );
static void WalkDirtyRegions( TESStesselator *tess, ActiveRegion *regUp );
static int CheckForRightSplice( TESStesselator *tess, ActiveRegion *regUp );
static int EdgeLeq( TESStesselator *tess, ActiveRegion *reg1, ActiveRegion *reg2 )
{
TESSvertex *event = tess->event;
TESShalfEdge *e1, *e2;
TESSreal t1, t2;
e1 = reg1->eUp;
e2 = reg2->eUp;
if( e1->Dst == event ) {
if( e2->Dst == event ) {
if( VertLeq( e1->Org, e2->Org )) {
return EdgeSign( e2->Dst, e1->Org, e2->Org ) <= 0;
}
return EdgeSign( e1->Dst, e2->Org, e1->Org ) >= 0;
}
return EdgeSign( e2->Dst, event, e2->Org ) <= 0;
}
if( e2->Dst == event ) {
return EdgeSign( e1->Dst, event, e1->Org ) >= 0;
}
t1 = EdgeEval( e1->Dst, event, e1->Org );
t2 = EdgeEval( e2->Dst, event, e2->Org );
return (t1 >= t2);
}
static void DeleteRegion( TESStesselator *tess, ActiveRegion *reg )
{
if( reg->fixUpperEdge ) {
assert( reg->eUp->winding == 0 );
}
reg->eUp->activeRegion = NULL;
dictDelete( tess->dict, reg->nodeUp );
bucketFree( tess->regionPool, reg );
}
static int FixUpperEdge( TESStesselator *tess, ActiveRegion *reg, TESShalfEdge *newEdge )
{
assert( reg->fixUpperEdge );
if ( !tessMeshDelete( tess->mesh, reg->eUp ) ) return 0;
reg->fixUpperEdge = FALSE;
reg->eUp = newEdge;
newEdge->activeRegion = reg;
return 1;
}
static ActiveRegion *TopLeftRegion( TESStesselator *tess, ActiveRegion *reg )
{
TESSvertex *org = reg->eUp->Org;
TESShalfEdge *e;
do {
reg = RegionAbove( reg );
} while( reg->eUp->Org == org );
if( reg->fixUpperEdge ) {
e = tessMeshConnect( tess->mesh, RegionBelow(reg)->eUp->Sym, reg->eUp->Lnext );
if (e == NULL) return NULL;
if ( !FixUpperEdge( tess, reg, e ) ) return NULL;
reg = RegionAbove( reg );
}
return reg;
}
static ActiveRegion *TopRightRegion( ActiveRegion *reg )
{
TESSvertex *dst = reg->eUp->Dst;
do {
reg = RegionAbove( reg );
} while( reg->eUp->Dst == dst );
return reg;
}
static ActiveRegion *AddRegionBelow( TESStesselator *tess,
ActiveRegion *regAbove,
TESShalfEdge *eNewUp )
{
ActiveRegion *regNew = (ActiveRegion *)bucketAlloc( tess->regionPool );
if (regNew == NULL) longjmp(tess->env,1);
regNew->eUp = eNewUp;
regNew->nodeUp = dictInsertBefore( tess->dict, regAbove->nodeUp, regNew );
if (regNew->nodeUp == NULL) longjmp(tess->env,1);
regNew->fixUpperEdge = FALSE;
regNew->sentinel = FALSE;
regNew->dirty = FALSE;
eNewUp->activeRegion = regNew;
return regNew;
}
static int IsWindingInside( TESStesselator *tess, int n )
{
switch( tess->windingRule ) {
case TESS_WINDING_ODD:
return (n & 1);
case TESS_WINDING_NONZERO:
return (n != 0);
case TESS_WINDING_POSITIVE:
return (n > 0);
case TESS_WINDING_NEGATIVE:
return (n < 0);
case TESS_WINDING_ABS_GEQ_TWO:
return (n >= 2) || (n <= -2);
}
assert( FALSE );
return( FALSE );
}
static void ComputeWinding( TESStesselator *tess, ActiveRegion *reg )
{
reg->windingNumber = RegionAbove(reg)->windingNumber + reg->eUp->winding;
reg->inside = IsWindingInside( tess, reg->windingNumber );
}
static void FinishRegion( TESStesselator *tess, ActiveRegion *reg )
{
TESShalfEdge *e = reg->eUp;
TESSface *f = e->Lface;
f->inside = reg->inside;
f->anEdge = e;
DeleteRegion( tess, reg );
}
static TESShalfEdge *FinishLeftRegions( TESStesselator *tess,
ActiveRegion *regFirst, ActiveRegion *regLast )
{
ActiveRegion *reg, *regPrev;
TESShalfEdge *e, *ePrev;
regPrev = regFirst;
ePrev = regFirst->eUp;
while( regPrev != regLast ) {
regPrev->fixUpperEdge = FALSE;
reg = RegionBelow( regPrev );
e = reg->eUp;
if( e->Org != ePrev->Org ) {
if( ! reg->fixUpperEdge ) {
FinishRegion( tess, regPrev );
break;
}
e = tessMeshConnect( tess->mesh, ePrev->Lprev, e->Sym );
if (e == NULL) longjmp(tess->env,1);
if ( !FixUpperEdge( tess, reg, e ) ) longjmp(tess->env,1);
}
if( ePrev->Onext != e ) {
if ( !tessMeshSplice( tess->mesh, e->Oprev, e ) ) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, ePrev, e ) ) longjmp(tess->env,1);
}
FinishRegion( tess, regPrev );
ePrev = reg->eUp;
regPrev = reg;
}
return ePrev;
}
static void AddRightEdges( TESStesselator *tess, ActiveRegion *regUp,
TESShalfEdge *eFirst, TESShalfEdge *eLast, TESShalfEdge *eTopLeft,
int cleanUp )
{
ActiveRegion *reg, *regPrev;
TESShalfEdge *e, *ePrev;
int firstTime = TRUE;
e = eFirst;
do {
assert( VertLeq( e->Org, e->Dst ));
AddRegionBelow( tess, regUp, e->Sym );
e = e->Onext;
} while ( e != eLast );
if( eTopLeft == NULL ) {
eTopLeft = RegionBelow( regUp )->eUp->Rprev;
}
regPrev = regUp;
ePrev = eTopLeft;
for( ;; ) {
reg = RegionBelow( regPrev );
e = reg->eUp->Sym;
if( e->Org != ePrev->Org ) break;
if( e->Onext != ePrev ) {
if ( !tessMeshSplice( tess->mesh, e->Oprev, e ) ) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, ePrev->Oprev, e ) ) longjmp(tess->env,1);
}
reg->windingNumber = regPrev->windingNumber - e->winding;
reg->inside = IsWindingInside( tess, reg->windingNumber );
regPrev->dirty = TRUE;
if( ! firstTime && CheckForRightSplice( tess, regPrev )) {
AddWinding( e, ePrev );
DeleteRegion( tess, regPrev );
if ( !tessMeshDelete( tess->mesh, ePrev ) ) longjmp(tess->env,1);
}
firstTime = FALSE;
regPrev = reg;
ePrev = e;
}
regPrev->dirty = TRUE;
assert( regPrev->windingNumber - e->winding == reg->windingNumber );
if( cleanUp ) {
WalkDirtyRegions( tess, regPrev );
}
}
static void SpliceMergeVertices( TESStesselator *tess, TESShalfEdge *e1,
TESShalfEdge *e2 )
{
if ( !tessMeshSplice( tess->mesh, e1, e2 ) ) longjmp(tess->env,1);
}
static void VertexWeights( TESSvertex *isect, TESSvertex *org, TESSvertex *dst,
TESSreal *weights )
{
TESSreal t1 = VertL1dist( org, isect );
TESSreal t2 = VertL1dist( dst, isect );
weights[0] = (TESSreal)0.5 * t2 / (t1 + t2);
weights[1] = (TESSreal)0.5 * t1 / (t1 + t2);
isect->coords[0] += weights[0]*org->coords[0] + weights[1]*dst->coords[0];
isect->coords[1] += weights[0]*org->coords[1] + weights[1]*dst->coords[1];
isect->coords[2] += weights[0]*org->coords[2] + weights[1]*dst->coords[2];
}
static void GetIntersectData( TESStesselator *tess, TESSvertex *isect,
TESSvertex *orgUp, TESSvertex *dstUp,
TESSvertex *orgLo, TESSvertex *dstLo )
{
TESSreal weights[4];
TESS_NOTUSED( tess );
isect->coords[0] = isect->coords[1] = isect->coords[2] = 0;
isect->idx = TESS_UNDEF;
VertexWeights( isect, orgUp, dstUp, &weights[0] );
VertexWeights( isect, orgLo, dstLo, &weights[2] );
}
static int CheckForRightSplice( TESStesselator *tess, ActiveRegion *regUp )
{
ActiveRegion *regLo = RegionBelow(regUp);
TESShalfEdge *eUp = regUp->eUp;
TESShalfEdge *eLo = regLo->eUp;
if( VertLeq( eUp->Org, eLo->Org )) {
if( EdgeSign( eLo->Dst, eUp->Org, eLo->Org ) > 0 ) return FALSE;
if( ! VertEq( eUp->Org, eLo->Org )) {
if ( tessMeshSplitEdge( tess->mesh, eLo->Sym ) == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eUp, eLo->Oprev ) ) longjmp(tess->env,1);
regUp->dirty = regLo->dirty = TRUE;
} else if( eUp->Org != eLo->Org ) {
pqDelete( tess->pq, eUp->Org->pqHandle );
SpliceMergeVertices( tess, eLo->Oprev, eUp );
}
} else {
if( EdgeSign( eUp->Dst, eLo->Org, eUp->Org ) < 0 ) return FALSE;
RegionAbove(regUp)->dirty = regUp->dirty = TRUE;
if (tessMeshSplitEdge( tess->mesh, eUp->Sym ) == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eLo->Oprev, eUp ) ) longjmp(tess->env,1);
}
return TRUE;
}
static int CheckForLeftSplice( TESStesselator *tess, ActiveRegion *regUp )
{
ActiveRegion *regLo = RegionBelow(regUp);
TESShalfEdge *eUp = regUp->eUp;
TESShalfEdge *eLo = regLo->eUp;
TESShalfEdge *e;
assert( ! VertEq( eUp->Dst, eLo->Dst ));
if( VertLeq( eUp->Dst, eLo->Dst )) {
if( EdgeSign( eUp->Dst, eLo->Dst, eUp->Org ) < 0 ) return FALSE;
RegionAbove(regUp)->dirty = regUp->dirty = TRUE;
e = tessMeshSplitEdge( tess->mesh, eUp );
if (e == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eLo->Sym, e ) ) longjmp(tess->env,1);
e->Lface->inside = regUp->inside;
} else {
if( EdgeSign( eLo->Dst, eUp->Dst, eLo->Org ) > 0 ) return FALSE;
regUp->dirty = regLo->dirty = TRUE;
e = tessMeshSplitEdge( tess->mesh, eLo );
if (e == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eUp->Lnext, eLo->Sym ) ) longjmp(tess->env,1);
e->Rface->inside = regUp->inside;
}
return TRUE;
}
static int CheckForIntersect( TESStesselator *tess, ActiveRegion *regUp )
{
ActiveRegion *regLo = RegionBelow(regUp);
TESShalfEdge *eUp = regUp->eUp;
TESShalfEdge *eLo = regLo->eUp;
TESSvertex *orgUp = eUp->Org;
TESSvertex *orgLo = eLo->Org;
TESSvertex *dstUp = eUp->Dst;
TESSvertex *dstLo = eLo->Dst;
TESSreal tMinUp, tMaxLo;
TESSvertex isect, *orgMin;
TESShalfEdge *e;
assert( ! VertEq( dstLo, dstUp ));
assert( EdgeSign( dstUp, tess->event, orgUp ) <= 0 );
assert( EdgeSign( dstLo, tess->event, orgLo ) >= 0 );
assert( orgUp != tess->event && orgLo != tess->event );
assert( ! regUp->fixUpperEdge && ! regLo->fixUpperEdge );
if( orgUp == orgLo ) return FALSE;
tMinUp = MIN( orgUp->t, dstUp->t );
tMaxLo = MAX( orgLo->t, dstLo->t );
if( tMinUp > tMaxLo ) return FALSE;
if( VertLeq( orgUp, orgLo )) {
if( EdgeSign( dstLo, orgUp, orgLo ) > 0 ) return FALSE;
} else {
if( EdgeSign( dstUp, orgLo, orgUp ) < 0 ) return FALSE;
}
DebugEvent( tess );
tesedgeIntersect( dstUp, orgUp, dstLo, orgLo, &isect );
assert( MIN( orgUp->t, dstUp->t ) <= isect.t );
assert( isect.t <= MAX( orgLo->t, dstLo->t ));
assert( MIN( dstLo->s, dstUp->s ) <= isect.s );
assert( isect.s <= MAX( orgLo->s, orgUp->s ));
if( VertLeq( &isect, tess->event )) {
isect.s = tess->event->s;
isect.t = tess->event->t;
}
orgMin = VertLeq( orgUp, orgLo ) ? orgUp : orgLo;
if( VertLeq( orgMin, &isect )) {
isect.s = orgMin->s;
isect.t = orgMin->t;
}
if( VertEq( &isect, orgUp ) || VertEq( &isect, orgLo )) {
(void) CheckForRightSplice( tess, regUp );
return FALSE;
}
if( (! VertEq( dstUp, tess->event )
&& EdgeSign( dstUp, tess->event, &isect ) >= 0)
|| (! VertEq( dstLo, tess->event )
&& EdgeSign( dstLo, tess->event, &isect ) <= 0 ))
{
if( dstLo == tess->event ) {
if (tessMeshSplitEdge( tess->mesh, eUp->Sym ) == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eLo->Sym, eUp ) ) longjmp(tess->env,1);
regUp = TopLeftRegion( tess, regUp );
if (regUp == NULL) longjmp(tess->env,1);
eUp = RegionBelow(regUp)->eUp;
FinishLeftRegions( tess, RegionBelow(regUp), regLo );
AddRightEdges( tess, regUp, eUp->Oprev, eUp, eUp, TRUE );
return TRUE;
}
if( dstUp == tess->event ) {
if (tessMeshSplitEdge( tess->mesh, eLo->Sym ) == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eUp->Lnext, eLo->Oprev ) ) longjmp(tess->env,1);
regLo = regUp;
regUp = TopRightRegion( regUp );
e = RegionBelow(regUp)->eUp->Rprev;
regLo->eUp = eLo->Oprev;
eLo = FinishLeftRegions( tess, regLo, NULL );
AddRightEdges( tess, regUp, eLo->Onext, eUp->Rprev, e, TRUE );
return TRUE;
}
if( EdgeSign( dstUp, tess->event, &isect ) >= 0 ) {
RegionAbove(regUp)->dirty = regUp->dirty = TRUE;
if (tessMeshSplitEdge( tess->mesh, eUp->Sym ) == NULL) longjmp(tess->env,1);
eUp->Org->s = tess->event->s;
eUp->Org->t = tess->event->t;
}
if( EdgeSign( dstLo, tess->event, &isect ) <= 0 ) {
regUp->dirty = regLo->dirty = TRUE;
if (tessMeshSplitEdge( tess->mesh, eLo->Sym ) == NULL) longjmp(tess->env,1);
eLo->Org->s = tess->event->s;
eLo->Org->t = tess->event->t;
}
return FALSE;
}
if (tessMeshSplitEdge( tess->mesh, eUp->Sym ) == NULL) longjmp(tess->env,1);
if (tessMeshSplitEdge( tess->mesh, eLo->Sym ) == NULL) longjmp(tess->env,1);
if ( !tessMeshSplice( tess->mesh, eLo->Oprev, eUp ) ) longjmp(tess->env,1);
eUp->Org->s = isect.s;
eUp->Org->t = isect.t;
eUp->Org->pqHandle = pqInsert( &tess->alloc, tess->pq, eUp->Org );
if (eUp->Org->pqHandle == INV_HANDLE) {
pqDeletePriorityQ( &tess->alloc, tess->pq );
tess->pq = NULL;
longjmp(tess->env,1);
}
GetIntersectData( tess, eUp->Org, orgUp, dstUp, orgLo, dstLo );
RegionAbove(regUp)->dirty = regUp->dirty = regLo->dirty = TRUE;
return FALSE;
}
static void WalkDirtyRegions( TESStesselator *tess, ActiveRegion *regUp )
{
ActiveRegion *regLo = RegionBelow(regUp);
TESShalfEdge *eUp, *eLo;
for( ;; ) {
while( regLo->dirty ) {
regUp = regLo;
regLo = RegionBelow(regLo);
}
if( ! regUp->dirty ) {
regLo = regUp;
regUp = RegionAbove( regUp );
if( regUp == NULL || ! regUp->dirty ) {
return;
}
}
regUp->dirty = FALSE;
eUp = regUp->eUp;
eLo = regLo->eUp;
if( eUp->Dst != eLo->Dst ) {
if( CheckForLeftSplice( tess, regUp )) {
if( regLo->fixUpperEdge ) {
DeleteRegion( tess, regLo );
if ( !tessMeshDelete( tess->mesh, eLo ) ) longjmp(tess->env,1);
regLo = RegionBelow( regUp );
eLo = regLo->eUp;
} else if( regUp->fixUpperEdge ) {
DeleteRegion( tess, regUp );
if ( !tessMeshDelete( tess->mesh, eUp ) ) longjmp(tess->env,1);
regUp = RegionAbove( regLo );
eUp = regUp->eUp;
}
}
}
if( eUp->Org != eLo->Org ) {
if( eUp->Dst != eLo->Dst
&& ! regUp->fixUpperEdge && ! regLo->fixUpperEdge
&& (eUp->Dst == tess->event || eLo->Dst == tess->event) )
{
if( CheckForIntersect( tess, regUp )) {
return;
}
} else {
(void) CheckForRightSplice( tess, regUp );
}
}
if( eUp->Org == eLo->Org && eUp->Dst == eLo->Dst ) {
AddWinding( eLo, eUp );
DeleteRegion( tess, regUp );
if ( !tessMeshDelete( tess->mesh, eUp ) ) longjmp(tess->env,1);
regUp = RegionAbove( regLo );
}
}
}
static void ConnectRightVertex( TESStesselator *tess, ActiveRegion *regUp,
TESShalfEdge *eBottomLeft )
{
TESShalfEdge *eNew;
TESShalfEdge *eTopLeft = eBottomLeft->Onext;
ActiveRegion *regLo = RegionBelow(regUp);
TESShalfEdge *eUp = regUp->eUp;
TESShalfEdge *eLo = regLo->eUp;
int degenerate = FALSE;
if( eUp->Dst != eLo->Dst ) {
(void) CheckForIntersect( tess, regUp );
}
if( VertEq( eUp->Org, tess->event )) {
if ( !tessMeshSplice( tess->mesh, eTopLeft->Oprev, eUp ) ) longjmp(tess->env,1);
regUp = TopLeftRegion( tess, regUp );
if (regUp == NULL) longjmp(tess->env,1);
eTopLeft = RegionBelow( regUp )->eUp;
FinishLeftRegions( tess, RegionBelow(regUp), regLo );
degenerate = TRUE;
}
if( VertEq( eLo->Org, tess->event )) {
if ( !tessMeshSplice( tess->mesh, eBottomLeft, eLo->Oprev ) ) longjmp(tess->env,1);
eBottomLeft = FinishLeftRegions( tess, regLo, NULL );
degenerate = TRUE;
}
if( degenerate ) {
AddRightEdges( tess, regUp, eBottomLeft->Onext, eTopLeft, eTopLeft, TRUE );
return;
}
if( VertLeq( eLo->Org, eUp->Org )) {
eNew = eLo->Oprev;
} else {
eNew = eUp;
}
eNew = tessMeshConnect( tess->mesh, eBottomLeft->Lprev, eNew );
if (eNew == NULL) longjmp(tess->env,1);
AddRightEdges( tess, regUp, eNew, eNew->Onext, eNew->Onext, FALSE );
eNew->Sym->activeRegion->fixUpperEdge = TRUE;
WalkDirtyRegions( tess, regUp );
}
#define TOLERANCE_NONZERO FALSE
static void ConnectLeftDegenerate( TESStesselator *tess,
ActiveRegion *regUp, TESSvertex *vEvent )
{
TESShalfEdge *e, *eTopLeft, *eTopRight, *eLast;
ActiveRegion *reg;
e = regUp->eUp;
if( VertEq( e->Org, vEvent )) {
assert( TOLERANCE_NONZERO );
SpliceMergeVertices( tess, e, vEvent->anEdge );
return;
}
if( ! VertEq( e->Dst, vEvent )) {
if (tessMeshSplitEdge( tess->mesh, e->Sym ) == NULL) longjmp(tess->env,1);
if( regUp->fixUpperEdge ) {
if ( !tessMeshDelete( tess->mesh, e->Onext ) ) longjmp(tess->env,1);
regUp->fixUpperEdge = FALSE;
}
if ( !tessMeshSplice( tess->mesh, vEvent->anEdge, e ) ) longjmp(tess->env,1);
SweepEvent( tess, vEvent );
return;
}
assert( TOLERANCE_NONZERO );
regUp = TopRightRegion( regUp );
reg = RegionBelow( regUp );
eTopRight = reg->eUp->Sym;
eTopLeft = eLast = eTopRight->Onext;
if( reg->fixUpperEdge ) {
assert( eTopLeft != eTopRight );
DeleteRegion( tess, reg );
if ( !tessMeshDelete( tess->mesh, eTopRight ) ) longjmp(tess->env,1);
eTopRight = eTopLeft->Oprev;
}
if ( !tessMeshSplice( tess->mesh, vEvent->anEdge, eTopRight ) ) longjmp(tess->env,1);
if( ! EdgeGoesLeft( eTopLeft )) {
eTopLeft = NULL;
}
AddRightEdges( tess, regUp, eTopRight->Onext, eLast, eTopLeft, TRUE );
}
static void ConnectLeftVertex( TESStesselator *tess, TESSvertex *vEvent )
{
ActiveRegion *regUp, *regLo, *reg;
TESShalfEdge *eUp, *eLo, *eNew;
ActiveRegion tmp;
tmp.eUp = vEvent->anEdge->Sym;
regUp = (ActiveRegion *)dictKey( dictSearch( tess->dict, &tmp ));
regLo = RegionBelow( regUp );
if( !regLo ) {
return;
}
eUp = regUp->eUp;
eLo = regLo->eUp;
if( EdgeSign( eUp->Dst, vEvent, eUp->Org ) == 0 ) {
ConnectLeftDegenerate( tess, regUp, vEvent );
return;
}
reg = VertLeq( eLo->Dst, eUp->Dst ) ? regUp : regLo;
if( regUp->inside || reg->fixUpperEdge) {
if( reg == regUp ) {
eNew = tessMeshConnect( tess->mesh, vEvent->anEdge->Sym, eUp->Lnext );
if (eNew == NULL) longjmp(tess->env,1);
} else {
TESShalfEdge *tempHalfEdge= tessMeshConnect( tess->mesh, eLo->Dnext, vEvent->anEdge);
if (tempHalfEdge == NULL) longjmp(tess->env,1);
eNew = tempHalfEdge->Sym;
}
if( reg->fixUpperEdge ) {
if ( !FixUpperEdge( tess, reg, eNew ) ) longjmp(tess->env,1);
} else {
ComputeWinding( tess, AddRegionBelow( tess, regUp, eNew ));
}
SweepEvent( tess, vEvent );
} else {
AddRightEdges( tess, regUp, vEvent->anEdge, vEvent->anEdge, NULL, TRUE );
}
}
static void SweepEvent( TESStesselator *tess, TESSvertex *vEvent )
{
ActiveRegion *regUp, *reg;
TESShalfEdge *e, *eTopLeft, *eBottomLeft;
tess->event = vEvent;
DebugEvent( tess );
e = vEvent->anEdge;
while( e->activeRegion == NULL ) {
e = e->Onext;
if( e == vEvent->anEdge ) {
ConnectLeftVertex( tess, vEvent );
return;
}
}
regUp = TopLeftRegion( tess, e->activeRegion );
if (regUp == NULL) longjmp(tess->env,1);
reg = RegionBelow( regUp );
eTopLeft = reg->eUp;
eBottomLeft = FinishLeftRegions( tess, reg, NULL );
if( eBottomLeft->Onext == eTopLeft ) {
ConnectRightVertex( tess, regUp, eBottomLeft );
} else {
AddRightEdges( tess, regUp, eBottomLeft->Onext, eTopLeft, eTopLeft, TRUE );
}
}
static void AddSentinel( TESStesselator *tess, TESSreal smin, TESSreal smax, TESSreal t )
{
TESShalfEdge *e;
ActiveRegion *reg = (ActiveRegion *)bucketAlloc( tess->regionPool );
if (reg == NULL) longjmp(tess->env,1);
e = tessMeshMakeEdge( tess->mesh );
if (e == NULL) longjmp(tess->env,1);
e->Org->s = smax;
e->Org->t = t;
e->Dst->s = smin;
e->Dst->t = t;
tess->event = e->Dst;
reg->eUp = e;
reg->windingNumber = 0;
reg->inside = FALSE;
reg->fixUpperEdge = FALSE;
reg->sentinel = TRUE;
reg->dirty = FALSE;
reg->nodeUp = dictInsert( tess->dict, reg );
if (reg->nodeUp == NULL) longjmp(tess->env,1);
}
static void InitEdgeDict( TESStesselator *tess )
{
TESSreal w, h;
TESSreal smin, smax, tmin, tmax;
tess->dict = dictNewDict( &tess->alloc, tess, (int (*)(void *, DictKey, DictKey)) EdgeLeq );
if (tess->dict == NULL) longjmp(tess->env,1);
w = (tess->bmax[0] - tess->bmin[0]) + (TESSreal)0.01;
h = (tess->bmax[1] - tess->bmin[1]) + (TESSreal)0.01;
smin = tess->bmin[0] - w;
smax = tess->bmax[0] + w;
tmin = tess->bmin[1] - h;
tmax = tess->bmax[1] + h;
AddSentinel( tess, smin, smax, tmin );
AddSentinel( tess, smin, smax, tmax );
}
static void DoneEdgeDict( TESStesselator *tess )
{
ActiveRegion *reg;
int fixedEdges = 0;
while( (reg = (ActiveRegion *)dictKey( dictMin( tess->dict ))) != NULL ) {
if( ! reg->sentinel ) {
assert( reg->fixUpperEdge );
assert( ++fixedEdges == 1 );
}
assert( reg->windingNumber == 0 );
DeleteRegion( tess, reg );
}
dictDeleteDict( &tess->alloc, tess->dict );
}
static void RemoveDegenerateEdges( TESStesselator *tess )
{
TESShalfEdge *e, *eNext, *eLnext;
TESShalfEdge *eHead = &tess->mesh->eHead;
for( e = eHead->next; e != eHead; e = eNext ) {
eNext = e->next;
eLnext = e->Lnext;
if( VertEq( e->Org, e->Dst ) && e->Lnext->Lnext != e ) {
SpliceMergeVertices( tess, eLnext, e );
if ( !tessMeshDelete( tess->mesh, e ) ) longjmp(tess->env,1);
e = eLnext;
eLnext = e->Lnext;
}
if( eLnext->Lnext == e ) {
if( eLnext != e ) {
if( eLnext == eNext || eLnext == eNext->Sym ) { eNext = eNext->next; }
if ( !tessMeshDelete( tess->mesh, eLnext ) ) longjmp(tess->env,1);
}
if( e == eNext || e == eNext->Sym ) { eNext = eNext->next; }
if ( !tessMeshDelete( tess->mesh, e ) ) longjmp(tess->env,1);
}
}
}
static int InitPriorityQ( TESStesselator *tess )
{
PriorityQ *pq;
TESSvertex *v, *vHead;
int vertexCount = 0;
vHead = &tess->mesh->vHead;
for( v = vHead->next; v != vHead; v = v->next ) {
vertexCount++;
}
vertexCount += MAX( 8, tess->alloc.extraVertices );
pq = tess->pq = pqNewPriorityQ( &tess->alloc, vertexCount, (int (*)(PQkey, PQkey)) tesvertLeq );
if (pq == NULL) return 0;
vHead = &tess->mesh->vHead;
for( v = vHead->next; v != vHead; v = v->next ) {
v->pqHandle = pqInsert( &tess->alloc, pq, v );
if (v->pqHandle == INV_HANDLE)
break;
}
if (v != vHead || !pqInit( &tess->alloc, pq ) ) {
pqDeletePriorityQ( &tess->alloc, tess->pq );
tess->pq = NULL;
return 0;
}
return 1;
}
static void DonePriorityQ( TESStesselator *tess )
{
pqDeletePriorityQ( &tess->alloc, tess->pq );
}
static int RemoveDegenerateFaces( TESStesselator *tess, TESSmesh *mesh )
{
TESSface *f, *fNext;
TESShalfEdge *e;
for( f = mesh->fHead.next; f != &mesh->fHead; f = fNext ) {
fNext = f->next;
e = f->anEdge;
assert( e->Lnext != e );
if( e->Lnext->Lnext == e ) {
AddWinding( e->Onext, e );
if ( !tessMeshDelete( tess->mesh, e ) ) return 0;
}
}
return 1;
}
int tessComputeInterior( TESStesselator *tess )
{
TESSvertex *v, *vNext;
RemoveDegenerateEdges( tess );
if ( !InitPriorityQ( tess ) ) return 0;
InitEdgeDict( tess );
while( (v = (TESSvertex *)pqExtractMin( tess->pq )) != NULL ) {
for( ;; ) {
vNext = (TESSvertex *)pqMinimum( tess->pq );
if( vNext == NULL || ! VertEq( vNext, v )) break;
vNext = (TESSvertex *)pqExtractMin( tess->pq );
SpliceMergeVertices( tess, v->anEdge, vNext->anEdge );
}
SweepEvent( tess, v );
}
tess->event = ((ActiveRegion *) dictKey( dictMin( tess->dict )))->eUp->Org;
DebugEvent( tess );
DoneEdgeDict( tess );
DonePriorityQ( tess );
if ( !RemoveDegenerateFaces( tess, tess->mesh ) ) return 0;
tessMeshCheckMesh( tess->mesh );
return 1;
}