1 /* Dia -- an diagram creation/manipulation program
2 * Copyright (C) 1999 Alexander Larsson
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17 */
18
19 #ifdef HAVE_CONFIG_H
20 #include <config.h>
21 #endif
22
23 #include <assert.h>
24 #include <math.h>
25
26 #include "intl.h"
27 #include "object.h"
28 #include "polyshape.h"
29 #include "connectionpoint.h"
30 #include "diarenderer.h"
31 #include "attributes.h"
32 #include "widgets.h"
33 #include "diamenu.h"
34 #include "message.h"
35 #include "properties.h"
36
37 #include "tool-icons.h"
38
39 #include "create.h"
40
41 /*
42 TODO:
43 Have connections be remembered across delete corner
44 Move connections correctly on delete corner
45 Add/remove connection points
46 Fix crashes:)
47 */
48
49 #define DEFAULT_WIDTH 0.15
50
51 typedef struct _Polygon {
52 PolyShape poly;
53
54 Color line_color;
55 LineStyle line_style;
56 Color inner_color;
57 gboolean show_background;
58 real dashlength;
59 real line_width;
60 } Polygon;
61
62 static struct _PolygonProperties {
63 gboolean show_background;
64 } default_properties = { TRUE };
65
66 static ObjectChange* polygon_move_handle(Polygon *polygon, Handle *handle,
67 Point *to, ConnectionPoint *cp,
68 HandleMoveReason reason, ModifierKeys modifiers);
69 static ObjectChange* polygon_move(Polygon *polygon, Point *to);
70 static void polygon_select(Polygon *polygon, Point *clicked_point,
71 DiaRenderer *interactive_renderer);
72 static void polygon_draw(Polygon *polygon, DiaRenderer *renderer);
73 static DiaObject *polygon_create(Point *startpoint,
74 void *user_data,
75 Handle **handle1,
76 Handle **handle2);
77 static real polygon_distance_from(Polygon *polygon, Point *point);
78 static void polygon_update_data(Polygon *polygon);
79 static void polygon_destroy(Polygon *polygon);
80 static DiaObject *polygon_copy(Polygon *polygon);
81
82 static PropDescription *polygon_describe_props(Polygon *polygon);
83 static void polygon_get_props(Polygon *polygon, GPtrArray *props);
84 static void polygon_set_props(Polygon *polygon, GPtrArray *props);
85
86 static void polygon_save(Polygon *polygon, ObjectNode obj_node,
87 const char *filename);
88 static DiaObject *polygon_load(ObjectNode obj_node, int version,
89 const char *filename);
90 static DiaMenu *polygon_get_object_menu(Polygon *polygon, Point *clickedpoint);
91
92 static ObjectTypeOps polygon_type_ops =
93 {
94 (CreateFunc)polygon_create, /* create */
95 (LoadFunc) polygon_load, /* load */
96 (SaveFunc) polygon_save, /* save */
97 (GetDefaultsFunc) NULL,
98 (ApplyDefaultsFunc) NULL,
99 };
100
101 static DiaObjectType polygon_type =
102 {
103 "Standard - Polygon", /* name */
104 0, /* version */
105 (char **) polygon_icon, /* pixmap */
106
107 &polygon_type_ops, /* ops */
108 NULL, /* pixmap_file */
109 0 /* default_user_data */
110 };
111
112 DiaObjectType *_polygon_type = (DiaObjectType *) &polygon_type;
113
114
115 static ObjectOps polygon_ops = {
116 (DestroyFunc) polygon_destroy,
117 (DrawFunc) polygon_draw,
118 (DistanceFunc) polygon_distance_from,
119 (SelectFunc) polygon_select,
120 (CopyFunc) polygon_copy,
121 (MoveFunc) polygon_move,
122 (MoveHandleFunc) polygon_move_handle,
123 (GetPropertiesFunc) object_create_props_dialog,
124 (ApplyPropertiesDialogFunc) object_apply_props_from_dialog,
125 (ObjectMenuFunc) polygon_get_object_menu,
126 (DescribePropsFunc) polygon_describe_props,
127 (GetPropsFunc) polygon_get_props,
128 (SetPropsFunc) polygon_set_props,
129 (TextEditFunc) 0,
130 (ApplyPropertiesListFunc) object_apply_props,
131 };
132
133 static PropDescription polygon_props[] = {
134 POLYSHAPE_COMMON_PROPERTIES,
135 PROP_STD_LINE_WIDTH,
136 PROP_STD_LINE_COLOUR,
137 PROP_STD_LINE_STYLE,
138 PROP_STD_FILL_COLOUR,
139 PROP_STD_SHOW_BACKGROUND,
140 PROP_DESC_END
141 };
142
143 static PropDescription *
polygon_describe_props(Polygon * polygon)144 polygon_describe_props(Polygon *polygon)
145 {
146 if (polygon_props[0].quark == 0)
147 prop_desc_list_calculate_quarks(polygon_props);
148 return polygon_props;
149 }
150
151 static PropOffset polygon_offsets[] = {
152 POLYSHAPE_COMMON_PROPERTIES_OFFSETS,
153 { PROP_STDNAME_LINE_WIDTH, PROP_STDTYPE_LINE_WIDTH, offsetof(Polygon, line_width) },
154 { "line_colour", PROP_TYPE_COLOUR, offsetof(Polygon, line_color) },
155 { "line_style", PROP_TYPE_LINESTYLE,
156 offsetof(Polygon, line_style), offsetof(Polygon, dashlength) },
157 { "fill_colour", PROP_TYPE_COLOUR, offsetof(Polygon, inner_color) },
158 { "show_background", PROP_TYPE_BOOL, offsetof(Polygon, show_background) },
159 { NULL, 0, 0 }
160 };
161
162 static void
polygon_get_props(Polygon * polygon,GPtrArray * props)163 polygon_get_props(Polygon *polygon, GPtrArray *props)
164 {
165 object_get_props_from_offsets(&polygon->poly.object,
166 polygon_offsets, props);
167 }
168
169 static void
polygon_set_props(Polygon * polygon,GPtrArray * props)170 polygon_set_props(Polygon *polygon, GPtrArray *props)
171 {
172 object_set_props_from_offsets(&polygon->poly.object,
173 polygon_offsets, props);
174 polygon_update_data(polygon);
175 }
176
177 static real
polygon_distance_from(Polygon * polygon,Point * point)178 polygon_distance_from(Polygon *polygon, Point *point)
179 {
180 return polyshape_distance_from(&polygon->poly, point, polygon->line_width);
181 }
182
polygon_closest_handle(Polygon * polygon,Point * point)183 static Handle *polygon_closest_handle(Polygon *polygon, Point *point) {
184 return polyshape_closest_handle(&polygon->poly, point);
185 }
186
polygon_closest_segment(Polygon * polygon,Point * point)187 static int polygon_closest_segment(Polygon *polygon, Point *point) {
188 return polyshape_closest_segment(&polygon->poly, point, polygon->line_width);
189 }
190
191 static void
polygon_select(Polygon * polygon,Point * clicked_point,DiaRenderer * interactive_renderer)192 polygon_select(Polygon *polygon, Point *clicked_point,
193 DiaRenderer *interactive_renderer)
194 {
195 polyshape_update_data(&polygon->poly);
196 }
197
198 static ObjectChange*
polygon_move_handle(Polygon * polygon,Handle * handle,Point * to,ConnectionPoint * cp,HandleMoveReason reason,ModifierKeys modifiers)199 polygon_move_handle(Polygon *polygon, Handle *handle,
200 Point *to, ConnectionPoint *cp,
201 HandleMoveReason reason, ModifierKeys modifiers)
202 {
203 assert(polygon!=NULL);
204 assert(handle!=NULL);
205 assert(to!=NULL);
206
207 polyshape_move_handle(&polygon->poly, handle, to, cp, reason, modifiers);
208 polygon_update_data(polygon);
209
210 return NULL;
211 }
212
213
214 static ObjectChange*
polygon_move(Polygon * polygon,Point * to)215 polygon_move(Polygon *polygon, Point *to)
216 {
217 polyshape_move(&polygon->poly, to);
218 polygon_update_data(polygon);
219
220 return NULL;
221 }
222
223 static void
polygon_draw(Polygon * polygon,DiaRenderer * renderer)224 polygon_draw(Polygon *polygon, DiaRenderer *renderer)
225 {
226 DiaRendererClass *renderer_ops = DIA_RENDERER_GET_CLASS (renderer);
227 PolyShape *poly = &polygon->poly;
228 Point *points;
229 int n;
230
231 points = &poly->points[0];
232 n = poly->numpoints;
233
234 renderer_ops->set_linewidth(renderer, polygon->line_width);
235 renderer_ops->set_linestyle(renderer, polygon->line_style);
236 renderer_ops->set_dashlength(renderer, polygon->dashlength);
237 renderer_ops->set_linejoin(renderer, LINEJOIN_MITER);
238 renderer_ops->set_linecaps(renderer, LINECAPS_BUTT);
239
240 if (polygon->show_background)
241 renderer_ops->fill_polygon(renderer, points, n, &polygon->inner_color);
242
243 renderer_ops->draw_polygon(renderer, points, n, &polygon->line_color);
244 }
245
246 static DiaObject *
polygon_create(Point * startpoint,void * user_data,Handle ** handle1,Handle ** handle2)247 polygon_create(Point *startpoint,
248 void *user_data,
249 Handle **handle1,
250 Handle **handle2)
251 {
252 Polygon *polygon;
253 PolyShape *poly;
254 DiaObject *obj;
255 Point defaultx = { 1.0, 0.0 };
256 Point defaulty = { 0.0, 1.0 };
257
258 polygon = g_new0(Polygon, 1);
259 poly = &polygon->poly;
260 obj = &poly->object;
261
262 obj->type = &polygon_type;
263 obj->ops = &polygon_ops;
264
265 if (user_data == NULL) {
266 polyshape_init(poly, 3);
267
268 poly->points[0] = *startpoint;
269 poly->points[1] = *startpoint;
270 point_add(&poly->points[1], &defaultx);
271 poly->points[2] = *startpoint;
272 point_add(&poly->points[2], &defaulty);
273 } else {
274 MultipointCreateData *pcd = (MultipointCreateData *)user_data;
275
276 polyshape_init(poly, pcd->num_points);
277 polyshape_set_points(poly, pcd->num_points, pcd->points);
278 }
279
280 polygon->line_width = attributes_get_default_linewidth();
281 polygon->line_color = attributes_get_foreground();
282 polygon->inner_color = attributes_get_background();
283 attributes_get_default_line_style(&polygon->line_style,
284 &polygon->dashlength);
285 polygon->show_background = default_properties.show_background;
286
287 polygon_update_data(polygon);
288
289 *handle1 = poly->object.handles[0];
290 *handle2 = poly->object.handles[2];
291 return &polygon->poly.object;
292 }
293
294 static void
polygon_destroy(Polygon * polygon)295 polygon_destroy(Polygon *polygon)
296 {
297 polyshape_destroy(&polygon->poly);
298 }
299
300 static DiaObject *
polygon_copy(Polygon * polygon)301 polygon_copy(Polygon *polygon)
302 {
303 Polygon *newpolygon;
304 PolyShape *poly, *newpoly;
305 DiaObject *newobj;
306
307 poly = &polygon->poly;
308
309 newpolygon = g_malloc0(sizeof(Polygon));
310 newpoly = &newpolygon->poly;
311 newobj = &newpoly->object;
312
313 polyshape_copy(poly, newpoly);
314
315 newpolygon->line_color = polygon->line_color;
316 newpolygon->line_width = polygon->line_width;
317 newpolygon->line_style = polygon->line_style;
318 newpolygon->dashlength = polygon->dashlength;
319 newpolygon->inner_color = polygon->inner_color;
320 newpolygon->show_background = polygon->show_background;
321
322 return &newpolygon->poly.object;
323 }
324
325
326 static void
polygon_update_data(Polygon * polygon)327 polygon_update_data(Polygon *polygon)
328 {
329 PolyShape *poly = &polygon->poly;
330 DiaObject *obj = &poly->object;
331 ElementBBExtras *extra = &poly->extra_spacing;
332
333 polyshape_update_data(poly);
334
335 extra->border_trans = polygon->line_width / 2.0;
336 polyshape_update_boundingbox(poly);
337
338 obj->position = poly->points[0];
339 }
340
341 static void
polygon_save(Polygon * polygon,ObjectNode obj_node,const char * filename)342 polygon_save(Polygon *polygon, ObjectNode obj_node,
343 const char *filename)
344 {
345 polyshape_save(&polygon->poly, obj_node);
346
347 if (!color_equals(&polygon->line_color, &color_black))
348 data_add_color(new_attribute(obj_node, "line_color"),
349 &polygon->line_color);
350
351 if (polygon->line_width != 0.1)
352 data_add_real(new_attribute(obj_node, PROP_STDNAME_LINE_WIDTH),
353 polygon->line_width);
354
355 if (!color_equals(&polygon->inner_color, &color_white))
356 data_add_color(new_attribute(obj_node, "inner_color"),
357 &polygon->inner_color);
358
359 data_add_boolean(new_attribute(obj_node, "show_background"),
360 polygon->show_background);
361
362 if (polygon->line_style != LINESTYLE_SOLID)
363 data_add_enum(new_attribute(obj_node, "line_style"),
364 polygon->line_style);
365
366 if (polygon->line_style != LINESTYLE_SOLID &&
367 polygon->dashlength != DEFAULT_LINESTYLE_DASHLEN)
368 data_add_real(new_attribute(obj_node, "dashlength"),
369 polygon->dashlength);
370
371 }
372
373 static DiaObject *
polygon_load(ObjectNode obj_node,int version,const char * filename)374 polygon_load(ObjectNode obj_node, int version, const char *filename)
375 {
376 Polygon *polygon;
377 PolyShape *poly;
378 DiaObject *obj;
379 AttributeNode attr;
380
381 polygon = g_malloc0(sizeof(Polygon));
382
383 poly = &polygon->poly;
384 obj = &poly->object;
385
386 obj->type = &polygon_type;
387 obj->ops = &polygon_ops;
388
389 polyshape_load(poly, obj_node);
390
391 polygon->line_color = color_black;
392 attr = object_find_attribute(obj_node, "line_color");
393 if (attr != NULL)
394 data_color(attribute_first_data(attr), &polygon->line_color);
395
396 polygon->line_width = 0.1;
397 attr = object_find_attribute(obj_node, PROP_STDNAME_LINE_WIDTH);
398 if (attr != NULL)
399 polygon->line_width = data_real(attribute_first_data(attr));
400
401 polygon->inner_color = color_white;
402 attr = object_find_attribute(obj_node, "inner_color");
403 if (attr != NULL)
404 data_color(attribute_first_data(attr), &polygon->inner_color);
405
406 polygon->show_background = TRUE;
407 attr = object_find_attribute(obj_node, "show_background");
408 if (attr != NULL)
409 polygon->show_background = data_boolean( attribute_first_data(attr) );
410
411 polygon->line_style = LINESTYLE_SOLID;
412 attr = object_find_attribute(obj_node, "line_style");
413 if (attr != NULL)
414 polygon->line_style = data_enum(attribute_first_data(attr));
415
416 polygon->dashlength = DEFAULT_LINESTYLE_DASHLEN;
417 attr = object_find_attribute(obj_node, "dashlength");
418 if (attr != NULL)
419 polygon->dashlength = data_real(attribute_first_data(attr));
420
421 polygon_update_data(polygon);
422
423 return &polygon->poly.object;
424 }
425
426 static ObjectChange *
polygon_add_corner_callback(DiaObject * obj,Point * clicked,gpointer data)427 polygon_add_corner_callback (DiaObject *obj, Point *clicked, gpointer data)
428 {
429 Polygon *poly = (Polygon*) obj;
430 int segment;
431 ObjectChange *change;
432
433 segment = polygon_closest_segment(poly, clicked);
434 change = polyshape_add_point(&poly->poly, segment, clicked);
435
436 polygon_update_data(poly);
437 return change;
438 }
439
440 static ObjectChange *
polygon_delete_corner_callback(DiaObject * obj,Point * clicked,gpointer data)441 polygon_delete_corner_callback (DiaObject *obj, Point *clicked, gpointer data)
442 {
443 Handle *handle;
444 int handle_nr, i;
445 Polygon *poly = (Polygon*) obj;
446 ObjectChange *change;
447
448 handle = polygon_closest_handle(poly, clicked);
449
450 for (i = 0; i < obj->num_handles; i++) {
451 if (handle == obj->handles[i]) break;
452 }
453 handle_nr = i;
454 change = polyshape_remove_point(&poly->poly, handle_nr);
455
456 polygon_update_data(poly);
457 return change;
458 }
459
460
461 static DiaMenuItem polygon_menu_items[] = {
462 { N_("Add Corner"), polygon_add_corner_callback, NULL, 1 },
463 { N_("Delete Corner"), polygon_delete_corner_callback, NULL, 1 },
464 };
465
466 static DiaMenu polygon_menu = {
467 "Polygon",
468 sizeof(polygon_menu_items)/sizeof(DiaMenuItem),
469 polygon_menu_items,
470 NULL
471 };
472
473 static DiaMenu *
polygon_get_object_menu(Polygon * polygon,Point * clickedpoint)474 polygon_get_object_menu(Polygon *polygon, Point *clickedpoint)
475 {
476 /* Set entries sensitive/selected etc here */
477 polygon_menu_items[0].active = 1;
478 polygon_menu_items[1].active = polygon->poly.numpoints > 3;
479 return &polygon_menu;
480 }
481