6 #include <gdk/gdkkeysyms.h>
9 typedef struct _point {
12 typedef struct _edge {
21 typedef struct _polygon {
22 struct _polygon *next, *prev;
28 typedef struct _PolygonView {
31 cairo_surface_t *pixmap;
32 int pixmap_width, pixmap_height;
43 gint mag_drag_x, mag_drag_y;
46 typedef struct _PolygonViewClass {
47 GtkWidgetClass parent_class;
50 G_DEFINE_TYPE (PolygonView, polygon_view, GTK_TYPE_WIDGET)
52 double highlight = -1;
54 static void draw_edges (cairo_t *cr, polygon_t *p, gdouble sf, int dir)
59 cairo_set_source_rgb (cr, 0.0, 0.0, 1.0);
61 cairo_set_source_rgb (cr, 1.0, 0.0, 0.0);
63 for (n = 0; n < p->num_edges; n++) {
64 const edge_t *e = &p->edges[n];
71 dx = e->p2.x - e->p1.x;
72 dy = e->p2.y - e->p1.y;
74 x1 = e->p1.x + (e->top - e->p1.y) / dy * dx;
75 x2 = e->p1.x + (e->bot - e->p1.y) / dy * dx;
77 cairo_arc (cr, x1, e->top, 2/sf, 0, 2*M_PI);
78 cairo_arc (cr, x2, e->bot, 2/sf, 0, 2*M_PI);
83 cairo_set_source_rgba (cr, 0.0, 0.0, 1.0, 0.5);
85 cairo_set_source_rgba (cr, 1.0, 0.0, 0.0, 0.5);
87 for (n = 0; n < p->num_edges; n++) {
88 const edge_t *e = &p->edges[n];
93 cairo_move_to (cr, e->p1.x, e->p1.y);
94 cairo_line_to (cr, e->p2.x, e->p2.y);
97 cairo_identity_matrix (cr);
98 cairo_set_line_width (cr, 1.);
100 } cairo_restore (cr);
103 cairo_set_source_rgb (cr, 0.0, 0.0, 1.0);
105 cairo_set_source_rgb (cr, 1.0, 0.0, 0.0);
107 for (n = 0; n < p->num_edges; n++) {
108 const edge_t *e = &p->edges[n];
115 dx = e->p2.x - e->p1.x;
116 dy = e->p2.y - e->p1.y;
118 x1 = e->p1.x + (e->top - e->p1.y) / dy * dx;
119 x2 = e->p1.x + (e->bot - e->p1.y) / dy * dx;
121 cairo_move_to (cr, x1, e->top);
122 cairo_line_to (cr, x2, e->bot);
125 cairo_identity_matrix (cr);
126 cairo_set_line_width (cr, 1.);
128 } cairo_restore (cr);
131 static void draw_polygon (cairo_t *cr, polygon_t *p, gdouble sf)
133 draw_edges (cr, p, sf, -1);
135 draw_edges (cr, p, sf, +1);
138 static cairo_surface_t *
139 pixmap_create (PolygonView *self, cairo_surface_t *target)
141 cairo_surface_t *surface =
142 cairo_surface_create_similar (target, CAIRO_CONTENT_COLOR,
143 self->widget.allocation.width,
144 self->widget.allocation.height);
145 cairo_t *cr = cairo_create (surface);
147 gdouble sf_x, sf_y, sf;
152 cairo_set_source_rgb (cr, 1, 1, 1);
155 if (self->polygons == NULL) {
160 extents = self->extents;
162 mid = (extents.p2.x + extents.p1.x) / 2.;
163 dim = (extents.p2.x - extents.p1.x) / 2. * 1.25;
164 sf_x = self->widget.allocation.width / dim / 2;
166 mid = (extents.p2.y + extents.p1.y) / 2.;
167 dim = (extents.p2.y - extents.p1.y) / 2. * 1.25;
168 sf_y = self->widget.allocation.height / dim / 2;
170 sf = MIN (sf_x, sf_y);
172 mid = (extents.p2.x + extents.p1.x) / 2.;
173 dim = sf_x / sf * (extents.p2.x - extents.p1.x) / 2. * 1.25;
175 mid = (extents.p2.y + extents.p1.y) / 2.;
176 dim = sf_y / sf * (extents.p2.y - extents.p1.y) / 2. * 1.25;
180 cairo_scale (cr, sf, sf);
181 cairo_translate (cr, -x0, -y0);
183 for (polygon = self->polygons; polygon; polygon = polygon->next) {
184 if (polygon->num_edges == 0)
187 draw_polygon (cr, polygon, sf);
190 if (highlight != -1) {
191 cairo_move_to (cr, extents.p1.x, highlight);
192 cairo_line_to (cr, extents.p2.x, highlight);
193 cairo_set_source_rgb (cr, 0, .7, 0);
195 cairo_identity_matrix (cr);
196 cairo_set_line_width (cr, 1.);
200 } cairo_restore (cr);
207 polygon_view_draw (PolygonView *self, cairo_t *cr)
210 gdouble sf_x, sf_y, sf;
215 extents = self->extents;
217 mid = (extents.p2.x + extents.p1.x) / 2.;
218 dim = (extents.p2.x - extents.p1.x) / 2. * 1.25;
219 sf_x = self->widget.allocation.width / dim / 2;
221 mid = (extents.p2.y + extents.p1.y) / 2.;
222 dim = (extents.p2.y - extents.p1.y) / 2. * 1.25;
223 sf_y = self->widget.allocation.height / dim / 2;
225 sf = MIN (sf_x, sf_y);
227 mid = (extents.p2.x + extents.p1.x) / 2.;
228 dim = sf_x / sf * (extents.p2.x - extents.p1.x) / 2. * 1.25;
230 mid = (extents.p2.y + extents.p1.y) / 2.;
231 dim = sf_y / sf * (extents.p2.y - extents.p1.y) / 2. * 1.25;
234 if (self->pixmap_width != self->widget.allocation.width ||
235 self->pixmap_height != self->widget.allocation.height)
237 cairo_surface_destroy (self->pixmap);
238 self->pixmap = pixmap_create (self, cairo_get_target (cr));
239 self->pixmap_width = self->widget.allocation.width;
240 self->pixmap_height = self->widget.allocation.height;
243 cairo_set_source_surface (cr, self->pixmap, 0, 0);
246 if (self->polygons == NULL)
249 /* draw a zoom view of the area around the mouse */
251 double zoom = self->mag_zoom;
252 int size = self->mag_size;
253 int mag_x = self->mag_x;
254 int mag_y = self->mag_y;
257 if (self->px + size < self->widget.allocation.width/2)
258 mag_x = self->px + size/4;
260 mag_x = self->px - size/4 - size;
261 mag_y = self->py - size/2;
264 if (mag_y + size > self->widget.allocation.height)
265 mag_y = self->widget.allocation.height - size;
270 cairo_rectangle (cr, mag_x, mag_y, size, size);
271 cairo_stroke_preserve (cr);
272 cairo_set_source_rgb (cr, 1, 1, 1);
273 cairo_fill_preserve (cr);
276 /* compute roi in extents */
277 cairo_translate (cr, mag_x + size/2, mag_y + size/2);
280 cairo_scale (cr, zoom, zoom);
281 cairo_translate (cr, -(self->px / sf + x0), -(self->py /sf + y0));
282 for (polygon = self->polygons; polygon; polygon = polygon->next) {
283 if (polygon->num_edges == 0)
286 draw_polygon (cr, polygon, zoom);
289 if (highlight != -1) {
290 cairo_move_to (cr, extents.p1.x, highlight);
291 cairo_line_to (cr, extents.p2.x, highlight);
292 cairo_set_source_rgb (cr, 0, .7, 0);
294 cairo_identity_matrix (cr);
295 cairo_set_line_width (cr, 1.);
299 } cairo_restore (cr);
306 -zoom*fmod (self->px/sf + x0, 1.),
307 -zoom*fmod (self->py/sf + y0, 1.));
309 for (i = -size/2/zoom; i <= size/2/zoom + 1; i+=2) {
310 cairo_move_to (cr, zoom*i, -size/2);
311 cairo_line_to (cr, zoom*i, size/2 + zoom);
312 cairo_move_to (cr, -size/2, zoom*i);
313 cairo_line_to (cr, size/2 + zoom, zoom*i);
316 cairo_set_source_rgba (cr, .7, .7, .7, .5);
317 cairo_set_line_width (cr, 1.);
320 for (i = -size/2/zoom - 1; i <= size/2/zoom + 1; i++) {
321 cairo_move_to (cr, zoom*i, -size/2);
322 cairo_line_to (cr, zoom*i, size/2 + zoom);
323 cairo_move_to (cr, -size/2, zoom*i);
324 cairo_line_to (cr, size/2 + zoom, zoom*i);
326 cairo_set_source_rgba (cr, .1, .1, .1, .5);
327 cairo_set_line_width (cr, 2.);
329 } cairo_restore (cr);
331 } cairo_restore (cr);
336 polygon_view_expose (GtkWidget *w, GdkEventExpose *ev)
338 PolygonView *self = (PolygonView *) w;
341 cr = gdk_cairo_create (w->window);
342 gdk_cairo_region (cr, ev->region);
345 polygon_view_draw (self, cr);
352 polygon_view_key_press (GtkWidget *w, GdkEventKey *ev)
354 switch (ev->keyval) {
365 polygon_view_button_press (GtkWidget *w, GdkEventButton *ev)
367 PolygonView *self = (PolygonView *) w;
369 if (ev->x < self->mag_x ||
370 ev->y < self->mag_y ||
371 ev->x > self->mag_x + self->mag_size ||
372 ev->y > self->mag_y + self->mag_size)
377 self->in_mag_drag = TRUE;
378 self->mag_drag_x = ev->x;
379 self->mag_drag_y = ev->y;
386 polygon_view_button_release (GtkWidget *w, GdkEventButton *ev)
388 PolygonView *self = (PolygonView *) w;
390 self->in_mag_drag = FALSE;
396 polygon_view_update_mouse (PolygonView *self, GdkEventMotion *ev)
401 gtk_widget_queue_draw (&self->widget);
405 polygon_view_update_magnifier (PolygonView *self, gint *xy)
410 gtk_widget_queue_draw (&self->widget);
414 polygon_view_motion (GtkWidget *w, GdkEventMotion *ev)
416 PolygonView *self = (PolygonView *) w;
418 if (self->in_mag_drag) {
421 xy[0] = self->mag_x + ev->x - self->mag_drag_x;
422 xy[1] = self->mag_y + ev->y - self->mag_drag_y;
424 polygon_view_update_magnifier (self, xy);
426 self->mag_drag_x = ev->x;
427 self->mag_drag_y = ev->y;
428 } else if (ev->x < self->mag_x ||
429 ev->y < self->mag_y ||
430 ev->x > self->mag_x + self->mag_size ||
431 ev->y > self->mag_y + self->mag_size)
433 polygon_view_update_mouse (self, ev);
440 polygon_view_realize (GtkWidget *widget)
442 GdkWindowAttr attributes;
444 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
446 attributes.window_type = GDK_WINDOW_CHILD;
447 attributes.x = widget->allocation.x;
448 attributes.y = widget->allocation.y;
449 attributes.width = widget->allocation.width;
450 attributes.height = widget->allocation.height;
451 attributes.wclass = GDK_INPUT_OUTPUT;
452 attributes.visual = gtk_widget_get_visual (widget);
453 attributes.colormap = gtk_widget_get_colormap (widget);
454 attributes.event_mask = gtk_widget_get_events (widget) |
455 GDK_BUTTON_PRESS_MASK |
456 GDK_BUTTON_RELEASE_MASK |
458 GDK_KEY_RELEASE_MASK |
459 GDK_POINTER_MOTION_MASK |
460 GDK_BUTTON_MOTION_MASK |
463 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget),
465 GDK_WA_X | GDK_WA_Y |
466 GDK_WA_VISUAL | GDK_WA_COLORMAP);
467 gdk_window_set_user_data (widget->window, widget);
469 widget->style = gtk_style_attach (widget->style, widget->window);
470 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
474 polygon_view_size_allocate (GtkWidget *w, GdkRectangle *r)
476 PolygonView *self = (PolygonView *) w;
478 GTK_WIDGET_CLASS (polygon_view_parent_class)->size_allocate (w, r);
480 self->mag_x = w->allocation.width - self->mag_size - 10;
481 self->mag_y = w->allocation.height - self->mag_size - 10;
485 polygon_view_finalize (GObject *obj)
487 G_OBJECT_CLASS (polygon_view_parent_class)->finalize (obj);
491 polygon_view_class_init (PolygonViewClass *klass)
493 GObjectClass *object_class = (GObjectClass *) klass;
494 GtkWidgetClass *widget_class = (GtkWidgetClass *) klass;
496 object_class->finalize = polygon_view_finalize;
498 widget_class->realize = polygon_view_realize;
499 widget_class->size_allocate = polygon_view_size_allocate;
500 widget_class->expose_event = polygon_view_expose;
501 widget_class->key_press_event = polygon_view_key_press;
502 widget_class->button_press_event = polygon_view_button_press;
503 widget_class->button_release_event = polygon_view_button_release;
504 widget_class->motion_notify_event = polygon_view_motion;
508 polygon_view_init (PolygonView *self)
511 self->mag_size = 200;
513 self->extents.p1.x = G_MAXDOUBLE;
514 self->extents.p1.y = G_MAXDOUBLE;
515 self->extents.p2.x = -G_MAXDOUBLE;
516 self->extents.p2.y = -G_MAXDOUBLE;
518 GTK_WIDGET_SET_FLAGS (self, GTK_CAN_FOCUS);
522 _polygon_add_edge (PolygonView *view, polygon_t *polygon,
523 point_t *p1, point_t *p2,
524 gdouble top, gdouble bot, int dir)
529 if (top < view->extents.p1.y)
530 view->extents.p1.y = top;
531 if (bot > view->extents.p2.y)
532 view->extents.p2.y = bot;
534 if (p1->x < view->extents.p1.x)
535 view->extents.p1.x = p1->x;
536 if (p1->x > view->extents.p2.x)
537 view->extents.p2.x = p1->x;
538 if (p2->x < view->extents.p1.x)
539 view->extents.p1.x = p2->x;
540 if (p2->x > view->extents.p2.x)
541 view->extents.p2.x = p2->x;
543 if (polygon->num_edges == polygon->size) {
544 int newsize = 2 * polygon->size;
547 newpolygon = g_realloc (polygon,
548 sizeof (polygon_t) + newsize * sizeof (edge_t));
549 if (newpolygon == NULL)
552 polygon = newpolygon;
553 polygon->size = newsize;
555 if (polygon->next != NULL)
556 polygon->next->prev = newpolygon;
557 if (polygon->prev != NULL)
558 polygon->prev->next = newpolygon;
560 view->polygons = newpolygon;
563 polygon->edges[polygon->num_edges].p1 = *p1;
564 polygon->edges[polygon->num_edges].p2 = *p2;
565 polygon->edges[polygon->num_edges].top = top;
566 polygon->edges[polygon->num_edges].bot = bot;
567 polygon->edges[polygon->num_edges].dir = dir;
568 polygon->num_edges++;
574 polygon_new (PolygonView *view)
578 t = g_malloc (sizeof (polygon_t) + 128 * sizeof (edge_t));
580 t->next = view->polygons;
582 view->polygons->prev = t;
592 main (int argc, char **argv)
595 polygon_t *polygon = NULL;
601 gtk_init (&argc, &argv);
603 view = g_object_new (polygon_view_get_type (), NULL);
605 file = fopen (argv[1], "r");
607 while (getline (&line, &len, file) != -1) {
612 if (strncmp (line, "polygon: ", sizeof("polygon: ")-1) == 0) {
613 if (polygon && polygon->num_edges) {
614 g_print ("read polygon with %d edges\n", polygon->num_edges);
616 polygon = polygon_new (view);
617 } else if (polygon == NULL)
618 polygon = polygon_new (view);
619 } else if (sscanf (line, " [%*d] = [(%lf, %lf), (%lf, %lf)], top=%lf, bottom=%lf, dir=%d", &p1.x, &p1.y, &p2.x, &p2.y, &top, &bottom, &dir) == 7) {
620 polygon = _polygon_add_edge (view, polygon, &p1, &p2,
625 if (polygon && polygon->num_edges)
626 g_print ("read polygon with %d edges\n", polygon->num_edges);
628 g_print ("extents=(%lg, %lg), (%lg, %lg)\n",
629 view->extents.p1.x, view->extents.p1.y,
630 view->extents.p2.x, view->extents.p2.y);
635 highlight = atof (argv[2]);
637 window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
638 g_signal_connect (window, "delete-event",
639 G_CALLBACK (gtk_main_quit), NULL);
640 gtk_widget_set_size_request (window, 800, 800);
641 gtk_container_add (GTK_CONTAINER (window), &view->widget);
642 gtk_widget_show_all (window);