1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
|
/*************************************************************************
* Copyright (c) 2011 AT&T Intellectual Property
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-v10.html
*
* Contributors: Details at https://graphviz.org
*************************************************************************/
#include <neatogen/geometry.h>
#include <neatogen/edges.h>
#include <neatogen/hedges.h>
#include <neatogen/heap.h>
#include <neatogen/voronoi.h>
#include <util/gv_math.h>
#include <util/list.h>
void voronoi(Site *(*nextsite)(void *context), void *context) {
Site *newsite, *bot, *top, *p;
Site *v;
Point newintstar = {0};
char pm;
Halfedge *lbnd, *rbnd, *llbnd, *rrbnd, *bisector;
Edge *e;
pq_t *pq = PQinitialize();
bottomsite = nextsite(context);
el_state_t st = {0};
ELinitialize(&st);
newsite = nextsite(context);
while (1) {
if (!PQempty(pq))
newintstar = PQ_min(pq);
if (newsite != NULL &&
(PQempty(pq) || newsite->coord.y < newintstar.y ||
(newsite->coord.y ==newintstar.y && newsite->coord.x < newintstar.x))) {
/* new site is smallest */
lbnd = ELleftbnd(&st, &newsite->coord);
rbnd = ELright(lbnd);
bot = rightreg(lbnd);
e = gvbisect(bot, newsite, &st.allocated);
bisector = HEcreate(&st, e, le);
ELinsert(lbnd, bisector);
if ((p = hintersect(lbnd, bisector, &st.allocated)) != NULL) {
PQdelete(pq, lbnd);
PQinsert(pq, lbnd, p, ngdist(p, newsite));
}
lbnd = bisector;
bisector = HEcreate(&st, e, re);
ELinsert(lbnd, bisector);
if ((p = hintersect(bisector, rbnd, &st.allocated)) != NULL)
PQinsert(pq, bisector, p, ngdist(p, newsite));
newsite = nextsite(context);
} else if (!PQempty(pq)) {
/* intersection is smallest */
lbnd = PQextractmin(pq);
llbnd = ELleft(lbnd);
rbnd = ELright(lbnd);
rrbnd = ELright(rbnd);
bot = leftreg(lbnd);
top = rightreg(rbnd);
v = lbnd->vertex;
endpoint(lbnd->ELedge, lbnd->ELpm, v, &st.allocated);
endpoint(rbnd->ELedge, rbnd->ELpm, v, &st.allocated);
ELdelete(lbnd);
PQdelete(pq, rbnd);
ELdelete(rbnd);
pm = le;
if (bot->coord.y > top->coord.y) {
SWAP(&bot, &top);
pm = re;
}
e = gvbisect(bot, top, &st.allocated);
bisector = HEcreate(&st, e, pm);
ELinsert(llbnd, bisector);
endpoint(e, re - pm, v, &st.allocated);
if ((p = hintersect(llbnd, bisector, &st.allocated)) != NULL) {
PQdelete(pq, llbnd);
PQinsert(pq, llbnd, p, ngdist(p, bot));
}
if ((p = hintersect(bisector, rrbnd, &st.allocated)) != NULL) {
PQinsert(pq, bisector, p, ngdist(p, bot));
}
} else
break;
}
for (lbnd = ELright(st.leftend); lbnd != st.rightend;
lbnd = ELright(lbnd)) {
e = lbnd->ELedge;
clip_line(e);
}
ELcleanup(&st);
// `PQcleanup` relies on the number of sites, so should be discarded and
// at least every time we use `vAdjust`. See note in adjust.c:cleanup().
PQcleanup(pq);
}
|