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