diff options
Diffstat (limited to 'plug-ins/fractal-explorer/fractal-explorer.c')
-rw-r--r-- | plug-ins/fractal-explorer/fractal-explorer.c | 1201 |
1 files changed, 1201 insertions, 0 deletions
diff --git a/plug-ins/fractal-explorer/fractal-explorer.c b/plug-ins/fractal-explorer/fractal-explorer.c new file mode 100644 index 0000000..3322464 --- /dev/null +++ b/plug-ins/fractal-explorer/fractal-explorer.c @@ -0,0 +1,1201 @@ +/********************************************************************** + Fractal Explorer Plug-in (Version 2.00 Beta 2) + Daniel Cotting (cotting@multimania.com) + ********************************************************************** + ********************************************************************** + Official homepages: http://www.multimania.com/cotting + http://cotting.citeweb.net + *********************************************************************/ + +/********************************************************************** + GIMP - The GNU Image Manipulation Program + Copyright (C) 1995 Spencer Kimball and Peter Mattis + + 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/>. + *********************************************************************/ + +/********************************************************************** + Some code has been 'stolen' from: + - Peter Kirchgessner (Pkirchg@aol.com) + - Scott Draves (spot@cs.cmu.edu) + - Andy Thomas (alt@picnic.demon.co.uk) + . + . + . + ********************************************************************** + "If you steal from one author it's plagiarism; if you steal from + many it's research." --Wilson Mizner + *********************************************************************/ + + +/********************************************************************** + Include necessary files + *********************************************************************/ + +#include "config.h" + +#include <errno.h> +#include <stdlib.h> +#include <string.h> + +#ifdef HAVE_UNISTD_H +#include <unistd.h> +#endif + +#include <glib/gstdio.h> + +#ifdef G_OS_WIN32 +#include <libgimpbase/gimpwin32-io.h> +#endif + +#include <gtk/gtk.h> + +#include <libgimp/gimp.h> +#include <libgimp/gimpui.h> + +#include "fractal-explorer.h" +#include "fractal-explorer-dialogs.h" + +#include "libgimp/stdplugins-intl.h" + + +/********************************************************************** + Global variables + *********************************************************************/ + +gdouble xmin = -2; +gdouble xmax = 1; +gdouble ymin = -1.5; +gdouble ymax = 1.5; +gdouble xbild; +gdouble ybild; +gdouble xdiff; +gdouble ydiff; +gint sel_x; +gint sel_y; +gint preview_width; +gint preview_height; +gdouble *gg; +gint line_no; +gchar *filename; +clrmap colormap; +vlumap valuemap; +gchar *fractalexplorer_path = NULL; + +static gfloat cx = -0.75; +static gfloat cy = -0.2; +gint32 drawable_id; +static GList *fractalexplorer_list = NULL; + +explorer_interface_t wint = +{ + NULL, /* preview */ + NULL, /* wimage */ + FALSE /* run */ +}; /* wint */ + +explorer_vals_t wvals = +{ + 0, + -2.0, + 2.0, + -1.5, + 1.5, + 50.0, + -0.75, + -0.2, + 0, + 1.0, + 1.0, + 1.0, + 1, + 1, + 0, + 0, + 0, + 0, + 1, + 256, + 0, + 0 +}; /* wvals */ + +fractalexplorerOBJ *current_obj = NULL; +static GtkWidget *delete_dialog = NULL; + +static void query (void); +static void run (const gchar *name, + gint nparams, + const GimpParam *param, + gint *nreturn_vals, + GimpParam **return_vals); + +static void explorer (gint32 drawable_id); + +/********************************************************************** + Declare local functions + *********************************************************************/ + +/* Functions for dialog widgets */ + +static void delete_dialog_callback (GtkWidget *widget, + gboolean value, + gpointer data); +static gboolean delete_fractal_callback (GtkWidget *widget, + gpointer data); +static gint fractalexplorer_list_pos (fractalexplorerOBJ *feOBJ); +static gint fractalexplorer_list_insert (fractalexplorerOBJ *feOBJ); +static fractalexplorerOBJ *fractalexplorer_new (void); +static void fill_list_store (GtkListStore *list_store); +static void activate_fractal (fractalexplorerOBJ *sel_obj); +static void activate_fractal_callback (GtkTreeView *view, + GtkTreePath *path, + GtkTreeViewColumn *col, + gpointer data); +static gboolean apply_fractal_callback (GtkWidget *widget, + gpointer data); + +static void fractalexplorer_free (fractalexplorerOBJ *feOBJ); +static void fractalexplorer_free_everything (fractalexplorerOBJ *feOBJ); +static void fractalexplorer_list_free_all (void); +static fractalexplorerOBJ * fractalexplorer_load (const gchar *filename, + const gchar *name); + +static void fractalexplorer_list_load_all (const gchar *path); +static void fractalexplorer_rescan_list (GtkWidget *widget, + gpointer data); + +const GimpPlugInInfo PLUG_IN_INFO = +{ + NULL, /* init_proc */ + NULL, /* quit_proc */ + query, /* query_proc */ + run, /* run_proc */ +}; + +/********************************************************************** + MAIN() + *********************************************************************/ + +MAIN() + +/********************************************************************** + FUNCTION: query + *********************************************************************/ + +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" }, + { GIMP_PDB_DRAWABLE, "drawable", "Input drawable" }, + { GIMP_PDB_INT8, "fractaltype", "0: Mandelbrot; " + "1: Julia; " + "2: Barnsley 1; " + "3: Barnsley 2; " + "4: Barnsley 3; " + "5: Spider; " + "6: ManOWar; " + "7: Lambda; " + "8: Sierpinski" }, + { GIMP_PDB_FLOAT, "xmin", "xmin fractal image delimiter" }, + { GIMP_PDB_FLOAT, "xmax", "xmax fractal image delimiter" }, + { GIMP_PDB_FLOAT, "ymin", "ymin fractal image delimiter" }, + { GIMP_PDB_FLOAT, "ymax", "ymax fractal image delimiter" }, + { GIMP_PDB_FLOAT, "iter", "Iteration value" }, + { GIMP_PDB_FLOAT, "cx", "cx value ( only Julia)" }, + { GIMP_PDB_FLOAT, "cy", "cy value ( only Julia)" }, + { GIMP_PDB_INT8, "colormode", "0: Apply colormap as specified by the parameters below; " + "1: Apply active gradient to final image" }, + { GIMP_PDB_FLOAT, "redstretch", "Red stretching factor" }, + { GIMP_PDB_FLOAT, "greenstretch", "Green stretching factor" }, + { GIMP_PDB_FLOAT, "bluestretch", "Blue stretching factor" }, + { GIMP_PDB_INT8, "redmode", "Red application mode (0:SIN;1:COS;2:NONE)" }, + { GIMP_PDB_INT8, "greenmode", "Green application mode (0:SIN;1:COS;2:NONE)" }, + { GIMP_PDB_INT8, "bluemode", "Blue application mode (0:SIN;1:COS;2:NONE)" }, + { GIMP_PDB_INT8, "redinvert", "Red inversion mode (1: enabled; 0: disabled)" }, + { GIMP_PDB_INT8, "greeninvert", "Green inversion mode (1: enabled; 0: disabled)" }, + { GIMP_PDB_INT8, "blueinvert", "Green inversion mode (1: enabled; 0: disabled)" }, + { GIMP_PDB_INT32, "ncolors", "Number of Colors for mapping (2<=ncolors<=8192)" } + }; + + gimp_install_procedure (PLUG_IN_PROC, + N_("Render fractal art"), + "No help yet.", + "Daniel Cotting (cotting@multimania.com, www.multimania.com/cotting)", + "Daniel Cotting (cotting@multimania.com, www.multimania.com/cotting)", + "December, 1998", + N_("_Fractal Explorer..."), + "RGB*, GRAY*", + GIMP_PLUGIN, + G_N_ELEMENTS (args), 0, + args, NULL); + + gimp_plugin_menu_register (PLUG_IN_PROC, "<Image>/Filters/Render/Fractals"); +} + +/********************************************************************** + FUNCTION: run + *********************************************************************/ + +static void +run (const gchar *name, + gint nparams, + const GimpParam *param, + gint *nreturn_vals, + GimpParam **return_vals) +{ + static GimpParam values[1]; + GimpRunMode run_mode; + gint pwidth; + gint pheight; + GimpPDBStatusType status = GIMP_PDB_SUCCESS; + gint sel_width; + gint sel_height; + + values[0].type = GIMP_PDB_STATUS; + values[0].data.d_status = status; + + *nreturn_vals = 1; + *return_vals = values; + + INIT_I18N (); + gegl_init (NULL, NULL); + + run_mode = param[0].data.d_int32; + drawable_id = param[2].data.d_drawable; + + if (! gimp_drawable_mask_intersect (drawable_id, + &sel_x, &sel_y, + &sel_width, &sel_height)) + 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); + preview_height = MAX (pheight, 2); + + /* See how we will run */ + switch (run_mode) + { + case GIMP_RUN_INTERACTIVE: + /* Possibly retrieve data */ + gimp_get_data ("plug_in_fractalexplorer", &wvals); + + /* Get information from the dialog */ + if (!explorer_dialog ()) + return; + + break; + + case GIMP_RUN_NONINTERACTIVE: + /* Make sure all the arguments are present */ + if (nparams != 22) + { + status = GIMP_PDB_CALLING_ERROR; + } + else + { + wvals.fractaltype = param[3].data.d_int8; + wvals.xmin = param[4].data.d_float; + wvals.xmax = param[5].data.d_float; + wvals.ymin = param[6].data.d_float; + wvals.ymax = param[7].data.d_float; + wvals.iter = param[8].data.d_float; + wvals.cx = param[9].data.d_float; + wvals.cy = param[10].data.d_float; + wvals.colormode = param[11].data.d_int8; + wvals.redstretch = param[12].data.d_float; + wvals.greenstretch = param[13].data.d_float; + wvals.bluestretch = param[14].data.d_float; + wvals.redmode = param[15].data.d_int8; + wvals.greenmode = param[16].data.d_int8; + wvals.bluemode = param[17].data.d_int8; + wvals.redinvert = param[18].data.d_int8; + wvals.greeninvert = param[19].data.d_int8; + wvals.blueinvert = param[20].data.d_int8; + wvals.ncolors = CLAMP (param[21].data.d_int32, 2, MAXNCOLORS); + } + make_color_map(); + break; + + case GIMP_RUN_WITH_LAST_VALS: + /* Possibly retrieve data */ + gimp_get_data ("plug_in_fractalexplorer", &wvals); + make_color_map (); + break; + + default: + break; + } + + xmin = wvals.xmin; + xmax = wvals.xmax; + ymin = wvals.ymin; + ymax = wvals.ymax; + cx = wvals.cx; + cy = wvals.cy; + + if (status == GIMP_PDB_SUCCESS) + { + gimp_progress_init (_("Rendering fractal")); + + explorer (drawable_id); + + if (run_mode != GIMP_RUN_NONINTERACTIVE) + gimp_displays_flush (); + + /* Store data */ + if (run_mode == GIMP_RUN_INTERACTIVE) + gimp_set_data ("plug_in_fractalexplorer", + &wvals, sizeof (explorer_vals_t)); + } + + values[0].data.d_status = status; +} + +/********************************************************************** + FUNCTION: explorer + *********************************************************************/ + +static void +explorer (gint32 drawable_id) +{ + GeglBuffer *src_buffer; + GeglBuffer *dest_buffer; + const Babl *format; + gint width; + gint height; + gint bpp; + gint row; + gint x; + gint y; + gint w; + gint h; + guchar *src_row; + guchar *dest_row; + + /* Get the input area. This is the bounding box of the selection in + * the image (or the entire image if there is no selection). Only + * operating on the input area is simply an optimization. It doesn't + * need to be done for correct operation. (It simply makes it go + * faster, since fewer pixels need to be operated on). + */ + if (! gimp_drawable_mask_intersect (drawable_id, &x, &y, &w, &h)) + return; + + /* Get the size of the input image. (This will/must be the same + * as the size of the output image. + */ + width = gimp_drawable_width (drawable_id); + height = gimp_drawable_height (drawable_id); + + if (gimp_drawable_has_alpha (drawable_id)) + format = babl_format ("R'G'B'A u8"); + else + format = babl_format ("R'G'B' u8"); + + bpp = babl_format_get_bytes_per_pixel (format); + + /* allocate row buffers */ + src_row = g_new (guchar, bpp * w); + dest_row = g_new (guchar, bpp * w); + + src_buffer = gimp_drawable_get_buffer (drawable_id); + dest_buffer = gimp_drawable_get_shadow_buffer (drawable_id); + + xbild = width; + ybild = height; + xdiff = (xmax - xmin) / xbild; + ydiff = (ymax - ymin) / ybild; + + for (row = y; row < y + h; row++) + { + gegl_buffer_get (src_buffer, GEGL_RECTANGLE (x, row, w, 1), 1.0, + format, src_row, + GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_NONE); + + explorer_render_row (src_row, + dest_row, + row, + w, + bpp); + + gegl_buffer_set (dest_buffer, GEGL_RECTANGLE (x, row, w, 1), 0, + format, dest_row, + GEGL_AUTO_ROWSTRIDE); + + if ((row % 10) == 0) + gimp_progress_update ((double) row / (double) h); + } + + g_object_unref (src_buffer); + g_object_unref (dest_buffer); + + g_free (src_row); + g_free (dest_row); + + gimp_progress_update (1.0); + + /* update the processed region */ + gimp_drawable_merge_shadow (drawable_id, TRUE); + gimp_drawable_update (drawable_id, x, y, w, h); +} + +/********************************************************************** + FUNCTION: explorer_render_row + *********************************************************************/ + +void +explorer_render_row (const guchar *src_row, + guchar *dest_row, + gint row, + gint row_width, + gint bpp) +{ + gint col; + gdouble a; + gdouble b; + gdouble x; + gdouble y; + gdouble oldx; + gdouble oldy; + gdouble tempsqrx; + gdouble tempsqry; + gdouble tmpx = 0; + gdouble tmpy = 0; + gdouble foldxinitx; + gdouble foldxinity; + gdouble foldyinitx; + gdouble foldyinity; + gdouble xx = 0; + gdouble adjust; + gdouble cx; + gdouble cy; + gint counter; + gint color; + gint iteration; + gint useloglog; + gdouble log2; + + cx = wvals.cx; + cy = wvals.cy; + useloglog = wvals.useloglog; + iteration = wvals.iter; + log2 = log (2.0); + + for (col = 0; col < row_width; col++) + { + a = xmin + (double) col * xdiff; + b = ymin + (double) row * ydiff; + if (wvals.fractaltype != 0) + { + tmpx = x = a; + tmpy = y = b; + } + else + { + x = 0; + y = 0; + } + + for (counter = 0; counter < iteration; counter++) + { + oldx=x; + oldy=y; + + switch (wvals.fractaltype) + { + case TYPE_MANDELBROT: + xx = x * x - y * y + a; + y = 2.0 * x * y + b; + break; + + case TYPE_JULIA: + xx = x * x - y * y + cx; + y = 2.0 * x * y + cy; + break; + + case TYPE_BARNSLEY_1: + foldxinitx = oldx * cx; + foldyinity = oldy * cy; + foldxinity = oldx * cy; + foldyinitx = oldy * cx; + /* orbit calculation */ + if (oldx >= 0) + { + xx = (foldxinitx - cx - foldyinity); + y = (foldyinitx - cy + foldxinity); + } + else + { + xx = (foldxinitx + cx - foldyinity); + y = (foldyinitx + cy + foldxinity); + } + break; + + case TYPE_BARNSLEY_2: + foldxinitx = oldx * cx; + foldyinity = oldy * cy; + foldxinity = oldx * cy; + foldyinitx = oldy * cx; + /* orbit calculation */ + if (foldxinity + foldyinitx >= 0) + { + xx = foldxinitx - cx - foldyinity; + y = foldyinitx - cy + foldxinity; + } + else + { + xx = foldxinitx + cx - foldyinity; + y = foldyinitx + cy + foldxinity; + } + break; + + case TYPE_BARNSLEY_3: + foldxinitx = oldx * oldx; + foldyinity = oldy * oldy; + foldxinity = oldx * oldy; + /* orbit calculation */ + if (oldx > 0) + { + xx = foldxinitx - foldyinity - 1.0; + y = foldxinity * 2; + } + else + { + xx = foldxinitx - foldyinity -1.0 + cx * oldx; + y = foldxinity * 2; + y += cy * oldx; + } + break; + + case TYPE_SPIDER: + /* { c=z=pixel: z=z*z+c; c=c/2+z, |z|<=4 } */ + xx = x*x - y*y + tmpx + cx; + y = 2 * oldx * oldy + tmpy +cy; + tmpx = tmpx/2 + xx; + tmpy = tmpy/2 + y; + break; + + case TYPE_MAN_O_WAR: + xx = x*x - y*y + tmpx + cx; + y = 2.0 * x * y + tmpy + cy; + tmpx = oldx; + tmpy = oldy; + break; + + case TYPE_LAMBDA: + tempsqrx = x * x; + tempsqry = y * y; + tempsqrx = oldx - tempsqrx + tempsqry; + tempsqry = -(oldy * oldx); + tempsqry += tempsqry + oldy; + xx = cx * tempsqrx - cy * tempsqry; + y = cx * tempsqry + cy * tempsqrx; + break; + + case TYPE_SIERPINSKI: + xx = oldx + oldx; + y = oldy + oldy; + if (oldy > .5) + y = y - 1; + else if (oldx > .5) + xx = xx - 1; + break; + + default: + break; + } + + x = xx; + + if (((x * x) + (y * y)) >= 4.0) + break; + } + + if (useloglog) + { + gdouble modulus_square = (x * x) + (y * y); + + if (modulus_square > (G_E * G_E)) + adjust = log (log (modulus_square) / 2.0) / log2; + else + adjust = 0.0; + } + else + { + adjust = 0.0; + } + + color = (int) (((counter - adjust) * (wvals.ncolors - 1)) / iteration); + if (bpp >= 3) + { + dest_row[col * bpp + 0] = colormap[color].r; + dest_row[col * bpp + 1] = colormap[color].g; + dest_row[col * bpp + 2] = colormap[color].b; + } + else + dest_row[col * bpp + 0] = valuemap[color]; + + if (! ( bpp % 2)) + dest_row [col * bpp + bpp - 1] = 255; + + } +} + +static void +delete_dialog_callback (GtkWidget *widget, + gboolean delete, + gpointer data) +{ + GtkWidget *view = (GtkWidget *) data; + GtkTreeSelection *selection; + GtkTreeModel *model; + GtkTreeIter iter; + gboolean valid; + fractalexplorerOBJ *sel_obj; + + if (delete) + { + /* Must update which object we are editing */ + /* Get the list and which item is selected */ + /* Only allow single selections */ + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); + gtk_tree_selection_get_selected (selection, &model, &iter); + + gtk_tree_model_get (model, &iter, 1, &sel_obj, -1); + + /* Delete the current item + associated file */ + valid = gtk_list_store_remove (GTK_LIST_STORE(model), &iter); + + /* Try to select first item if last one was deleted */ + if (!valid) + valid = gtk_tree_model_get_iter_first (model, &iter); + + /* Shadow copy for ordering info */ + fractalexplorer_list = g_list_remove (fractalexplorer_list, sel_obj); + /* + if(sel_obj == current_obj) + { + clear_undo(); + } + */ + /* Free current obj */ + fractalexplorer_free_everything (sel_obj); + + /* Check whether there are items left */ + if (valid) + { + gtk_tree_selection_select_iter (selection, &iter); + + gtk_tree_model_get (model, &iter, 1, ¤t_obj, -1); + } + } + + delete_dialog = NULL; +} + +static gboolean +delete_fractal_callback (GtkWidget *widget, + gpointer data) +{ + gchar *str; + GtkWidget *view = (GtkWidget *) data; + GtkTreeSelection *selection; + GtkTreeModel *model; + GtkTreeIter iter; + fractalexplorerOBJ *sel_obj; + + if (delete_dialog) + return FALSE; + + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); + if (gtk_tree_selection_get_selected (selection, &model, &iter)) + { + gtk_tree_model_get (model, &iter, 1, &sel_obj, -1); + + str = g_strdup_printf (_("Are you sure you want to delete " + "\"%s\" from the list and from disk?"), + sel_obj->draw_name); + + delete_dialog = gimp_query_boolean_box (_("Delete Fractal"), + gtk_widget_get_toplevel (view), + gimp_standard_help_func, NULL, + GIMP_ICON_DIALOG_QUESTION, + str, + _("_Delete"), _("_Cancel"), + G_OBJECT (widget), "destroy", + delete_dialog_callback, + data); + g_free (str); + + gtk_widget_show (delete_dialog); + } + + return FALSE; +} + +static gint +fractalexplorer_list_pos (fractalexplorerOBJ *fractalexplorer) +{ + fractalexplorerOBJ *g; + gint n; + GList *tmp; + + n = 0; + + for (tmp = fractalexplorer_list; tmp; tmp = g_list_next (tmp)) + { + g = tmp->data; + + if (strcmp (fractalexplorer->draw_name, g->draw_name) <= 0) + break; + + n++; + } + return n; +} + +static gint +fractalexplorer_list_insert (fractalexplorerOBJ *fractalexplorer) +{ + gint n = fractalexplorer_list_pos (fractalexplorer); + + /* + * Insert fractalexplorers in alphabetical order + */ + + fractalexplorer_list = g_list_insert (fractalexplorer_list, + fractalexplorer, n); + + return n; +} + +static fractalexplorerOBJ * +fractalexplorer_new (void) +{ + return g_new0 (fractalexplorerOBJ, 1); +} + +static void +fill_list_store (GtkListStore *list_store) +{ + GList *tmp; + GtkTreeIter iter; + + + for (tmp = fractalexplorer_list; tmp; tmp = tmp->next) + { + fractalexplorerOBJ *g; + g = tmp->data; + + gtk_list_store_append (list_store, &iter); + gtk_list_store_set (list_store, &iter, 0, g->draw_name, 1, g, -1); + } +} + +static void +activate_fractal (fractalexplorerOBJ *sel_obj) +{ + current_obj = sel_obj; + wvals = current_obj->opts; + dialog_change_scale (); + set_cmap_preview (); + dialog_update_preview (); +} + +static void +activate_fractal_callback (GtkTreeView *view, + GtkTreePath *path, + GtkTreeViewColumn *col, + gpointer data) +{ + GtkTreeModel *model; + GtkTreeIter iter; + fractalexplorerOBJ *sel_obj; + + model = gtk_tree_view_get_model (view); + + if (gtk_tree_model_get_iter (model, &iter, path)) + { + gtk_tree_model_get (model, &iter, 1, &sel_obj, -1); + activate_fractal (sel_obj); + } + +} + +static gboolean +apply_fractal_callback (GtkWidget *widget, + gpointer data) +{ + GtkWidget *view = (GtkWidget *) data; + GtkTreeSelection *selection; + GtkTreeModel *model; + GtkTreeIter iter; + fractalexplorerOBJ *sel_obj; + + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); + if (gtk_tree_selection_get_selected (selection, &model, &iter)) + { + gtk_tree_model_get (model, &iter, 1, &sel_obj, -1); + activate_fractal (sel_obj); + } + + return FALSE; +} + +static void +fractalexplorer_free (fractalexplorerOBJ *fractalexplorer) +{ + g_assert (fractalexplorer != NULL); + + g_free (fractalexplorer->name); + g_free (fractalexplorer->filename); + g_free (fractalexplorer->draw_name); + g_free (fractalexplorer); +} + +static void +fractalexplorer_free_everything (fractalexplorerOBJ *fractalexplorer) +{ + g_assert (fractalexplorer != NULL); + + if (fractalexplorer->filename) + g_remove (fractalexplorer->filename); + + fractalexplorer_free (fractalexplorer); +} + +static void +fractalexplorer_list_free_all (void) +{ + g_list_free_full (fractalexplorer_list, (GDestroyNotify) fractalexplorer_free); + fractalexplorer_list = NULL; +} + +static fractalexplorerOBJ * +fractalexplorer_load (const gchar *filename, + const gchar *name) +{ + fractalexplorerOBJ * fractalexplorer; + FILE * fp; + gchar load_buf[MAX_LOAD_LINE]; + + g_assert (filename != NULL); + + fp = g_fopen (filename, "rt"); + if (!fp) + { + g_message (_("Could not open '%s' for reading: %s"), + gimp_filename_to_utf8 (filename), g_strerror (errno)); + return NULL; + } + + fractalexplorer = fractalexplorer_new (); + + fractalexplorer->name = g_strdup (name); + fractalexplorer->draw_name = g_strdup (name); + fractalexplorer->filename = g_strdup (filename); + + /* HEADER + * draw_name + * version + * obj_list + */ + + get_line (load_buf, MAX_LOAD_LINE, fp, 1); + + if (strncmp (fractalexplorer_HEADER, load_buf, strlen (load_buf))) + { + g_message (_("File '%s' is not a FractalExplorer file"), + gimp_filename_to_utf8 (filename)); + fclose (fp); + fractalexplorer_free (fractalexplorer); + + return NULL; + } + + if (load_options (fractalexplorer, fp)) + { + g_message (_("File '%s' is corrupt.\nLine %d Option section incorrect"), + gimp_filename_to_utf8 (filename), line_no); + fclose (fp); + fractalexplorer_free (fractalexplorer); + + return NULL; + } + + fclose (fp); + + fractalexplorer->obj_status = fractalexplorer_OK; + + return fractalexplorer; +} + +static void +fractalexplorer_list_load_all (const gchar *explorer_path) +{ + GList *path; + GList *list; + + /* Make sure to clear any existing fractalexplorers */ + current_obj = NULL; + fractalexplorer_list_free_all (); + + path = gimp_config_path_expand_to_files (explorer_path, NULL); + + for (list = path; list; list = g_list_next (list)) + { + GFileEnumerator *enumerator; + + enumerator = g_file_enumerate_children (list->data, + G_FILE_ATTRIBUTE_STANDARD_NAME "," + G_FILE_ATTRIBUTE_STANDARD_IS_HIDDEN "," + G_FILE_ATTRIBUTE_STANDARD_TYPE, + G_FILE_QUERY_INFO_NONE, + NULL, NULL); + + if (enumerator) + { + GFileInfo *info; + + while ((info = g_file_enumerator_next_file (enumerator, NULL, NULL))) + { + GFileType file_type = g_file_info_get_file_type (info); + + if (file_type == G_FILE_TYPE_REGULAR && + ! g_file_info_get_is_hidden (info)) + { + fractalexplorerOBJ *fractalexplorer; + GFile *child; + gchar *filename; + gchar *basename; + + child = g_file_enumerator_get_child (enumerator, info); + + filename = g_file_get_path (child); + basename = g_file_get_basename (child); + + fractalexplorer = fractalexplorer_load (filename, + basename); + + g_free (filename); + g_free (basename); + + if (fractalexplorer) + fractalexplorer_list_insert (fractalexplorer); + + g_object_unref (child); + } + + g_object_unref (info); + } + + g_object_unref (enumerator); + } + } + + g_list_free_full (path, (GDestroyNotify) g_object_unref); + + if (!fractalexplorer_list) + { + fractalexplorerOBJ *fractalexplorer; + + /* lets have at least one! */ + fractalexplorer = fractalexplorer_new (); + fractalexplorer->draw_name = g_strdup (_("My first fractal")); + fractalexplorer_list_insert (fractalexplorer); + } + current_obj = fractalexplorer_list->data; /* set to first entry */ +} + + +GtkWidget * +add_objects_list (void) +{ + GtkWidget *table; + GtkWidget *scrolled_win; + GtkTreeViewColumn *col; + GtkCellRenderer *renderer; + GtkWidget *view; + GtkTreeSelection *selection; + GtkListStore *list_store; + GtkWidget *button; + + table = gtk_table_new (3, 2, FALSE); + gtk_table_set_row_spacings (GTK_TABLE (table), 6); + gtk_table_set_col_spacings (GTK_TABLE (table), 6); + gtk_container_set_border_width (GTK_CONTAINER (table), 12); + gtk_widget_show (table); + + scrolled_win = gtk_scrolled_window_new (NULL, NULL); + gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_win), + GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolled_win), + GTK_SHADOW_IN); + gtk_table_attach (GTK_TABLE (table), scrolled_win, 0, 3, 0, 1, + GTK_FILL|GTK_EXPAND , GTK_FILL|GTK_EXPAND, 0, 0); + gtk_widget_show (scrolled_win); + + view = gtk_tree_view_new (); + col = gtk_tree_view_column_new (); + gtk_tree_view_append_column (GTK_TREE_VIEW (view), col); + + renderer = gtk_cell_renderer_text_new (); + gtk_tree_view_column_pack_start (col, renderer, TRUE); + gtk_tree_view_column_add_attribute (col, renderer, "text", 0); + gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (view), FALSE); + + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); + gtk_tree_selection_set_mode (selection, GTK_SELECTION_BROWSE); + g_signal_connect (view, "row-activated", + G_CALLBACK (activate_fractal_callback), + NULL); + gtk_container_add (GTK_CONTAINER (scrolled_win), view); + gtk_widget_show (view); + + fractalexplorer_list_load_all (fractalexplorer_path); + list_store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_POINTER); + fill_list_store (list_store); + gtk_tree_view_set_model (GTK_TREE_VIEW (view), GTK_TREE_MODEL (list_store)); + g_object_unref (list_store); /* destroy model automatically with view */ + + /* Put buttons in */ + button = gtk_button_new_with_mnemonic (_("_Refresh")); + gtk_table_attach (GTK_TABLE (table), button, 0, 1, 1, 2, + GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); + gtk_widget_show (button); + + gimp_help_set_help_data (button, + _("Select folder and rescan collection"), NULL); + + g_signal_connect (button, "clicked", + G_CALLBACK (fractalexplorer_rescan_list), + view); + + button = gtk_button_new_with_mnemonic (_("_Apply")); + gtk_table_attach (GTK_TABLE (table), button, 1, 2, 1, 2, + GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); + gtk_widget_show (button); + + gimp_help_set_help_data (button, + _("Apply currently selected fractal"), NULL); + + g_signal_connect (button, "clicked", + G_CALLBACK (apply_fractal_callback), + view); + + button = gtk_button_new_with_mnemonic (_("_Delete")); + gtk_table_attach (GTK_TABLE (table), button, 2, 3, 1, 2, + GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); + gtk_widget_show (button); + + gimp_help_set_help_data (button, + _("Delete currently selected fractal"), NULL); + + g_signal_connect (button, "clicked", + G_CALLBACK (delete_fractal_callback), + view); + + return table; +} + +static void +fractalexplorer_rescan_list (GtkWidget *widget, + gpointer data) +{ + static GtkWidget *dlg = NULL; + GtkWidget *view = data; + GtkWidget *patheditor; + + if (dlg) + { + gtk_window_present (GTK_WINDOW (dlg)); + return; + } + + dlg = gimp_dialog_new (_("Rescan for Fractals"), PLUG_IN_ROLE, + gtk_widget_get_toplevel (view), + GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, + gimp_standard_help_func, PLUG_IN_PROC, + + _("_Cancel"), GTK_RESPONSE_CANCEL, + _("_OK"), GTK_RESPONSE_OK, + + NULL); + + gtk_dialog_set_alternative_button_order (GTK_DIALOG (dlg), + GTK_RESPONSE_OK, + GTK_RESPONSE_CANCEL, + -1); + + g_signal_connect (dlg, "destroy", + G_CALLBACK (gtk_widget_destroyed), + &dlg); + + patheditor = gimp_path_editor_new (_("Add FractalExplorer Path"), + fractalexplorer_path); + gtk_container_set_border_width (GTK_CONTAINER (patheditor), 12); + gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dlg))), + patheditor, TRUE, TRUE, 0); + gtk_widget_show (patheditor); + + if (gimp_dialog_run (GIMP_DIALOG (dlg)) == GTK_RESPONSE_OK) + { + g_free (fractalexplorer_path); + fractalexplorer_path = + gimp_path_editor_get_path (GIMP_PATH_EDITOR (patheditor)); + + if (fractalexplorer_path) + { + GtkTreeModel *model; + GtkTreeSelection *selection; + GtkTreePath *path; + GtkTreeIter iter; + + fractalexplorer_list_load_all (fractalexplorer_path); + + model = gtk_tree_view_get_model (GTK_TREE_VIEW (view)); + gtk_list_store_clear (GTK_LIST_STORE (model)); + fill_list_store (GTK_LIST_STORE (model)); + + /* select active fractal, otherwise first fractal */ + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); + if (gtk_tree_model_get_iter_first (model, &iter)) + { + gtk_tree_selection_select_iter (selection, &iter); + path = gtk_tree_model_get_path (model, &iter); + current_obj = fractalexplorer_list->data; + gtk_tree_view_scroll_to_cell (GTK_TREE_VIEW (view), path, NULL, + FALSE, 0.0, 0.0); + gtk_tree_path_free (path); + } + } + } + + gtk_widget_destroy (dlg); +} |