Graphviz 13.0.0~dev.20250402.0402
Loading...
Searching...
No Matches
rescale_layout.c
Go to the documentation of this file.
1/*************************************************************************
2 * Copyright (c) 2011 AT&T Intellectual Property
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * https://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors: Details at https://graphviz.org
9 *************************************************************************/
10
12// //
13// This file contains the functions //
14// for distorting the layout. //
15// //
16// Four methods are available: //
17// 1) Uniform denisity - rectilinear //
18// 2) Uniform denisity - polar //
19// 3) Fisheye - rectilinear //
20// 4) Fisheye - Ploar //
21// //
23
24#include <assert.h>
25#include <limits.h>
26#include <stdio.h>
27#include <stdlib.h>
28#include <string.h>
29#include <math.h>
30#include <time.h>
31#include <neatogen/matrix_ops.h>
32#include <neatogen/delaunay.h>
33#include <common/arith.h>
34#include <topfish/hierarchy.h>
35#include <util/alloc.h>
36#include <util/sort.h>
37
38static double *compute_densities(v_data *graph, size_t n, double *x, double *y)
39{
40// compute density of every node by calculating the average edge length in a 2-D layout
41 int j, neighbor;
42 double sum;
43 double *densities = gv_calloc(n, sizeof(double));
44
45 for (size_t i = 0; i < n; i++) {
46 sum = 0;
47 for (j = 1; j < graph[i].nedges; j++) {
48 neighbor = graph[i].edges[j];
49 sum += hypot(x[i] - x[neighbor], y[i] - y[neighbor]);
50 }
51 densities[i] = sum / graph[i].nedges;
52 }
53 return densities;
54}
55
56static double *smooth_vec(double *vec, int *ordering, size_t n, int interval) {
57// smooth 'vec' by setting each components to the average of is 'interval'-neighborhood in 'ordering'
58 assert(interval >= 0);
59 double sum;
60 double *smoothed_vec = gv_calloc(n, sizeof(double));
61 size_t n1 = MIN(1 + (size_t)interval, n);
62 sum = 0;
63 for (size_t i = 0; i < n1; i++) {
64 sum += vec[ordering[i]];
65 }
66
67 size_t len = n1;
68 for (size_t i = 0; i < MIN(n, (size_t)interval); i++) {
69 smoothed_vec[ordering[i]] = sum / (double)len;
70 if (len < n) {
71 sum += vec[ordering[len++]];
72 }
73 }
74 if (n <= (size_t)interval) {
75 return smoothed_vec;
76 }
77
78 for (size_t i = (size_t)interval; i < n - (size_t)interval - 1; i++) {
79 smoothed_vec[ordering[i]] = sum / (double)len;
80 sum +=
81 vec[ordering[i + (size_t)interval + 1]] - vec[ordering[i - (size_t)interval]];
82 }
83 for (size_t i = MAX(n - (size_t)interval - 1, (size_t)interval); i < n; i++) {
84 smoothed_vec[ordering[i]] = sum / (double)len;
85 sum -= vec[ordering[i - (size_t)interval]];
86 len--;
87 }
88 return smoothed_vec;
89}
90
91static int cmp(const void *a, const void *b, void *context) {
92 const int *x = a;
93 const int *y = b;
94 const double *place = context;
95
96 if (place[*x] < place[*y]) {
97 return -1;
98 }
99 if (place[*x] > place[*y]) {
100 return 1;
101 }
102 return 0;
103}
104
105void quicksort_place(double *place, int *ordering, size_t size) {
106 gv_sort(ordering, size, sizeof(ordering[0]), cmp, place);
107}
108
109#define DIST(x1, y1, x2, y2) hypot((x1) - (x2), (y1) - (y2))
110
111static void rescale_layout_polarFocus(v_data *graph, size_t n, double *x_coords,
112 double *y_coords, double x_focus,
113 double y_focus, int interval,
114 double distortion) {
115 // Polar distortion - auxiliary function
116 double *densities = NULL;
117 double *distances = gv_calloc(n, sizeof(double));
118 double *orig_distances = gv_calloc(n, sizeof(double));
119
120 for (size_t i = 0; i < n; i++)
121 {
122 distances[i] = DIST(x_coords[i], y_coords[i], x_focus, y_focus);
123 }
124 assert(n <= INT_MAX);
125 copy_vector((int)n, distances, orig_distances);
126
127 int *ordering = gv_calloc(n, sizeof(int));
128 for (size_t i = 0; i < n; i++)
129 {
130 ordering[i] = (int)i;
131 }
132 quicksort_place(distances, ordering, n);
133
134 densities = compute_densities(graph, n, x_coords, y_coords);
135 double *smoothed_densities = smooth_vec(densities, ordering, n, interval);
136
137 // rescale distances
138 if (distortion < 1.01 && distortion > 0.99)
139 {
140 for (size_t i = 1; i < n; i++)
141 {
142 distances[ordering[i]] = distances[ordering[i - 1]] + (orig_distances[ordering[i]] -
143 orig_distances[ordering
144 [i -
145 1]]) / smoothed_densities[ordering[i]];
146 }
147 } else
148 {
149 // just to make milder behavior:
150 const double factor = (signbit(distortion) ? -1 : 1) * sqrt(fabs(distortion));
151 for (size_t i = 1; i < n; i++)
152 {
153 distances[ordering[i]] =
154 distances[ordering[i - 1]] + (orig_distances[ordering[i]] -
155 orig_distances[ordering
156 [i -
157 1]]) /
158 pow(smoothed_densities[ordering[i]], factor);
159 }
160 }
161
162 // compute new coordinate:
163 for (size_t i = 0; i < n; i++)
164 {
165 const double ratio =
166 orig_distances[i] > 0 ? distances[i] / orig_distances[i] : 0;
167 x_coords[i] = x_focus + (x_coords[i] - x_focus) * ratio;
168 y_coords[i] = y_focus + (y_coords[i] - y_focus) * ratio;
169 }
170
171 free(densities);
172 free(smoothed_densities);
173 free(distances);
174 free(orig_distances);
175 free(ordering);
176}
177
178void
179rescale_layout_polar(double *x_coords, double *y_coords,
180 double *x_foci, double *y_foci, int num_foci,
181 size_t n, int interval, double width,
182 double height, double distortion) {
183 // Polar distortion - main function
184 double minX, maxX, minY, maxY;
185 double aspect_ratio;
186 v_data *graph;
187 double scaleX;
188 double scale_ratio;
189
190 // compute original aspect ratio
191 minX = maxX = x_coords[0];
192 minY = maxY = y_coords[0];
193 for (size_t i = 1; i < n; i++)
194 {
195 minX = fmin(minX, x_coords[i]);
196 minY = fmin(minY, y_coords[i]);
197 maxX = fmax(maxX, x_coords[i]);
198 maxY = fmax(maxY, y_coords[i]);
199 }
200 aspect_ratio = (maxX - minX) / (maxY - minY);
201
202 // construct mutual neighborhood graph
203 assert(n <= INT_MAX);
204 graph = UG_graph(x_coords, y_coords, (int)n);
205
206 if (num_foci == 1)
207 { // accelerate execution of most common case
208 rescale_layout_polarFocus(graph, n, x_coords, y_coords, x_foci[0],
209 y_foci[0], interval, distortion);
210 } else
211 {
212 // average-based rescale
213 double *final_x_coords = gv_calloc(n, sizeof(double));
214 double *final_y_coords = gv_calloc(n, sizeof(double));
215 double *cp_x_coords = gv_calloc(n, sizeof(double));
216 double *cp_y_coords = gv_calloc(n, sizeof(double));
217 assert(n <= INT_MAX);
218 for (int i = 0; i < num_foci; i++) {
219 copy_vector((int)n, x_coords, cp_x_coords);
220 copy_vector((int)n, y_coords, cp_y_coords);
221 rescale_layout_polarFocus(graph, n, cp_x_coords, cp_y_coords,
222 x_foci[i], y_foci[i], interval, distortion);
223 scadd(final_x_coords, (int)n - 1, 1.0 / num_foci, cp_x_coords);
224 scadd(final_y_coords, (int)n - 1, 1.0 / num_foci, cp_y_coords);
225 }
226 copy_vector((int)n, final_x_coords, x_coords);
227 copy_vector((int)n, final_y_coords, y_coords);
228 free(final_x_coords);
229 free(final_y_coords);
230 free(cp_x_coords);
231 free(cp_y_coords);
232 }
233 free(graph[0].edges);
234 free(graph);
235
236 minX = maxX = x_coords[0];
237 minY = maxY = y_coords[0];
238 for (size_t i = 1; i < n; i++) {
239 minX = fmin(minX, x_coords[i]);
240 minY = fmin(minY, y_coords[i]);
241 maxX = fmax(maxX, x_coords[i]);
242 maxY = fmax(maxY, y_coords[i]);
243 }
244
245 // shift points:
246 for (size_t i = 0; i < n; i++) {
247 x_coords[i] -= minX;
248 y_coords[i] -= minY;
249 }
250
251 // rescale x_coords to maintain aspect ratio:
252 scaleX = aspect_ratio * (maxY - minY) / (maxX - minX);
253 for (size_t i = 0; i < n; i++) {
254 x_coords[i] *= scaleX;
255 }
256
257
258 // scale the layout to fit full drawing area:
259 scale_ratio =
260 MIN((width) / (aspect_ratio * (maxY - minY)),
261 (height) / (maxY - minY));
262 for (size_t i = 0; i < n; i++) {
263 x_coords[i] *= scale_ratio;
264 y_coords[i] *= scale_ratio;
265 }
266}
Memory allocation wrappers that exit on failure.
static void * gv_calloc(size_t nmemb, size_t size)
Definition alloc.h:26
#define MIN(a, b)
Definition arith.h:28
v_data * UG_graph(double *x, double *y, int n)
Definition delaunay.c:756
static double len(glCompPoint p)
Definition glutils.c:150
void free(void *)
node NULL
Definition grammar.y:163
Agraph_t * graph(char *name)
Definition gv.cpp:30
#define neighbor(t, i, edim, elist)
Definition make_map.h:41
void scadd(double *vec1, int end, double fac, double *vec2)
Definition matrix_ops.c:220
void copy_vector(int n, const double *source, double *dest)
Definition matrix_ops.c:322
static double * compute_densities(v_data *graph, size_t n, double *x, double *y)
static void rescale_layout_polarFocus(v_data *graph, size_t n, double *x_coords, double *y_coords, double x_focus, double y_focus, int interval, double distortion)
void rescale_layout_polar(double *x_coords, double *y_coords, double *x_foci, double *y_foci, int num_foci, size_t n, int interval, double width, double height, double distortion)
void quicksort_place(double *place, int *ordering, size_t size)
static double * smooth_vec(double *vec, int *ordering, size_t n, int interval)
#define DIST(x1, y1, x2, y2)
static int cmp(const void *a, const void *b, void *context)
qsort with carried along context
static void gv_sort(void *base, size_t nmemb, size_t size, int(*compar)(const void *, const void *, void *), void *arg)
qsort with an extra state parameter, ala qsort_r
Definition sort.h:35
#define MAX(a, b)
Definition write.c:31