39 float **rgb_r,
float **rgb_g,
float **rgb_b){
51 double *colors =
NULL;
52 int n = A0->
m, i, cdim;
55 bool weightedQ =
true;
67 int lightness[] = {0, 100};
71 const double accuracy = 0.01;
87 for (i = 0; i < n; i++){
88 (*rgb_r)[i+1] = (float) colors[cdim*i];
89 (*rgb_g)[i+1] = (float) colors[cdim*i + 1];
90 (*rgb_b)[i+1] = (float) colors[cdim*i + 2];
103 rgb_r++; rgb_b++; rgb_g++;
105 for (i = 0; i < n; i++) rgb_r[i] = u[i];
107 for (i = 0; i < n; i++) rgb_g[i] = u[i];
109 for (i = 0; i < n; i++) rgb_b[i] = u[i];
115 return point_poly_map->
ja[point_poly_map->
ia[ip]];
124 int i, j, *ia, *ja, u, v;
132 assert(
graph->m == n);
133 ia =
D->ia; ja =
D->ja;
139 for (i = 0; i < n; i++){
141 for (j = ia[i]; j < ia[i+1]; j++){
144 if (grouping[u] != grouping[v]){
156 GV_INFO(
"ratio (edges among discontiguous regions vs total edges)=%f", (
double)nbad / ia[n]);
194 double xy[2], yz[2], nxy[2], nyz[2], ymx[2], ymz[2], beta, bot;
197 for (i = 0; i < 2; i++) ymx[i] = y[i] - x[i];
198 for (i = 0; i < 2; i++) ymz[i] = y[i] -
z[i];
199 for (i = 0; i < 2; i++) xy[i] = 0.5*(x[i] + y[i]);
200 for (i = 0; i < 2; i++) yz[i] = 0.5*(y[i] +
z[i]);
205 bot = nyz[0]*(x[0]-y[0])+nyz[1]*(x[1]-y[1]);
207 c[0] = xy[0]; c[1] = xy[1];
210 beta = ((x[0] - y[0])*(xy[0] - yz[0])+(x[1] - y[1])*(xy[1] - yz[1]))/bot;
211 c[0] = yz[0] + beta*nyz[0];
212 c[1] = yz[1] + beta*nyz[1];
231 for (i = 0; i < n; i++){
232 for (j = ia[i]; j < ia[i+1]; j++){
233 if (ja[j] == i)
continue;
234 fprintf(f,
"%d -- %d;\n",i,ja[j]);
242 for (i = 0; i < n; i++){
244 fprintf(f,
"%d [label=\"%s\", pos=\"%lf,%lf\", fontsize=%f];\n",i, labels[i], x[i*
dim], x[i*
dim+1], fsz[i]);
246 fprintf(f,
"%d [label=\"%s\", pos=\"%lf,%lf\"];\n",i, labels[i], x[i*
dim], x[i*
dim+1]);
254static void dot_polygon(
agxbuf *sbuff, doubles_t xp, doubles_t yp,
255 double line_width,
bool fill,
const char *cstring) {
262 strlen(cstring), cstring, strlen(cstring), cstring,
266 size_t len_swidth = (size_t)snprintf(
NULL, 0,
"%f", line_width);
268 " -setlinewidth(%f) L %" PRISIZE_T " ", strlen(cstring), cstring,
269 len_swidth + 14, line_width,
LIST_SIZE(&xp));
275 for (
size_t i = 0; i <
LIST_SIZE(&xp); i++) {
283 double *x_poly,
int *polys_groups,
float *r,
284 float *g,
float *b,
const char *opacity) {
285 int i, j, *ia = polys->
ia, *ja = polys->
ja, *a = polys->
a, npolys = polys->
m, nverts = polys->
n, ipoly,first;
286 const bool fill =
false;
287 const bool use_line = line_width >= 0;
289 agxbuf cstring_buffer = {0};
290 const char *cstring =
"#aaaaaaff";
295 GV_INFO(
"npolys = %d", npolys);
296 first = abs(a[0]); ipoly = first + 1;
297 for (i = 0; i < npolys; i++){
298 for (j = ia[i]; j < ia[i+1]; j++){
299 assert(ja[j] < nverts && ja[j] >= 0);
301 if (abs(a[j]) != ipoly){
304 rgb2hex(r[polys_groups[i]], g[polys_groups[i]], b[polys_groups[i]],
305 &cstring_buffer, opacity);
306 cstring =
agxbuse(&cstring_buffer);
308 dot_polygon(sbuff, xp, yp, line_width, fill, cstring);
317 dot_polygon(sbuff, xp, yp, line_width, fill, line_color);
320 dot_polygon(sbuff, xp, yp, -1,
true, cstring);
330 const char *line_color,
double *x_poly,
int *polys_groups,
331 char **labels,
float *fsz,
float *r,
float *g,
float *b,
334 bool plot_polyQ =
true;
337 if (!r || !g || !b) plot_polyQ =
false;
340 fprintf(f,
"graph map {\n node [margin = 0 width=0.0001 height=0.00001 shape=plaintext];\n graph [outputorder=edgesfirst, bgcolor=\"#dae2ff\"]\n edge [color=\"#55555515\",fontname=\"Helvetica-Bold\"]\n");
355 if (!gr) fprintf(f,
"_background = \"");
360 if (line_width >= 0){
364 fprintf(f,
"%s",
agxbuse(&sbuff));
378 if (!gr) fprintf(f,
"}\n");
399 int i, j, i0, i1, i2, ntri;
403 if (trilist ==
NULL) {
410 for (i = 0; i < ntri; i++) {
411 for (j = 0; j < 3; j++) {
412 (*T)[i].vertices[j] = trilist[i * 3 + j];
414 i0 = (*T)[i].vertices[0]; i1 = (*T)[i].vertices[1]; i2 = (*T)[i].vertices[2];
437 int one = 1, jj, i, j, ig1, ig2;
439 int min_grp, max_grp;
441 min_grp = max_grp = groups[0];
442 for (i = 0; i < n; i++) {
443 max_grp =
MAX(groups[i], max_grp);
444 min_grp =
MIN(groups[i], min_grp);
446 if (min_grp <= 0)
return NULL;
450 for (i = 0; i < n; i++){
453 for (j = ia[i]; j < ia[i+1]; j++){
455 if (i != jj && groups[i] != groups[jj] && groups[jj] != GRP_RANDOM && groups[jj] != GRP_BBOX){
456 ig1 = groups[i]-1; ig2 = groups[jj]-1;
469 int one = 1, jj, i, j;
471 int ncomps, *comps =
NULL;
476 for (i = 0; i < n; i++){
477 for (j = ia[i]; j < ia[i+1]; j++){
479 if (i != jj && groups[i] == groups[jj]){
491 free((*poly_point_map)->ia);
492 free((*poly_point_map)->ja);
493 (*poly_point_map)->ia = comps_ptr;
494 (*poly_point_map)->ja = comps;
495 (*poly_point_map)->nz = (size_t)n;
501 int **polys_groups,
int GRP_RANDOM,
int GRP_BBOX) {
507 int i, *tlist, nz, ipoly, nnt, ii, jj, t1, t2, t, cur, next, nn, j, nlink, sta;
508 int *
elist, edim = 3;
511 int *ie =
E->ia, *je =
E->ja, *e =
E->a;
515 for (i = 0; i < nt; i++) mask[i] = -1;
520 *polys_groups =
gv_calloc(ncomps,
sizeof(
int));
522 for (i = 0; i < nt; i++)
elist[i*edim + 2] = 0;
523 nz = ie[
E->m] - ie[0];
527 for (i = 0; i < ncomps; i++){
529 for (j = comps_ptr[i]; j < comps_ptr[i+1]; j++){
532 (*polys_groups)[i] = groups[ii];
535 if (groups[ii] == GRP_RANDOM || groups[ii] == GRP_BBOX)
continue;
537 for (jj = ie[ii]; jj < ie[ii+1]; jj++){
538 if (groups[je[jj]] != groups[ii] && jj < nz - 1 && je[jj] == je[jj+1]){
542 nlink =
elist[t1*edim + 2]%2;
543 elist[t1*edim + nlink] = t2;
544 elist[t1*edim + 2]++;
546 nlink =
elist[t2*edim + 2]%2;
547 elist[t2*edim + nlink] = t1;
548 elist[t2*edim + 2]++;
550 tlist[nnt++] = t1; tlist[nnt++] = t2;
557 for (j = 0; j < nnt; j++){
560 cur = sta = t; mask[cur] = i;
599 fprintf(stderr,
"cycle (edges): {");
601 fprintf(stderr,
"%d,",cur);
604 fprintf(stderr,
"%d}\n",cur);
607 fprintf(stderr,
"cycle (vertices): ");
615static int same_edge(
int ecur,
int elast,
int *edge_table){
637 int n =
E->
m, *ie =
E->ia, *je =
E->ja, *e =
E->a, ne, i, j, t1, t2, jj, ii;
638 int *cycle, cycle_head = 0;
641 int *edge_cycle_map, NOT_ON_CYCLE = -1;
643 enum {NO_DUPLICATE = -1};
644 int *
elist, edim = 3;
648 int k, duplicate, ee = 0, ecur, enext, eprev, cur, next, nn, nlink,
head, elast = 0, etail, tail, ehead, efirst;
653 edge_table =
gv_calloc(
E->nz * 2,
sizeof(
int));
658 for (i = 0; i < n; i++){
659 for (j = ie[i]; j < ie[i+1]; j++){
660 if (j < ie[n] - ie[0] - 1 && i > je[j] && je[j] == je[j+1]){
665 edge_table[ne*2] = t1;
666 edge_table[ne*2+1] = t2;
671 edge_table[ne*2] = t2;
672 edge_table[ne*2+1] = t1;
682 assert(
E->nz >= (
size_t)ne);
688 edge_cycle_map =
gv_calloc(ne,
sizeof(
int));
690 for (i = 0; i < ne; i++) edge_cycle_map[i] = NOT_ON_CYCLE;
691 for (i = 0; i < ne; i++) emask[i] = -1;
697 for (i = 0; i < nt; i++)
elist[i*edim + 2] = 0;
701 for (i = 0; i < ncomps; i++){
702 if (DEBUG_CYCLE) fprintf(stderr,
"\n ============ comp %d has %d members\n",i, comps_ptr[i+1]-comps_ptr[i]);
703 for (k = comps_ptr[i]; k < comps_ptr[i+1]; k++){
705 duplicate = NO_DUPLICATE;
706 if (DEBUG_CYCLE) fprintf(stderr,
"member = %d has %d neighbors\n",ii, ie[ii+1]-ie[ii]);
707 for (j = ie[ii]; j < ie[ii+1]; j++){
711 if (DEBUG_CYCLE) fprintf(stderr,
" linked with %d using half-edge %d, {head,tail} of the edge = {%d, %d}\n",jj, ee, t1,
edge_tail(ee));
712 nlink =
elist[t1*edim + 2]%2;
713 elist[t1*edim + nlink] = ee;
714 elist[t1*edim + 2]++;
716 if (edge_cycle_map[ee] != NOT_ON_CYCLE) duplicate = ee;
720 if (duplicate == NO_DUPLICATE){
726 edge_cycle_map[ecur] = 1;
729 if (DEBUG_CYCLE) fprintf(stderr,
"NEW CYCLE\n starting with edge %d, {head,tail}={%d,%d}\n", ee,
head, next);
730 while (next !=
head){
736 if (DEBUG_CYCLE) fprintf(stderr,
"cur edge = %d, next edge %d, {head,tail}={%d,%d},\n",ecur, enext,
edge_head(enext),
edge_tail(enext));
743 edge_cycle_map[enext] = 1;
749 if (DEBUG_CYCLE)
cycle_print(ee, cycle,edge_table);
753 ecur = ee = duplicate;
754 while (emask[ecur] == ii){
761 if (DEBUG_CYCLE) fprintf(stderr,
" duplicating edge = %d, starting from the a non-duplicating edge %d, search backwards\n",ee, ecur);
765 while (emask[ecur] == ii){
766 if (DEBUG_CYCLE) fprintf(stderr,
" remove edge %d (%d--%d)\n",ecur,
edge_head(ecur),
edge_tail(ecur));
768 edge_cycle_map[ecur] = NOT_ON_CYCLE;
780 fprintf(stderr,
"remaining (broken) cycle = ");
800 if (DEBUG_CYCLE) fprintf(stderr,
"forwarding now from edge %d = {%d, %d}, try to reach vtx %d, first edge from voro = %d\n",
808 if (
same_edge(ecur, efirst, edge_table)){
809 if (DEBUG_CYCLE) fprintf(stderr,
"this voro cell fill in a hole completely!!!!\n");
812 edge_cycle_map[ecur] = 1;
815 if (DEBUG_CYCLE) fprintf(stderr,
"starting with edge %d, {head,tail}={%d,%d}\n", ecur,
head, next);
816 while (next != tail){
822 if (DEBUG_CYCLE) fprintf(stderr,
"cur edge = %d, next edge %d, {head,tail}={%d,%d},\n",ecur, enext,
edge_head(enext),
edge_tail(enext));
831 edge_cycle_map[enext] = 1;
844 while ((enext =
cycle_next(ecur)) != cycle_head){
845 edge_cycle_map[ecur] = NOT_ON_CYCLE;
850 edge_cycle_map[ecur] = NOT_ON_CYCLE;
865 free(edge_cycle_map);
880 int *comps =
NULL, *comps_ptr =
NULL, ncomps;
881 int GRP_RANDOM, GRP_BBOX;
886 groups =
gv_calloc(n + nrandom,
sizeof(
int));
887 maxgrp = grouping[0];
888 for (i = 0; i < n; i++) {
889 maxgrp =
MAX(maxgrp, grouping[i]);
890 groups[i] = grouping[i];
893 GRP_RANDOM = maxgrp + 1; GRP_BBOX = maxgrp + 2;
894 for (i = n; i < n + nrandom - 4; i++) {
895 groups[i] = GRP_RANDOM;
897 for (i = n + nrandom - 4; i < n + nrandom; i++) {
898 groups[i] = GRP_BBOX;
902 conn_comp(n + nrandom,
E, groups, poly_point_map);
904 ncomps = (*poly_point_map)->m;
905 comps = (*poly_point_map)->ja;
906 comps_ptr = (*poly_point_map)->ia;
910 for (i = ncomps - 1; i >= 0; i--) {
911 if (groups[comps[comps_ptr[i]]] != GRP_RANDOM &&
912 groups[comps[comps_ptr[i]]] != GRP_BBOX)
break;
915 GV_INFO(
"ncomps = %d", ncomps);
918 for (i = 0; i < nt; i++){
919 for (j = 0; j <
dim; j++){
930 polys_groups, GRP_RANDOM, GRP_BBOX);
946 double bounding_box_margin,
int nrandom,
947 int nedgep,
double shore_depth_tol,
int *nverts,
954 double xmax[2],
xmin[2], area, *x = x0;
957 int dim2 = 2, nn = 0;
958 int max_qtree_level = 10;
960 int imin, nzok = 0, nzok0 = 0, nt;
961 double *xran,
point[2];
965 bool INCLUDE_OK_POINTS = include_OK_points;
967 int *grouping = grouping0;
969 int HIGHLIGHT_SET = highlight_cluster;
971 for (j = 0; j < dim2; j++) {
976 for (i = 0; i < n; i++){
977 for (j = 0; j < dim2; j++) {
984 area = boxsize[0]*boxsize[1];
988 }
else if (nrandom < 0){
989 nrandom = -nrandom * n;
990 }
else if (nrandom < 4) {
996 if (shore_depth_tol < 0) shore_depth_tol = sqrt(area/(
double) n);
997 GV_INFO(
"nrandom=%d shore_depth_tol=%.08f", nrandom, shore_depth_tol);
1005 int k, t, np=nedgep;
1007 fprintf(stderr,
"add art np = %d\n",np);
1008 assert(
graph->nz <= INT_MAX);
1009 nz = (int)
graph->nz;
1011 for (i = 0; i < n*
dim; i++) y[i] = x[i];
1012 grouping =
gv_calloc(n + nz * np,
sizeof(
int));
1013 for (i = 0; i < n; i++) grouping[i] = grouping0[i];
1015 for (i = 0; i <
graph->m; i++){
1017 for (j =
graph->ia[i]; j <
graph->ia[i+1]; j++){
1018 if (!HIGHLIGHT_SET || (grouping[i] == grouping[
graph->ja[j]] && grouping[i] == HIGHLIGHT_SET)){
1019 for (t = 0; t < np; t++){
1020 for (k = 0; k <
dim; k++){
1021 y[nz*
dim+k] = t/((double) np)*x[i*
dim+k] + (1-t/((double) np))*x[(
graph->ja[j])*
dim + k];
1023 assert(n + (nz-n)*np + t < n + nz*np && n + (nz-n)*np + t >= 0);
1024 if (t/((
double) np) > 0.5){
1025 grouping[nz] = grouping[i];
1027 grouping[nz] = grouping[
graph->ja[j]];
1034 fprintf(stderr,
"after adding edge points, n:%d->%d\n",n, nz);
1045 for (i = 0; i < dim2; i++) {
1046 if (bounding_box_margin > 0){
1047 xmin[i] -= bounding_box_margin;
1048 xmax[i] += bounding_box_margin;
1049 }
else if (bounding_box_margin < 0) {
1050 xmin[i] -= boxsize[i]*(-bounding_box_margin);
1051 xmax[i] += boxsize[i]*(-bounding_box_margin);
1053 xmin[i] -= fmax(boxsize[i] * 0.2, 2.* shore_depth_tol);
1054 xmax[i] += fmax(boxsize[i] * 0.2, 2 * shore_depth_tol);
1058 double bbm = bounding_box_margin;
1060 fprintf (stderr,
"bounding box margin: %.06f", bbm);
1062 fprintf (stderr,
"bounding box margin: (%.06f * %.06f)", boxsize[0], -bbm);
1064 fprintf(stderr,
"bounding box margin: %.06f",
1065 fmax(boxsize[0] * 0.2, 2 * shore_depth_tol));
1069 const double n1 = floor(
area2 / (shore_depth_tol * shore_depth_tol));
1070 const double n2 = n * floor(
area2 / area);
1071 nrandom = fmax(n1, n2);
1074 xran =
gv_calloc((nrandom + 4) * dim2,
sizeof(
double));
1076 if (INCLUDE_OK_POINTS){
1077 nzok0 = nzok = nrandom - 1;
1078 if (grouping == grouping0) {
1079 int *grouping2 =
gv_calloc(n + nrandom,
sizeof(
int));
1080 memcpy(grouping2, grouping,
sizeof(
int)*n);
1081 grouping = grouping2;
1083 grouping =
gv_recalloc(grouping, n, n + nrandom,
sizeof(
int));
1088 for (i = 0; i < nrandom; i++){
1090 for (j = 0; j < dim2; j++){
1096 if (min > shore_depth_tol){
1097 for (j = 0; j < dim2; j++){
1098 xran[nz*dim2+j] =
point[j];
1101 }
else if (INCLUDE_OK_POINTS && min > shore_depth_tol/10){
1102 for (j = 0; j < dim2; j++){
1103 xran[nzok*dim2+j] =
point[j];
1105 grouping[nn++] = grouping[
imin];
1112 if (
Verbose) fprintf(stderr,
"nn nrandom=%d\n", nrandom);
1114 xran =
gv_calloc(4 * dim2,
sizeof(
double));
1120 for (i = 0; i < dim2; i++)
xmin[i] -= 0.2*(
xmax[i]-
xmin[i]);
1121 for (i = 0; i < dim2; i++)
xmax[i] += 0.2*(
xmax[i]-
xmin[i]);
1123 for (j = 0; j < dim2; j++) xran[i*dim2+j] =
xmin[j];
1125 for (j = 0; j < dim2; j++) xran[i*dim2+j] =
xmax[j];
1127 xran[i*dim2] =
xmin[0]; xran[i*dim2+1] =
xmax[1];
1129 xran[i*dim2] =
xmax[0]; xran[i*dim2+1] =
xmin[1];
1134 if (INCLUDE_OK_POINTS){
1135 xcombined =
gv_calloc((nn + nrandom) * dim2,
sizeof(
double));
1137 xcombined =
gv_calloc((n + nrandom) * dim2,
sizeof(
double));
1139 for (i = 0; i < n; i++) {
1140 for (j = 0; j < dim2; j++) xcombined[i*dim2+j] = x[i*
dim+j];
1142 for (i = 0; i < nrandom; i++) {
1143 for (j = 0; j < dim2; j++) xcombined[(i + nn)*dim2+j] = xran[i*
dim+j];
1146 if (INCLUDE_OK_POINTS){
1147 for (i = 0; i < nn - n; i++) {
1148 for (j = 0; j < dim2; j++) xcombined[(i + n)*dim2+j] = xran[(nzok0 - i)*
dim+j];
1157 if (
Verbose) fprintf(stderr,
" highlight cluster %d, n = %d\n",HIGHLIGHT_SET, n);
1160 for (i = 0; i < n; i++){
1161 if (grouping[i] == HIGHLIGHT_SET){
1163 for (j = 0; j <
dim; j++){
1164 xcombined[nz++] = x[i*
dim+j];
1168 for (i = 0; i < n; i++){
1169 if (grouping[i] != HIGHLIGHT_SET){
1170 for (j = 0; j <
dim; j++){
1171 xcombined[nz++] = x[i*
dim+j];
1175 assert(nz == n*
dim);
1176 for (i = 0; i < nh; i++){
1179 for (i = nh; i < n; i++){
1184 if (
Verbose) fprintf(stderr,
"nh = %d\n",nh);
1189 if (
get_tri(n + nrandom, dim2, xcombined, &nt, &Tp, &
E) != 0) {
1193 get_polygons(n, nrandom, dim2, grouping, nt, Tp,
E, nverts, x_poly,
1194 poly_lines, polys, polys_groups, poly_point_map, country_graph);
1201 if (grouping != grouping0)
free(grouping);
1203 if (x != x0)
free(x);
1207static void add_point(
int *n,
int igrp,
double **x,
int *nmax,
double point[],
int **groups){
1210 int old_nmax = *nmax;
1212 *x =
gv_recalloc(*x, 2 * old_nmax, 2 * *nmax,
sizeof(
double));
1213 *groups =
gv_recalloc(*groups, old_nmax, *nmax,
sizeof(
int));
1216 (*x)[(*n)*2] =
point[0];
1217 (*x)[(*n)*2+1] =
point[1];
1218 (*groups)[*n] = igrp;
1227 for (i = 0; i < n; i++){
1236 int n,
int dim,
double *x,
double *sizes,
1237 int *grouping,
SparseMatrix graph,
double bounding_box_margin,
int nrandom,
int *nart,
int nedgep,
1238 double shore_depth_tol,
1239 int *nverts,
double **x_poly,
1297 int N, nmax, i, j, igrp;
1301 double avgsize[2], avgsz, h[2], p1, p0;
1306 K = round(10 / (1 + n / 400.0));
1310 int maxgp = grouping[0];
1311 int mingp = grouping[0];
1312 for (i = 0; i < n; i++) {
1313 maxgp =
MAX(maxgp, grouping[i]);
1314 mingp =
MIN(mingp, grouping[i]);
1316 fprintf(stderr,
"max grouping - min grouping + 1 = %d\n",maxgp - mingp + 1);
1322 bounding_box_margin, nrandom, nedgep,
1323 shore_depth_tol, nverts, x_poly, poly_lines, polys,
1324 polys_groups, poly_point_map, country_graph,
1331 for (i = 0; i < n; i++){
1332 for (j = 0; j < 2; j++) {
1333 avgsize[j] += sizes[i*
dim+j];
1336 for (i = 0; i < 2; i++) avgsize[i] /= n;
1337 avgsz = 0.5*(avgsize[0] + avgsize[1]);
1338 GV_INFO(
"avgsize = {%f, %f}", avgsize[0], avgsize[1]);
1342 groups =
gv_calloc(n + nmax,
sizeof(
int));
1343 for (i = 0; i < n; i++) {
1344 groups[i] = grouping[i];
1345 for (j = 0; j < 2; j++){
1346 X[i*2+j] = x[i*
dim+j];
1351 if (shore_depth_tol < 0) {
1352 shore_depth_tol = -(shore_depth_tol)*avgsz;
1353 }
else if (shore_depth_tol == 0){
1356 shore_depth_tol = sqrt(area / n);
1357 GV_INFO(
"setting shore length ======%f", shore_depth_tol);
1362 double delta[2] = {0};
1364 delta[0] = .5*avgsize[0]/K;
delta[1] = .5*avgsize[1]/K;
1366 for (i = 0; i < n; i++){
1368 double nadded[2] = {0};
1369 for (j = 0; j < 2; j++) {
1371 nadded[j] = round(K * sizes[i *
dim + j] / avgsz);
1377 h[0] = sizes[i*
dim]/nadded[0];
1381 for (
double k = 0; k < nadded[0] - 1; k++){
1391 for (
double k = 0; k < nadded[0] - 1; k++){
1400 h[1] = sizes[i*
dim + 1]/nadded[1];
1404 for (
double k = 0; k < nadded[1] - 1; k++){
1414 for (
double k = 0; k < nadded[1] - 1; k++){
1425 bounding_box_margin, nrandom, nedgep,
1426 shore_depth_tol, nverts, x_poly, poly_lines, polys,
1427 polys_groups, poly_point_map, country_graph,
void QuadTree_get_nearest(QuadTree qt, double *x, double *ymin, int *imin, double *min)
QuadTree QuadTree_new_from_point_list(int dim, int n, int max_level, double *coord)
void QuadTree_delete(QuadTree q)
SparseMatrix SparseMatrix_distance_matrix(SparseMatrix D0)
SparseMatrix SparseMatrix_from_coordinate_format(SparseMatrix A)
SparseMatrix SparseMatrix_transpose(SparseMatrix A)
SparseMatrix SparseMatrix_symmetrize(SparseMatrix A, bool pattern_symmetric_only)
void SparseMatrix_export(FILE *f, SparseMatrix A)
void SparseMatrix_delete(SparseMatrix A)
SparseMatrix SparseMatrix_get_real_adjacency_matrix_symmetrized(SparseMatrix A)
SparseMatrix SparseMatrix_sort(SparseMatrix A)
int * SparseMatrix_weakly_connected_components(SparseMatrix A0, int *ncomp, int **comps)
SparseMatrix SparseMatrix_from_coordinate_format_not_compacted(SparseMatrix A)
SparseMatrix SparseMatrix_remove_diagonal(SparseMatrix A)
SparseMatrix SparseMatrix_new(int m, int n, size_t nz, int type, int format)
#define SparseMatrix_coordinate_form_add_entry(A, irn, jcn, val)
wrap SparseMatrix_coordinate_form_add_entry_ for type safety
Dynamically expanding string buffers.
static void agxbfree(agxbuf *xb)
free any malloced resources
static int agxbprint(agxbuf *xb, const char *fmt,...)
Printf-style output to an agxbuf.
static WUR char * agxbuse(agxbuf *xb)
Memory allocation wrappers that exit on failure.
static void * gv_recalloc(void *ptr, size_t old_nmemb, size_t new_nmemb, size_t size)
static void * gv_calloc(size_t nmemb, size_t size)
abstract graph C library, Cgraph API
void rgb2hex(float r, float g, float b, agxbuf *cstring, const char *opacity)
void country_graph_coloring(int seed, SparseMatrix A, int **p)
helpers for verbose/debug printing
int * get_triangles(double *x, int n, int *tris)
static double dist(int dim, double *x, double *y)
#define X(prefix, name, str, type, subtype,...)
void vector_float_take(int n, float *v, int m, int *p, float **u)
double distance_cropped(double *x, int dim, int i, int j)
Agsym_t * agattr_text(Agraph_t *g, int kind, char *name, const char *value)
creates or looks up text attributes of a graph
int agwrite(Agraph_t *g, void *chan)
Return 0 on success, EOF on failure.
Agraph_t * graph(char *name)
static int imin(int a, int b)
minimum of two integers
type-generic dynamically expanding list
#define LIST_APPEND(list, item)
#define LIST_IS_EMPTY(list)
#define LIST_GET(list, index)
static int make_map_internal(bool include_OK_points, int n, int dim, double *x0, int *grouping0, SparseMatrix graph, double bounding_box_margin, int nrandom, int nedgep, double shore_depth_tol, int *nverts, double **x_poly, SparseMatrix *poly_lines, SparseMatrix *polys, int **polys_groups, SparseMatrix *poly_point_map, SparseMatrix *country_graph, int highlight_cluster)
static void triangle_center(double x[], double y[], double z[], double c[])
void map_palette_optimal_coloring(char *color_scheme, SparseMatrix A0, float **rgb_r, float **rgb_g, float **rgb_b)
static void plot_dot_edges(FILE *f, SparseMatrix A)
static void get_polygons(int n, int nrandom, int dim, int *grouping, int nt, struct Triangle *Tp, SparseMatrix E, int *nverts, double **x_poly, SparseMatrix *poly_lines, SparseMatrix *polys, int **polys_groups, SparseMatrix *poly_point_map, SparseMatrix *country_graph)
static void add_point(int *n, int igrp, double **x, int *nmax, double point[], int **groups)
static SparseMatrix get_country_graph(int n, SparseMatrix A, int *groups, int GRP_RANDOM, int GRP_BBOX)
static int same_edge(int ecur, int elast, int *edge_table)
static void get_boundingbox(int n, int dim, double *x, double *width, double *bbox)
int make_map_from_rectangle_groups(bool include_OK_points, int n, int dim, double *x, double *sizes, int *grouping, SparseMatrix graph, double bounding_box_margin, int nrandom, int *nart, int nedgep, double shore_depth_tol, int *nverts, double **x_poly, SparseMatrix *poly_lines, SparseMatrix *polys, int **polys_groups, SparseMatrix *poly_point_map, SparseMatrix *country_graph, int highlight_cluster)
static void get_poly_lines(int nt, SparseMatrix E, int ncomps, int *comps_ptr, int *comps, int *groups, SparseMatrix *poly_lines, int **polys_groups, int GRP_RANDOM, int GRP_BBOX)
static int get_tri(int n, int dim, double *x, int *nt, struct Triangle **T, SparseMatrix *E)
static SparseMatrix matrix_add_entry(SparseMatrix A, int i, int j, int val)
static void normal(double v[], double normal[])
static void cycle_print(int head, int *cycle, int *edge_table)
void map_optimal_coloring(int seed, SparseMatrix A, float *rgb_r, float *rgb_g, float *rgb_b)
static void plot_dot_labels(FILE *f, int n, int dim, double *x, char **labels, float *fsz)
static int get_poly_id(int ip, SparseMatrix point_poly_map)
static void get_polygon_solids(int nt, SparseMatrix E, int ncomps, int *comps_ptr, int *comps, SparseMatrix *polys)
static void plot_dot_polygons(agxbuf *sbuff, double line_width, const char *line_color, SparseMatrix polys, double *x_poly, int *polys_groups, float *r, float *g, float *b, const char *opacity)
void plot_dot_map(Agraph_t *gr, int n, int dim, double *x, SparseMatrix polys, SparseMatrix poly_lines, double line_width, const char *line_color, double *x_poly, int *polys_groups, char **labels, float *fsz, float *r, float *g, float *b, const char *opacity, SparseMatrix A, FILE *f)
void improve_contiguity(int n, int dim, int *grouping, SparseMatrix poly_point_map, double *x, SparseMatrix graph)
static void conn_comp(int n, SparseMatrix A, int *groups, SparseMatrix *poly_point_map)
#define neighbor(t, i, edim, elist)
static boxf bbox(Ppoly_t **obsp, int npoly, int *np)
int node_distinct_coloring(const char *color_scheme, int *lightness, bool weightedQ, SparseMatrix A0, double accuracy, int seed, int *cdim0, double **colors)
PATHUTIL_API COORD area2(Ppoint_t, Ppoint_t, Ppoint_t)
int stress_model(int dim, SparseMatrix B, double *x, int maxit_sm)
static point center(point vertex[], size_t n)