29 const int k1 = *(
int *)v1;
30 const int k2 = *(
int *)v2;
42 XLabels_t *xlp =
gv_alloc(
sizeof(XLabels_t));
46 fprintf(stderr,
"out of memory\n");
74 double maxx = xlp->params->bb.UR.x, maxy = xlp->params->bb.UR.y;
75 return (
unsigned)floor(log2(round(fmax(maxx, maxy)))) + 1;
111 int x = p.
x, y = p.
y;
114 for (
int i = n - 1; i >= 0; i--) {
115 int xi = (x >> i) & 1;
116 int yi = (y >> i) & 1;
117 s = 4 *
s + 2 * (unsigned)xi +
118 ((
unsigned)xi ^ (unsigned)yi);
121 y = y ^ (x & (yi - 1));
123 x = x ^ (-xi & (yi - 1));
124 y = y ^ (-xi & (yi - 1));
141 double iminx = fmax(r.
boundary[0],
s.boundary[0]);
143 double iminy = fmax(r.
boundary[1],
s.boundary[1]);
145 double imaxx = fmin(r.
boundary[2],
s.boundary[2]);
147 double imaxy = fmin(r.
boundary[3],
s.boundary[3]);
148 return (imaxx - iminx) * (imaxy - iminy);
159 assert(objp1->
sz.
x == 0 && objp1->
sz.
y == 0);
211 if (lp->
set == 0 || clp->set == 0)
213 if ((op->
pos.
x == 0.0 && op->
pos.
y == 0.0) ||
214 (cp->
pos.
x == 0.0 && cp->
pos.
y == 0.0))
245 if (intrsx[i] !=
NULL) {
246 double sa, maxa = 0.0;
253 if (intrsx[i]->lbl) {
257 maxa = fmax(sa, maxa);
275 if (intrsx[i] !=
NULL) {
276 double sa, maxa = 0.0;
283 if (intrsx[i]->lbl) {
287 maxa = fmax(sa, maxa);
304 BestPos_t bp = {.pos = objp->
lbl->
pos};
306 for (
size_t i = 0; i < xlp->n_objs; i++) {
307 if (objp == &xlp->objs[i])
309 if (xlp->objs[i].sz.x > 0 && xlp->objs[i].sz.y > 0)
359 double xincr = (2 * lp->
sz.
x + objp->
sz.
x) / XLXDENOM;
360 double yincr = (2 * lp->
sz.
y + objp->
sz.
y) / XLYDENOM;
377 if (nbp.area < bp.area)
384 if (nbp.area < bp.area)
394 if (nbp.area < bp.area)
401 if (nbp.area < bp.area)
411 if (nbp.area < bp.area)
418 if (nbp.area < bp.area)
425 if (nbp.area < bp.area)
429 if (intrsx[XLPXNY] || intrsx[XLCXNY] || intrsx[XLNXNY] || intrsx[XLPXCY] ||
431 if (!intrsx[XLCXNY] && !intrsx[XLNXNY]) {
439 if (nbp.area < bp.area)
443 if (!intrsx[XLPXCY] && !intrsx[XLPXPY]) {
451 if (nbp.area < bp.area)
460 if (intrsx[XLNXPY] || intrsx[XLCXPY] || intrsx[XLPXPY] || intrsx[XLNXCY] ||
462 if (!intrsx[XLCXPY] && !intrsx[XLPXPY]) {
470 if (nbp.area < bp.area)
474 if (!intrsx[XLNXCY] && !intrsx[XLNXNY]) {
482 if (nbp.area < bp.area)
494 for (
size_t i = 0; i < xlp->n_objs; i++) {
495 HDict_t *hp =
gv_alloc(
sizeof(HDict_t));
497 hp->d.data = &xlp->objs[i];
500 const double x = hp->d.rect.boundary[0] +
501 (hp->d.rect.boundary[2] - hp->d.rect.boundary[0]) / 2;
502 const double y = hp->d.rect.boundary[1] +
503 (hp->d.rect.boundary[3] - hp->d.rect.boundary[1]) / 2;
504 assert(x >= INT_MIN && x <= INT_MAX);
505 assert(y >= INT_MIN && y <= INT_MAX);
506 const point pi = {.
x = (int)x, .y = (
int)y};
517 int size =
dtsize(xlp->hdx), freed = 0;
518 while (
dtsize(xlp->hdx)) {
527 assert(size == freed);
532 for (HDict_t *op =
dtfirst(xlp->hdx); op; op =
dtnext(xlp->hdx, op)) {
534 RTreeInsert(xlp->spdx, op->d.rect, op->d.data, &xlp->spdx->root);
550 XLabels_t *xlp =
xlnew(objs, n_objs, lbls, n_lbls, params);
570 for (
size_t i = 0; i < n_objs; i++) {
571 if (objs[i].lbl == 0)
573 const BestPos_t bp =
xladjust(xlp, &objs[i]);
576 }
else if (bp.area == 0) {
577 objs[i].
lbl->
pos = bp.pos;
579 }
else if (params->
force) {
580 objs[i].
lbl->
pos = bp.pos;
Memory allocation wrappers that exit on failure.
static void * gv_alloc(size_t size)
CDT_API Dtmethod_t * Dtobag
ordered multiset
CDT_API int dtsize(Dt_t *)
CDT_API int dtclose(Dt_t *)
CDT_API Dt_t * dtopen(Dtdisc_t *, Dtmethod_t *)
static NORETURN void graphviz_exit(int status)
void RTreeLeafListFree(LeafList_t *llp)
RTree_t * RTreeOpen(void)
void RTreeClose(RTree_t *rtp)
int RTreeInsert(RTree_t *rtp, Rect_t r, void *data, Node_t **n)
LeafList_t * RTreeSearch(RTree_t *rtp, Node_t *n, Rect_t r)
bool Overlap(const Rect_t r, const Rect_t s)
double boundary[NUMSIDES]
bool force
if true, all labels must be placed
static BestPos_t xlintersections(XLabels_t *xlp, object_t *objp, object_t *intrsx[XLNBR])
static double aabbaabb(Rect_t r, Rect_t s)
static double recordlintrsx(object_t *op, object_t *cp, Rect_t *rp, double a, object_t *intrsx[XLNBR])
int placeLabels(object_t *objs, size_t n_objs, xlabel_t *lbls, size_t n_lbls, label_params_t *params)
static void xlfree(XLabels_t *xlp)
static void xlspdxload(XLabels_t *xlp)
static Rect_t objplp2rect(const object_t *objp)
static Rect_t objp2rect(const object_t *op)
static void xlhdxunload(XLabels_t *xlp)
static double recordointrsx(object_t *op, object_t *cp, Rect_t rp, double a, object_t *intrsx[XLNBR])
static int getintrsxi(object_t *op, object_t *cp)
static bool lblenclosing(object_t *objp, object_t *objp1)
static int xlinitialize(XLabels_t *xlp)
static XLabels_t * xlnew(object_t *objs, size_t n_objs, xlabel_t *lbls, size_t n_lbls, label_params_t *params)
static BestPos_t xladjust(XLabels_t *xlp, object_t *objp)
static Rect_t objplpmks(object_t *objp)
static unsigned int hd_hil_s_from_xy(point p, int n)
static unsigned int xlhorder(XLabels_t *xlp)
static int icompare(void *, void *)
static int xlhdxload(XLabels_t *xlp)