7 static AdgCanvas
* sample_canvas (void);
8 static AdgCanvas
* operations_canvas (void);
9 static void to_pdf (AdgWidget
*widget
,
11 static void to_png (AdgWidget
*widget
,
13 static void to_ps (AdgWidget
*widget
,
18 main(gint argc
, gchar
**argv
)
23 GtkWidget
*window
, *sample
, *operations
;
25 gtk_init(&argc
, &argv
);
27 path
= demo_find_data_file("adg-demo.ui");
29 g_print("adg-demo.ui not found!\n");
33 builder
= gtk_builder_new();
36 gtk_builder_add_from_file(builder
, path
, &error
);
38 g_print("%s\n", error
->message
);
42 window
= (GtkWidget
*) gtk_builder_get_object(builder
, "wndMain");
44 sample
= (GtkWidget
*) gtk_builder_get_object(builder
, "areaSample");
45 adg_widget_set_canvas(ADG_WIDGET(sample
), sample_canvas());
47 operations
= (GtkWidget
*) gtk_builder_get_object(builder
, "areaOperations");
48 adg_widget_set_canvas(ADG_WIDGET(operations
), operations_canvas());
51 g_signal_connect(window
, "delete-event",
52 G_CALLBACK(gtk_main_quit
), NULL
);
53 g_signal_connect(gtk_builder_get_object(builder
, "btnQuit"),
54 "clicked", G_CALLBACK(gtk_main_quit
), NULL
);
55 g_signal_connect_swapped(gtk_builder_get_object(builder
, "btnPng"),
56 "clicked", G_CALLBACK(to_png
), sample
);
57 g_signal_connect_swapped(gtk_builder_get_object(builder
, "btnPdf"),
58 "clicked", G_CALLBACK(to_pdf
), sample
);
59 g_signal_connect_swapped(gtk_builder_get_object(builder
, "btnPs"),
60 "clicked", G_CALLBACK(to_ps
), sample
);
62 g_object_unref(builder
);
64 gtk_widget_show_all(window
);
71 /**********************************************
72 * A sample mechanical part example
73 **********************************************/
75 #define SQRT3 1.732050808
78 typedef struct _SampleData SampleData
;
82 gdouble D1
, D2
, D3
, D4
, D5
, D6
, D7
;
84 gdouble LD2
, LD3
, LD5
, LD6
, LD7
;
87 static void sample_get (SampleData
*data
);
88 static AdgPath
*sample_path (const SampleData
*data
);
89 static void sample_add_dimensions (AdgCanvas
*canvas
,
90 const SampleData
*data
);
91 static void sample_add_stuff (AdgCanvas
*canvas
,
92 const SampleData
*data
);
101 AdgContainer
*container
;
103 AdgMatrix transformation
;
106 path
= sample_path(&data
);
107 canvas
= adg_canvas_new();
108 container
= (AdgContainer
*) canvas
;
110 entity
= adg_stroke_new(path
);
111 adg_container_add(container
, entity
);
113 sample_add_dimensions(canvas
, &data
);
114 sample_add_stuff(canvas
, &data
);
116 cairo_matrix_init_translate(&transformation
, 100, 70);
117 cairo_matrix_scale(&transformation
, 6.883, 6.883);
118 cairo_matrix_translate(&transformation
, 0, 10);
119 adg_container_set_model_transformation(container
, &transformation
);
125 sample_get(SampleData
*data
)
146 sample_path(const SampleData
*data
)
150 AdgSegment segment_org
;
152 cairo_matrix_t matrix
;
154 path
= (AdgPath
*) adg_path_new();
156 adg_path_move_to(path
, 0, data
->D1
/ 2);
157 adg_path_line_to(path
, data
->A
- data
->B
- data
->LD2
, data
->D1
/ 2);
158 y
= (data
->D1
- data
->D2
) / 2;
159 adg_path_line_to(path
, data
->A
- data
->B
- data
->LD2
+ y
* SQRT3
, data
->D1
/ 2 - y
);
160 adg_path_line_to(path
, data
->A
- data
->B
, data
->D2
/ 2);
161 adg_path_fillet(path
, 0.4);
162 adg_path_line_to(path
, data
->A
- data
->B
, data
->D3
/ 2);
163 adg_path_chamfer(path
, CHAMFER
, CHAMFER
);
164 adg_path_line_to(path
, data
->A
- data
->B
+ data
->LD3
, data
->D3
/ 2);
165 adg_path_chamfer(path
, CHAMFER
, CHAMFER
);
166 adg_path_line_to(path
, data
->A
- data
->B
+ data
->LD3
, data
->D4
/ 2);
167 adg_path_fillet(path
, data
->RD34
);
168 adg_path_line_to(path
, data
->A
- data
->C
- data
->LD5
, data
->D4
/ 2);
169 y
= (data
->D4
- data
->D5
) / 2;
170 adg_path_line_to(path
, data
->A
- data
->C
- data
->LD5
+ y
, data
->D4
/ 2 - y
);
171 adg_path_line_to(path
, data
->A
- data
->C
, data
->D5
/ 2);
172 adg_path_fillet(path
, 0.2);
173 adg_path_line_to(path
, data
->A
- data
->C
, data
->D6
/ 2);
174 adg_path_fillet(path
, 0.1);
175 adg_path_line_to(path
, data
->A
- data
->C
+ data
->LD6
, data
->D6
/ 2);
176 x
= data
->C
- data
->LD7
- data
->LD6
;
178 adg_path_line_to(path
, data
->A
- data
->C
+ data
->LD6
+ x
, data
->D6
/ 2 - y
);
179 adg_path_line_to(path
, data
->A
- data
->LD7
, data
->D7
/ 2);
180 adg_path_line_to(path
, data
->A
, data
->D7
/ 2);
182 /* Build the rounded shape by duplicating the first segment of
183 * the current path, reflecting it on the y=0 axis, reversing and
184 * joining it the result to the original path */
185 cpml_segment_from_cairo(&segment_org
, adg_path_get_cpml_path(path
));
186 segment
= adg_segment_deep_dup(&segment_org
);
187 cpml_segment_reverse(segment
);
188 adg_matrix_init_reflection(&matrix
, 0);
189 cpml_segment_transform(segment
, &matrix
);
190 segment
->data
[0].header
.type
= CAIRO_PATH_LINE_TO
;
192 adg_path_append_segment(path
, segment
);
196 adg_path_close(path
);
201 sample_add_dimensions(AdgCanvas
*canvas
, const SampleData
*data
)
209 entity
= adg_ldim_new_full_explicit(data
->A
- data
->B
- data
->LD2
, -data
->D1
/ 2, data
->A
- data
->B
,
210 -data
->D3
/ 2 + CHAMFER
, ADG_DIR_UP
, 0,
212 adg_container_add(ADG_CONTAINER(canvas
), entity
);
215 entity
= adg_ldim_new_full_explicit(data
->A
- data
->B
, -data
->D3
/ 2 + CHAMFER
, data
->A
- data
->B
+ data
->LD3
,
216 -data
->D3
/ 2 + CHAMFER
, ADG_DIR_UP
, 0,
218 adg_container_add(ADG_CONTAINER(canvas
), entity
);
223 entity
= adg_ldim_new_full_explicit(data
->A
- data
->B
, data
->D3
/ 2 - CHAMFER
, data
->A
, data
->D7
/ 2,
224 ADG_DIR_DOWN
, 0, data
->D3
/ 2);
225 adg_dim_set_tolerances(ADG_DIM(entity
), "+0.1", NULL
);
226 adg_container_add(ADG_CONTAINER(canvas
), entity
);
229 entity
= adg_ldim_new_full_explicit(0, data
->D1
/ 2, data
->A
, data
->D7
/ 2,
230 ADG_DIR_DOWN
, 0, data
->D3
/ 2);
231 adg_dim_set_tolerances(ADG_DIM(entity
), "+0.05", "-0.05");
232 adg_dim_set_level(ADG_DIM(entity
), 2);
233 adg_container_add(ADG_CONTAINER(canvas
), entity
);
238 x
= data
->A
- data
->B
+ data
->LD3
- CHAMFER
;
239 entity
= adg_ldim_new_full_explicit(x
, -data
->D3
/ 2, x
, data
->D3
/ 2,
240 ADG_DIR_RIGHT
, data
->A
, 0);
241 adg_dim_set_tolerances(ADG_DIM(entity
), "-0.25", NULL
);
242 adg_dim_set_level(ADG_DIM(entity
), 5);
243 adg_container_add(ADG_CONTAINER(canvas
), entity
);
246 x
= data
->A
- data
->C
+ data
->LD6
;
247 entity
= adg_ldim_new_full_explicit(x
, -data
->D6
/ 2, x
, data
->D6
/ 2,
248 ADG_DIR_RIGHT
, data
->A
, 0);
249 adg_dim_set_tolerances(ADG_DIM(entity
), "-0.1", NULL
);
250 adg_dim_set_level(ADG_DIM(entity
), 4);
251 adg_container_add(ADG_CONTAINER(canvas
), entity
);
254 x
= data
->A
- data
->C
- data
->LD5
;
255 entity
= adg_ldim_new_full_explicit(x
, -data
->D4
/ 2, x
, data
->D4
/ 2,
256 ADG_DIR_RIGHT
, data
->A
, 0);
257 adg_dim_set_level(ADG_DIM(entity
), 3);
258 adg_container_add(ADG_CONTAINER(canvas
), entity
);
261 x
= data
->A
- data
->C
;
262 entity
= adg_ldim_new_full_explicit(x
, -data
->D5
/ 2, x
, data
->D5
/ 2,
263 ADG_DIR_RIGHT
, data
->A
, 0);
264 adg_dim_set_tolerances(ADG_DIM(entity
), "-0.1", NULL
);
265 adg_dim_set_level(ADG_DIM(entity
), 2);
266 adg_container_add(ADG_CONTAINER(canvas
), entity
);
269 entity
= adg_ldim_new_full_explicit(data
->A
, -data
->D7
/ 2, data
->A
, data
->D7
/ 2,
270 ADG_DIR_RIGHT
, data
->A
, 0);
271 adg_container_add(ADG_CONTAINER(canvas
), entity
);
276 entity
= adg_ldim_new_full_explicit(0, -data
->D1
/ 2, 0, data
->D1
/ 2,
278 adg_dim_set_tolerances(ADG_DIM(entity
), "+0.05", "-0.05");
279 adg_dim_set_level(ADG_DIM(entity
), 2);
280 adg_container_add(ADG_CONTAINER(canvas
), entity
);
283 y
= (data
->D1
- data
->D2
) / 2;
284 x
= data
->A
- data
->B
- data
->LD2
+ y
* SQRT3
;
285 entity
= adg_ldim_new_full_explicit(x
, -data
->D2
/ 2, x
, data
->D2
/ 2,
287 adg_dim_set_tolerances(ADG_DIM(entity
), "-0.1", NULL
);
288 adg_container_add(ADG_CONTAINER(canvas
), entity
);
292 sample_add_stuff(AdgCanvas
*canvas
, const SampleData
*data
)
296 toy_text
= adg_toy_text_new("Rotate the mouse wheel to zoom in and out");
297 adg_translatable_set_origin_explicit(ADG_TRANSLATABLE(toy_text
),
300 adg_container_add(ADG_CONTAINER(canvas
), toy_text
);
302 toy_text
= adg_toy_text_new("Keep the wheel pressed while dragging the mouse to translate");
303 adg_translatable_set_origin_explicit(ADG_TRANSLATABLE(toy_text
),
306 adg_container_add(ADG_CONTAINER(canvas
), toy_text
);
310 #if defined(CAIRO_HAS_PNG_FUNCTIONS) || defined(CAIRO_HAS_PDF_SURFACE) || defined(CAIRO_HAS_PS_SURFACE)
312 /* Only needed if there is at least one supported surface */
314 file_generated(GtkWidget
*caller
, const gchar
*file
)
319 window
= (GtkWindow
*) gtk_widget_get_toplevel(caller
);
320 dialog
= gtk_message_dialog_new_with_markup(window
, GTK_DIALOG_MODAL
,
323 "The requested operation generated\n"
324 "<b>%s</b> in the current directory.",
326 gtk_window_set_title(GTK_WINDOW(dialog
), "Operation completed");
327 gtk_dialog_run(GTK_DIALOG(dialog
));
328 gtk_widget_destroy(dialog
);
333 #if !defined(CAIRO_HAS_PNG_FUNCTIONS) || !defined(CAIRO_HAS_PDF_SURFACE) || !defined(CAIRO_HAS_PS_SURFACE)
335 /* Only needed if there is a missing surface */
337 missing_feature(GtkWidget
*caller
, const gchar
*feature
)
342 window
= (GtkWindow
*) gtk_widget_get_toplevel(caller
);
343 dialog
= gtk_message_dialog_new(window
, GTK_DIALOG_MODAL
,
346 "The provided cairo library\n"
347 "was compiled with no %s support!",
349 gtk_window_set_title(GTK_WINDOW(dialog
), "Missing feature");
350 gtk_dialog_run(GTK_DIALOG(dialog
));
351 gtk_widget_destroy(dialog
);
357 #ifdef CAIRO_HAS_PNG_FUNCTIONS
360 to_png(AdgWidget
*widget
, GtkWidget
*caller
)
362 cairo_surface_t
*surface
;
365 surface
= cairo_image_surface_create(CAIRO_FORMAT_ARGB32
, 800, 600);
366 cr
= cairo_create(surface
);
367 cairo_surface_destroy(surface
);
369 /* Rendering process */
370 adg_entity_render(ADG_ENTITY(adg_widget_get_canvas(widget
)), cr
);
373 cairo_surface_write_to_png(surface
, "test.png");
376 file_generated(caller
, "test.png");
382 to_png(AdgWidget
*widget
, GtkWidget
*caller
)
384 missing_feature(caller
, "PNG");
389 #ifdef CAIRO_HAS_PDF_SURFACE
391 #include <cairo-pdf.h>
394 to_pdf(AdgWidget
*widget
, GtkWidget
*caller
)
396 cairo_surface_t
*surface
;
399 surface
= cairo_pdf_surface_create("test.pdf", 841, 595);
400 cr
= cairo_create(surface
);
401 cairo_surface_destroy(surface
);
403 adg_entity_render(ADG_ENTITY(adg_widget_get_canvas(widget
)), cr
);
408 file_generated(caller
, "test.pdf");
414 to_pdf(AdgWidget
*widget
, GtkWidget
*caller
)
416 missing_feature(caller
, "PDF");
421 #ifdef CAIRO_HAS_PS_SURFACE
423 #include <cairo-ps.h>
426 to_ps(AdgWidget
*widget
, GtkWidget
*caller
)
428 cairo_surface_t
*surface
;
431 /* Surface creation: A4 size */
432 surface
= cairo_ps_surface_create("test.ps", 841, 595);
433 cairo_ps_surface_dsc_comment(surface
,
434 "%%Title: Automatic Drawing Generation (ADG) demo");
435 cairo_ps_surface_dsc_comment(surface
,
436 "%%Copyright: Copyright (C) 2006-2009 Fontana Nicola");
437 cairo_ps_surface_dsc_comment(surface
, "%%Orientation: Portrait");
439 cairo_ps_surface_dsc_begin_setup(surface
);
441 cairo_ps_surface_dsc_begin_page_setup(surface
);
442 cairo_ps_surface_dsc_comment(surface
,
443 "%%IncludeFeature: *PageSize A4");
445 cr
= cairo_create(surface
);
446 cairo_surface_destroy(surface
);
448 adg_entity_render(ADG_ENTITY(adg_widget_get_canvas(widget
)), cr
);
453 file_generated(caller
, "test.ps");
459 to_ps(AdgWidget
*widget
, GtkWidget
*caller
)
461 missing_feature(caller
, "PostScript");
467 /**********************************************
468 * Test case for basic operations,
469 * such as chamfer and fillet
470 **********************************************/
472 static AdgPath
* operations_chamfer (const AdgPath
*path
,
475 static AdgPath
* operations_fillet (const AdgPath
*path
,
479 operations_canvas(void)
481 AdgPath
*path
, *chamfer_path
, *fillet_path
;
483 AdgContainer
*container
;
485 AdgMatrix transformation
;
487 /* Build the base model */
488 path
= (AdgPath
*) adg_path_new();
490 adg_path_move_to(path
, 2, 0);
491 adg_path_line_to(path
, 0, 5);
492 adg_path_line_to(path
, 2, 2);
493 adg_path_line_to(path
, 0, 8);
494 adg_path_line_to(path
, 2, 8);
495 adg_path_line_to(path
, 2, 10);
496 adg_path_line_to(path
, 3, 10);
497 adg_path_line_to(path
, 10, 9);
498 adg_path_line_to(path
, 5, 5);
499 adg_path_line_to(path
, 3, 0);
500 adg_path_close(path
);
502 /* Build the chamfer model */
503 chamfer_path
= operations_chamfer(path
, 0.25, 0.25);
505 /* Build the fillet model */
506 fillet_path
= operations_fillet(path
, 0.20);
508 /* Create the canvas */
509 canvas
= adg_canvas_new();
511 /* Add the original shape */
512 container
= (AdgContainer
*) adg_container_new();
513 adg_container_add(ADG_CONTAINER(canvas
), ADG_ENTITY(container
));
514 adg_container_add(container
, adg_stroke_new(path
));
515 entity
= adg_toy_text_new("Original shape");
516 adg_translatable_set_origin_explicit(ADG_TRANSLATABLE(entity
),
519 adg_container_add(ADG_CONTAINER(canvas
), entity
);
521 /* Add the shape with 0.25x0.25 chamfer */
522 container
= (AdgContainer
*) adg_container_new();
523 cairo_matrix_init_translate(&transformation
, 15., 0.);
524 adg_container_set_model_transformation(container
, &transformation
);
525 adg_container_add(ADG_CONTAINER(canvas
), ADG_ENTITY(container
));
526 adg_container_add(container
, adg_stroke_new(chamfer_path
));
527 entity
= adg_toy_text_new("Shape with 0.25x0.25 chamfer");
528 adg_translatable_set_origin_explicit(ADG_TRANSLATABLE(entity
),
531 adg_container_add(container
, entity
);
533 /* Add the shape with fillets with 0.20 of radius */
534 container
= (AdgContainer
*) adg_container_new();
535 cairo_matrix_init_translate(&transformation
, 30., 0.);
536 adg_container_set_model_transformation(container
, &transformation
);
537 adg_container_add(ADG_CONTAINER(canvas
), ADG_ENTITY(container
));
538 adg_container_add(container
, adg_stroke_new(fillet_path
));
539 entity
= adg_toy_text_new("Shape with R=20 fillet");
540 adg_translatable_set_origin_explicit(ADG_TRANSLATABLE(entity
),
543 adg_container_add(container
, entity
);
545 /* Set a decent start position and zoom */
546 cairo_matrix_init_translate(&transformation
, 10, -140);
547 cairo_matrix_scale(&transformation
, 15, 15);
548 cairo_matrix_translate(&transformation
, 0, 10);
549 adg_container_set_model_transformation(ADG_CONTAINER(canvas
), &transformation
);
555 operations_chamfer(const AdgPath
*model
, gdouble delta1
, gdouble delta2
)
559 CpmlPrimitive primitive
;
561 path
= (AdgPath
*) adg_path_new();
562 cpml_segment_from_cairo(&segment
, adg_path_get_cpml_path((AdgPath
*) model
));
563 cpml_primitive_from_segment(&primitive
, &segment
);
565 adg_path_move_to(path
, (primitive
.org
)->point
.x
, (primitive
.org
)->point
.y
);
568 adg_path_append_primitive(path
, &primitive
);
569 if (primitive
.data
[0].header
.type
== CAIRO_PATH_LINE_TO
)
570 adg_path_chamfer(path
, delta1
, delta2
);
571 } while (cpml_primitive_next(&primitive
));
577 operations_fillet(const AdgPath
*model
, gdouble radius
)
581 CpmlPrimitive primitive
;
583 path
= (AdgPath
*) adg_path_new();
584 cpml_segment_from_cairo(&segment
, adg_path_get_cpml_path((AdgPath
*) model
));
585 cpml_primitive_from_segment(&primitive
, &segment
);
587 adg_path_move_to(path
, (primitive
.org
)->point
.x
, (primitive
.org
)->point
.y
);
590 adg_path_append_primitive(path
, &primitive
);
591 if (primitive
.data
[0].header
.type
== CAIRO_PATH_LINE_TO
)
592 adg_path_fillet(path
, radius
);
593 } while (cpml_primitive_next(&primitive
));