summaryrefslogtreecommitdiffstats
path: root/plug-ins/gfig/gfig.c
diff options
context:
space:
mode:
Diffstat (limited to 'plug-ins/gfig/gfig.c')
-rw-r--r--plug-ins/gfig/gfig.c809
1 files changed, 809 insertions, 0 deletions
diff --git a/plug-ins/gfig/gfig.c b/plug-ins/gfig/gfig.c
new file mode 100644
index 0000000..118930f
--- /dev/null
+++ b/plug-ins/gfig/gfig.c
@@ -0,0 +1,809 @@
+/*
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * This is a plug-in for GIMP.
+ *
+ * Generates images containing vector type drawings.
+ *
+ * Copyright (C) 1997 Andy Thomas alt@picnic.demon.co.uk
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <string.h>
+#include <stdlib.h>
+#include <errno.h>
+
+#ifdef HAVE_UNISTD_H
+#include <unistd.h>
+#endif
+
+#include <glib/gstdio.h>
+
+#include <libgimp/gimp.h>
+#include <libgimp/gimpui.h>
+
+#include "gfig.h"
+#include "gfig-style.h"
+#include "gfig-dialog.h"
+#include "gfig-arc.h"
+#include "gfig-bezier.h"
+#include "gfig-circle.h"
+#include "gfig-dobject.h"
+#include "gfig-ellipse.h"
+#include "gfig-grid.h"
+#include "gfig-line.h"
+#include "gfig-poly.h"
+#include "gfig-preview.h"
+#include "gfig-spiral.h"
+#include "gfig-star.h"
+#include "gfig-stock.h"
+
+#include "libgimp/stdplugins-intl.h"
+
+
+#define GFIG_HEADER "GFIG Version 0.2\n"
+
+static void query (void);
+static void run (const gchar *name,
+ gint nparams,
+ const GimpParam *param,
+ gint *nreturn_vals,
+ GimpParam **return_vals);
+
+
+const GimpPlugInInfo PLUG_IN_INFO =
+{
+ NULL, /* init_proc */
+ NULL, /* quit_proc */
+ query, /* query_proc */
+ run, /* run_proc */
+};
+
+
+gint line_no;
+
+gint obj_show_single = -1; /* -1 all >= 0 object number */
+
+/* Structures etc for the objects */
+/* Points used to draw the object */
+
+GfigObject *obj_creating; /* Object we are creating */
+GfigObject *tmp_line; /* Needed when drawing lines */
+
+gboolean need_to_scale;
+
+static gint load_options (GFigObj *gfig,
+ FILE *fp);
+/* globals */
+
+GfigObjectClass dobj_class[10];
+GFigContext *gfig_context;
+GtkWidget *top_level_dlg;
+GList *gfig_list;
+gdouble org_scale_x_factor, org_scale_y_factor;
+
+
+/* Stuff for the preview bit */
+static gint sel_x, sel_y;
+static gint sel_width, sel_height;
+gint preview_width, preview_height;
+gdouble scale_x_factor, scale_y_factor;
+GdkPixbuf *back_pixbuf = NULL;
+
+MAIN ()
+
+static void
+query (void)
+{
+ static const GimpParamDef args[] =
+ {
+ { GIMP_PDB_INT32, "run-mode", "The run mode { RUN-INTERACTIVE (0), RUN-NONINTERACTIVE (1) }" },
+ { GIMP_PDB_IMAGE, "image", "Input image (unused)" },
+ { GIMP_PDB_DRAWABLE, "drawable", "Input drawable" },
+ { GIMP_PDB_INT32, "dummy", "dummy" }
+ };
+
+ gimp_install_procedure (PLUG_IN_PROC,
+ N_("Create geometric shapes"),
+ "Draw Vector Graphics and paint them onto your images. "
+ "Gfig allows you to draw many types of objects "
+ "including Lines, Circles, Ellipses, Curves, Polygons, "
+ "pointed stars, Bezier curves, and Spirals. "
+ "Objects can be painted using Brushes or other tools"
+ "or filled using colors or patterns. "
+ "Gfig objects can also be used to create selections. ",
+ "Andy Thomas",
+ "Andy Thomas",
+ "1997",
+ N_("_Gfig..."),
+ "RGB*, GRAY*",
+ GIMP_PLUGIN,
+ G_N_ELEMENTS (args), 0,
+ args, NULL);
+
+ gimp_plugin_menu_register (PLUG_IN_PROC, "<Image>/Filters/Render");
+}
+
+static void
+run (const gchar *name,
+ gint nparams,
+ const GimpParam *param,
+ gint *nreturn_vals,
+ GimpParam **return_vals)
+{
+ static GimpParam values[1];
+ gint32 drawable_id;
+ GimpRunMode run_mode;
+ GimpPDBStatusType status = GIMP_PDB_SUCCESS;
+ gint pwidth, pheight;
+
+ INIT_I18N ();
+
+ gfig_context = g_new0 (GFigContext, 1);
+ gfig_context->show_background = TRUE;
+ gfig_context->selected_obj = NULL;
+
+ drawable_id = param[2].data.d_drawable;
+
+ run_mode = param[0].data.d_int32;
+
+ gfig_context->image_id = param[1].data.d_image;
+ gfig_context->drawable_id = drawable_id;
+
+ *nreturn_vals = 1;
+ *return_vals = values;
+
+ values[0].type = GIMP_PDB_STATUS;
+ values[0].data.d_status = status;
+
+ gimp_image_undo_group_start (gfig_context->image_id);
+
+ gimp_context_push ();
+
+ /* TMP Hack - clear any selections */
+ if (! gimp_selection_is_empty (gfig_context->image_id))
+ gimp_selection_none (gfig_context->image_id);
+
+ if (! gimp_drawable_mask_intersect (drawable_id, &sel_x, &sel_y,
+ &sel_width, &sel_height))
+ {
+ gimp_context_pop ();
+
+ gimp_image_undo_group_end (gfig_context->image_id);
+ return;
+ }
+
+ /* Calculate preview size */
+
+ if (sel_width > sel_height)
+ {
+ pwidth = MIN (sel_width, PREVIEW_SIZE);
+ pheight = sel_height * pwidth / sel_width;
+ }
+ else
+ {
+ pheight = MIN (sel_height, PREVIEW_SIZE);
+ pwidth = sel_width * pheight / sel_height;
+ }
+
+
+ preview_width = MAX (pwidth, 2); /* Min size is 2 */
+ preview_height = MAX (pheight, 2);
+
+ org_scale_x_factor = scale_x_factor =
+ (gdouble) sel_width / (gdouble) preview_width;
+ org_scale_y_factor = scale_y_factor =
+ (gdouble) sel_height / (gdouble) preview_height;
+
+ /* initialize */
+ gfig_init_object_classes ();
+
+ switch (run_mode)
+ {
+ case GIMP_RUN_INTERACTIVE:
+ case GIMP_RUN_WITH_LAST_VALS:
+ /*gimp_get_data (PLUG_IN_PROC, &selvals);*/
+ if (! gfig_dialog ())
+ {
+ gimp_image_undo_group_end (gfig_context->image_id);
+
+ return;
+ }
+ break;
+
+ case GIMP_RUN_NONINTERACTIVE:
+ status = GIMP_PDB_CALLING_ERROR;
+ break;
+
+ default:
+ break;
+ }
+
+ gimp_context_pop ();
+
+ gimp_image_undo_group_end (gfig_context->image_id);
+
+ if (run_mode != GIMP_RUN_NONINTERACTIVE)
+ gimp_displays_flush ();
+ else
+#if 0
+ if (run_mode == GIMP_RUN_INTERACTIVE)
+ gimp_set_data (PLUG_IN_PROC, &selvals, sizeof (SelectItVals));
+ else
+#endif /* 0 */
+ {
+ status = GIMP_PDB_EXECUTION_ERROR;
+ }
+
+ values[0].data.d_status = status;
+}
+
+/*
+ Translate SPACE to "\\040", etc.
+ Taken from gflare plugin
+ */
+void
+gfig_name_encode (gchar *dest,
+ gchar *src)
+{
+ gint cnt = MAX_LOAD_LINE - 1;
+
+ while (*src && cnt--)
+ {
+ if (g_ascii_iscntrl (*src) || g_ascii_isspace (*src) || *src == '\\')
+ {
+ sprintf (dest, "\\%03o", *src++);
+ dest += 4;
+ }
+ else
+ *dest++ = *src++;
+ }
+ *dest = '\0';
+}
+
+/*
+ Translate "\\040" to SPACE, etc.
+ */
+void
+gfig_name_decode (gchar *dest,
+ const gchar *src)
+{
+ gint cnt = MAX_LOAD_LINE - 1;
+ guint tmp;
+
+ while (*src && cnt--)
+ {
+ if (*src == '\\' && *(src+1) && *(src+2) && *(src+3))
+ {
+ sscanf (src+1, "%3o", &tmp);
+ *dest++ = tmp;
+ src += 4;
+ }
+ else
+ *dest++ = *src++;
+ }
+ *dest = '\0';
+}
+
+
+/*
+ * Load all gfig, which are founded in gfig-path-list, into gfig_list.
+ * gfig-path-list must be initialized first. (plug_in_parse_gfig_path ())
+ * based on code from Gflare.
+ */
+
+gint
+gfig_list_pos (GFigObj *gfig)
+{
+ GFigObj *g;
+ gint n;
+ GList *tmp;
+
+ n = 0;
+
+ for (tmp = gfig_list; tmp; tmp = g_list_next (tmp))
+ {
+ g = tmp->data;
+
+ if (strcmp (gfig->draw_name, g->draw_name) <= 0)
+ break;
+
+ n++;
+ }
+ return n;
+}
+
+/*
+ * Insert gfigs in alphabetical order
+ */
+
+gint
+gfig_list_insert (GFigObj *gfig)
+{
+ gint n;
+
+ n = gfig_list_pos (gfig);
+
+ gfig_list = g_list_insert (gfig_list, gfig, n);
+
+ return n;
+}
+
+void
+gfig_free (GFigObj *gfig)
+{
+ g_assert (gfig != NULL);
+
+ free_all_objs (gfig->obj_list);
+
+ g_free (gfig->name);
+ g_free (gfig->filename);
+ g_free (gfig->draw_name);
+
+ g_free (gfig);
+}
+
+GFigObj *
+gfig_new (void)
+{
+ return g_new0 (GFigObj, 1);
+}
+
+static void
+gfig_load_objs (GFigObj *gfig,
+ gint load_count,
+ FILE *fp)
+{
+ GfigObject *obj;
+ gchar load_buf[MAX_LOAD_LINE];
+ glong offset;
+ glong offset2;
+ Style style;
+
+ while (load_count-- > 0)
+ {
+ obj = NULL;
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+
+ /* kludge */
+ offset = ftell (fp);
+ gfig_skip_style (&style, fp);
+
+ obj = d_load_object (load_buf, fp);
+
+ if (obj)
+ {
+ add_to_all_obj (gfig, obj);
+ offset2 = ftell (fp);
+ fseek (fp, offset, SEEK_SET);
+ gfig_load_style (&obj->style, fp);
+ fseek (fp, offset2, SEEK_SET);
+ }
+ else
+ {
+ g_message ("Failed to load object, load count = %d", load_count);
+ }
+ }
+}
+
+GFigObj *
+gfig_load (const gchar *filename,
+ const gchar *name)
+{
+ GFigObj *gfig;
+ FILE *fp;
+ gchar load_buf[MAX_LOAD_LINE];
+ gchar str_buf[MAX_LOAD_LINE];
+ gint chk_count;
+ gint load_count = 0;
+ gdouble version;
+ gchar magic1[20];
+ gchar magic2[20];
+
+ g_assert (filename != NULL);
+
+#ifdef DEBUG
+ printf ("Loading %s (%s)\n", filename, name);
+#endif /* DEBUG */
+
+ fp = g_fopen (filename, "rb");
+ if (!fp)
+ {
+ g_message (_("Could not open '%s' for reading: %s"),
+ gimp_filename_to_utf8 (filename), g_strerror (errno));
+ return NULL;
+ }
+
+ gfig = gfig_new ();
+
+ gfig->name = g_strdup (name);
+ gfig->filename = g_strdup (filename);
+
+
+ /* HEADER
+ * draw_name
+ * version
+ * obj_list
+ */
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 1);
+
+ sscanf (load_buf, "%10s %10s %lf", magic1, magic2, &version);
+
+ if (strcmp (magic1, "GFIG") || strcmp (magic2, "Version"))
+ {
+ g_message ("File '%s' is not a gfig file",
+ gimp_filename_to_utf8 (gfig->filename));
+ gfig_free (gfig);
+ fclose (fp);
+ return NULL;
+ }
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+ sscanf (load_buf, "Name: %100s", str_buf);
+ gfig_name_decode (load_buf, str_buf);
+ gfig->draw_name = g_strdup (load_buf);
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+ if (strncmp (load_buf, "Version: ", 9) == 0)
+ gfig->version = g_ascii_strtod (load_buf + 9, NULL);
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+ sscanf (load_buf, "ObjCount: %d", &load_count);
+
+ if (load_options (gfig, fp))
+ {
+ g_message ("File '%s' corrupt file - Line %d Option section incorrect",
+ gimp_filename_to_utf8 (filename), line_no);
+ gfig_free (gfig);
+ fclose (fp);
+ return NULL;
+ }
+
+ if (gfig_load_styles (gfig, fp))
+ {
+ g_message ("File '%s' corrupt file - Line %d Option section incorrect",
+ gimp_filename_to_utf8 (filename), line_no);
+ gfig_free (gfig);
+ fclose (fp);
+ return NULL;
+ }
+
+
+
+ gfig_load_objs (gfig, load_count, fp);
+
+ /* Check count ? */
+
+ chk_count = g_list_length (gfig->obj_list);
+
+ if (chk_count != load_count)
+ {
+ g_message ("File '%s' corrupt file - Line %d Object count to small",
+ gimp_filename_to_utf8 (filename), line_no);
+ gfig_free (gfig);
+ fclose (fp);
+ return NULL;
+ }
+
+ fclose (fp);
+
+ if (!gfig_context->current_obj)
+ gfig_context->current_obj = gfig;
+
+ gfig->obj_status = GFIG_OK;
+
+ return gfig;
+}
+
+void
+save_options (GString *string)
+{
+ /* Save options */
+ g_string_append_printf (string, "<OPTIONS>\n");
+ g_string_append_printf (string, "GridSpacing: %d\n",
+ selvals.opts.gridspacing);
+ if (selvals.opts.gridtype == RECT_GRID)
+ {
+ g_string_append_printf (string, "GridType: RECT_GRID\n");
+ }
+ else if (selvals.opts.gridtype == POLAR_GRID)
+ {
+ g_string_append_printf (string, "GridType: POLAR_GRID\n");
+ }
+ else if (selvals.opts.gridtype == ISO_GRID)
+ {
+ g_string_append_printf (string, "GridType: ISO_GRID\n");
+ }
+ else
+ {
+ /* default to RECT_GRID */
+ g_string_append_printf (string, "GridType: RECT_GRID\n");
+ }
+
+ g_string_append_printf (string, "DrawGrid: %s\n",
+ (selvals.opts.drawgrid) ? "TRUE" : "FALSE");
+ g_string_append_printf (string, "Snap2Grid: %s\n",
+ (selvals.opts.snap2grid) ? "TRUE" : "FALSE");
+ g_string_append_printf (string, "LockOnGrid: %s\n",
+ (selvals.opts.lockongrid) ? "TRUE" : "FALSE");
+ g_string_append_printf (string, "ShowControl: %s\n",
+ (selvals.opts.showcontrol) ? "TRUE" : "FALSE");
+ g_string_append_printf (string, "</OPTIONS>\n");
+}
+
+static void
+gfig_save_obj_start (GfigObject *obj,
+ GString *string)
+{
+ g_string_append_printf (string, "<%s ", obj->class->name);
+ gfig_style_save_as_attributes (&obj->style, string);
+ g_string_append_printf (string, ">\n");
+}
+
+static void
+gfig_save_obj_end (GfigObject *obj,
+ GString *string)
+{
+ g_string_append_printf (string, "</%s>\n",obj->class->name);
+}
+
+static gboolean
+load_bool (gchar *opt_buf,
+ gint *toset)
+{
+ if (!strcmp (opt_buf, "TRUE"))
+ *toset = 1;
+ else if (!strcmp (opt_buf, "FALSE"))
+ *toset = 0;
+ else
+ return TRUE;
+
+ return FALSE;
+}
+
+static gint
+load_options (GFigObj *gfig,
+ FILE *fp)
+{
+ gchar load_buf[MAX_LOAD_LINE];
+ gchar str_buf[MAX_LOAD_LINE];
+ gchar opt_buf[MAX_LOAD_LINE];
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+
+#ifdef DEBUG
+ printf ("load '%s'\n", load_buf);
+#endif /* DEBUG */
+
+ if (strcmp (load_buf, "<OPTIONS>"))
+ return (-1);
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+
+#ifdef DEBUG
+ printf ("opt line '%s'\n", load_buf);
+#endif /* DEBUG */
+
+ while (strcmp (load_buf, "</OPTIONS>"))
+ {
+ /* Get option name */
+#ifdef DEBUG
+ printf ("num = %d\n", sscanf (load_buf, "%255s %255s", str_buf, opt_buf));
+
+ printf ("option %s val %s\n", str_buf, opt_buf);
+#else
+ sscanf (load_buf, "%255s %255s", str_buf, opt_buf);
+#endif /* DEBUG */
+
+ if (!strcmp (str_buf, "GridSpacing:"))
+ {
+ /* Value is decimal */
+ int sp = 0;
+ sp = atoi (opt_buf);
+ if (sp <= 0)
+ return (-1);
+ gfig->opts.gridspacing = sp;
+ }
+ else if (!strcmp (str_buf, "DrawGrid:"))
+ {
+ /* Value is bool */
+ if (load_bool (opt_buf, &gfig->opts.drawgrid))
+ return (-1);
+ }
+ else if (!strcmp (str_buf, "Snap2Grid:"))
+ {
+ /* Value is bool */
+ if (load_bool (opt_buf, &gfig->opts.snap2grid))
+ return (-1);
+ }
+ else if (!strcmp (str_buf, "LockOnGrid:"))
+ {
+ /* Value is bool */
+ if (load_bool (opt_buf, &gfig->opts.lockongrid))
+ return (-1);
+ }
+ else if (!strcmp (str_buf, "ShowControl:"))
+ {
+ /* Value is bool */
+ if (load_bool (opt_buf, &gfig->opts.showcontrol))
+ return (-1);
+ }
+ else if (!strcmp (str_buf, "GridType:"))
+ {
+ /* Value is string */
+ if (!strcmp (opt_buf, "RECT_GRID"))
+ gfig->opts.gridtype = RECT_GRID;
+ else if (!strcmp (opt_buf, "POLAR_GRID"))
+ gfig->opts.gridtype = POLAR_GRID;
+ else if (!strcmp (opt_buf, "ISO_GRID"))
+ gfig->opts.gridtype = ISO_GRID;
+ else
+ return (-1);
+ }
+
+ get_line (load_buf, MAX_LOAD_LINE, fp, 0);
+
+#ifdef DEBUG
+ printf ("opt line '%s'\n", load_buf);
+#endif /* DEBUG */
+ }
+ return (0);
+}
+
+GString *
+gfig_save_as_string (void)
+{
+ GList *objs;
+ gint count;
+ gchar buf[G_ASCII_DTOSTR_BUF_SIZE];
+ gchar conv_buf[MAX_LOAD_LINE * 3 + 1];
+ GString *string;
+
+ string = g_string_new (GFIG_HEADER);
+
+ gfig_name_encode (conv_buf, gfig_context->current_obj->draw_name);
+ g_string_append_printf (string, "Name: %s\n", conv_buf);
+ g_string_append_printf (string, "Version: %s\n",
+ g_ascii_formatd (buf, G_ASCII_DTOSTR_BUF_SIZE, "%f",
+ gfig_context->current_obj->version));
+ objs = gfig_context->current_obj->obj_list;
+
+ count = g_list_length (objs);
+
+ g_string_append_printf (string, "ObjCount: %d\n", count);
+
+ save_options (string);
+
+ gfig_save_styles (string);
+
+ for (objs = gfig_context->current_obj->obj_list;
+ objs;
+ objs = g_list_next (objs))
+ {
+ GfigObject *object = objs->data;
+
+ gfig_save_obj_start (object, string);
+
+ gfig_save_style (&object->style, string);
+
+ if (object->points)
+ d_save_object (object, string);
+
+ gfig_save_obj_end (object, string);
+ }
+
+ return string;
+}
+
+
+gboolean
+gfig_save_as_parasite (void)
+{
+ GimpParasite *parasite;
+ GString *string;
+
+ string = gfig_save_as_string ();
+
+ parasite = gimp_parasite_new ("gfig",
+ GIMP_PARASITE_PERSISTENT |
+ GIMP_PARASITE_UNDOABLE,
+ string->len, string->str);
+
+ g_string_free (string, TRUE);
+
+ if (!gimp_item_attach_parasite (gfig_context->drawable_id, parasite))
+ {
+ g_message (_("Error trying to save figure as a parasite: "
+ "can't attach parasite to drawable."));
+ gimp_parasite_free (parasite);
+ return FALSE;
+ }
+
+ gimp_parasite_free (parasite);
+ return TRUE;
+}
+
+GFigObj *
+gfig_load_from_parasite (void)
+{
+ FILE *fp;
+ gchar *fname;
+ GimpParasite *parasite;
+ GFigObj *gfig;
+
+ parasite = gimp_item_get_parasite (gfig_context->drawable_id, "gfig");
+ if (! parasite)
+ return NULL;
+
+ fname = gimp_temp_name ("gfigtmp");
+
+ fp = g_fopen (fname, "wb");
+ if (!fp)
+ {
+ g_message (_("Error trying to open temporary file '%s' "
+ "for parasite loading: %s"),
+ gimp_filename_to_utf8 (fname), g_strerror (errno));
+ return NULL;
+ }
+
+ fwrite (gimp_parasite_data (parasite),
+ sizeof (guchar),
+ gimp_parasite_data_size (parasite),
+ fp);
+ fclose (fp);
+
+ gimp_parasite_free (parasite);
+
+ gfig = gfig_load (fname, "(none)");
+
+ g_unlink (fname);
+
+ g_free (fname);
+
+ return gfig;
+}
+
+void
+gfig_save_callbk (void)
+{
+ FILE *fp;
+ gchar *savename;
+ GString *string;
+
+ savename = gfig_context->current_obj->filename;
+
+ fp = g_fopen (savename, "w+b");
+
+ if (!fp)
+ {
+ g_message (_("Could not open '%s' for writing: %s"),
+ gimp_filename_to_utf8 (savename), g_strerror (errno));
+ return;
+ }
+
+ string = gfig_save_as_string ();
+
+ fwrite (string->str, string->len, 1, fp);
+
+ if (ferror (fp))
+ g_message ("Failed to write file.");
+ else
+ gfig_context->current_obj->obj_status &= ~(GFIG_MODIFIED | GFIG_READONLY);
+
+ fclose (fp);
+}