gimp/plug-ins/FractalExplorer/Dialogs.c

2002 lines
55 KiB
C

#include "config.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <libgimp/gimp.h>
#include <libgimp/gimpui.h>
#include "FractalExplorer.h"
#include "Dialogs.h"
#include "Events.h"
#include "logo.h"
#include "libgimp/stdplugins-intl.h"
#ifdef G_OS_WIN32
# include <io.h>
# ifndef W_OK
# define W_OK 2
# endif
# ifndef S_ISDIR
# define S_ISDIR(m) ((m) & _S_IFDIR)
# endif
# ifndef S_ISREG
# define S_ISREG(m) ((m) & _S_IFREG)
# endif
#endif
static gdouble *gradient_samples = NULL;
static gchar *gradient_name = NULL;
/**********************************************************************
FORWARD DECLARATIONS
*********************************************************************/
static void explorer_logo_dialog (void);
/**********************************************************************
CALLBACKS
*********************************************************************/
static void
dialog_ok_callback (GtkWidget *widget,
gpointer data)
{
wint.run = TRUE;
gtk_widget_destroy (GTK_WIDGET (data));
}
static void
dialog_reset_callback (GtkWidget *widget,
gpointer data)
{
wvals.xmin = standardvals.xmin;
wvals.xmax = standardvals.xmax;
wvals.ymin = standardvals.ymin;
wvals.ymax = standardvals.ymax;
wvals.iter = standardvals.iter;
wvals.cx = standardvals.cx;
wvals.cy = standardvals.cy;
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
static void
dialog_redraw_callback (GtkWidget *widget,
gpointer data)
{
gint alwaysprev = wvals.alwayspreview;
wvals.alwayspreview = TRUE;
set_cmap_preview ();
dialog_update_preview ();
wvals.alwayspreview = alwaysprev;
}
static void
dialog_undo_zoom_callback (GtkWidget *widget,
gpointer data)
{
if (zoomindex > 1)
{
zooms[zoomindex] = wvals;
zoomindex--;
wvals = zooms[zoomindex];
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
}
static void
dialog_redo_zoom_callback (GtkWidget *widget,
gpointer data)
{
if (zoomindex < zoommax)
{
zoomindex++;
wvals = zooms[zoomindex];
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
}
static void
dialog_step_in_callback (GtkWidget *widget,
gpointer data)
{
double xdifferenz;
double ydifferenz;
if (zoomindex < zoommax)
{
zooms[zoomindex]=wvals;
zoomindex++;
}
xdifferenz = wvals.xmax - wvals.xmin;
ydifferenz = wvals.ymax - wvals.ymin;
wvals.xmin += 1.0 / 6.0 * xdifferenz;
wvals.ymin += 1.0 / 6.0 * ydifferenz;
wvals.xmax -= 1.0 / 6.0 * xdifferenz;
wvals.ymax -= 1.0 / 6.0 * ydifferenz;
zooms[zoomindex] = wvals;
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
static void
dialog_step_out_callback (GtkWidget *widget,
gpointer data)
{
gdouble xdifferenz;
gdouble ydifferenz;
if (zoomindex < zoommax)
{
zooms[zoomindex]=wvals;
zoomindex++;
}
xdifferenz = wvals.xmax - wvals.xmin;
ydifferenz = wvals.ymax - wvals.ymin;
wvals.xmin -= 1.0 / 4.0 * xdifferenz;
wvals.ymin -= 1.0 / 4.0 * ydifferenz;
wvals.xmax += 1.0 / 4.0 * xdifferenz;
wvals.ymax += 1.0 / 4.0 * ydifferenz;
zooms[zoomindex] = wvals;
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
static void
explorer_toggle_update (GtkWidget *widget,
gpointer data)
{
gimp_toggle_button_update (widget, data);
set_cmap_preview ();
dialog_update_preview ();
}
static void
explorer_radio_update (GtkWidget *widget,
gpointer data)
{
gimp_radio_button_update (widget, data);
set_cmap_preview ();
dialog_update_preview ();
}
static void
explorer_double_adjustment_update (GtkAdjustment *adjustment,
gpointer data)
{
gimp_double_adjustment_update (adjustment, data);
set_cmap_preview ();
dialog_update_preview ();
}
static void
explorer_number_of_colors_callback (GtkAdjustment *adjustment,
gpointer data)
{
gint dummy;
gimp_int_adjustment_update (adjustment, data);
g_free (gradient_samples);
if (gradient_name == NULL)
gradient_name = gimp_gradients_get_active ();
gimp_gradients_get_gradient_data (gradient_name, &dummy, wvals.ncolors,
&gradient_samples);
set_cmap_preview ();
dialog_update_preview ();
}
static void
explorer_gradient_select_callback (gchar *name,
gint width,
gdouble *gradient_data,
gint dialog_closing,
gpointer data)
{
gint dummy;
g_free (gradient_name);
g_free (gradient_samples);
gradient_name = g_strdup (name);
gimp_gradients_get_gradient_data (gradient_name, &dummy, wvals.ncolors,
&gradient_samples);
if (wvals.colormode == 1)
{
set_cmap_preview ();
dialog_update_preview ();
}
}
/**********************************************************************
FUNCTION: explorer_dialog
*********************************************************************/
gint
explorer_dialog (void)
{
GtkWidget *dialog;
GtkWidget *top_hbox;
GtkWidget *left_vbox;
GtkWidget *abox;
GtkWidget *vbox;
GtkWidget *hbbox;
GtkWidget *frame;
GtkWidget *pframe;
GtkWidget *toggle;
GtkWidget *toggle_vbox;
GtkWidget *toggle_vbox2;
GtkWidget *toggle_vbox3;
GtkWidget *notebook;
GtkWidget *hbox;
GtkWidget *table;
GtkWidget *button;
GtkWidget *gradient;
GtkWidget *sep;
gchar *gradient_name;
GSList *group = NULL;
gint i;
gimp_ui_init ("fractalexplorer", TRUE);
plug_in_parse_fractalexplorer_path ();
wint.wimage = g_new (guchar, preview_width * preview_height * 3);
elements = g_new (DialogElements, 1);
dialog = maindlg =
gimp_dialog_new ("Fractal Explorer <Daniel Cotting/cotting@multimania.com>",
"fractalexplorer",
gimp_standard_help_func, "filters/fractalexplorer.html",
GTK_WIN_POS_NONE,
FALSE, TRUE, FALSE,
_("About"), explorer_logo_dialog,
NULL, NULL, NULL, FALSE, FALSE,
_("OK"), dialog_ok_callback,
NULL, NULL, NULL, TRUE, FALSE,
_("Cancel"), gtk_widget_destroy,
NULL, 1, NULL, FALSE, TRUE,
NULL);
gtk_signal_connect (GTK_OBJECT (dialog), "destroy",
GTK_SIGNAL_FUNC (gtk_main_quit),
NULL);
gimp_help_init ();
top_hbox = gtk_hbox_new (FALSE, 6);
gtk_container_set_border_width (GTK_CONTAINER (top_hbox), 6);
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (dialog)->vbox), top_hbox,
FALSE, FALSE, 0);
gtk_widget_show (top_hbox);
left_vbox = gtk_vbox_new (FALSE, 6);
gtk_box_pack_start (GTK_BOX (top_hbox), left_vbox, FALSE, FALSE, 0);
gtk_widget_show (left_vbox);
/* Preview */
frame = gtk_frame_new (_("Preview"));
gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_start (GTK_BOX (left_vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
vbox = gtk_vbox_new (FALSE, 4);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
gtk_container_add (GTK_CONTAINER (frame), vbox);
gtk_widget_show (vbox);
abox = gtk_alignment_new (0.5, 0.5, 0.0, 0.0);
gtk_box_pack_start (GTK_BOX (vbox), abox, FALSE, FALSE, 0);
gtk_widget_show (abox);
pframe = gtk_frame_new (NULL);
gtk_frame_set_shadow_type (GTK_FRAME (pframe), GTK_SHADOW_IN);
gtk_container_add (GTK_CONTAINER (abox), pframe);
gtk_widget_show (pframe);
wint.preview = gtk_preview_new (GTK_PREVIEW_COLOR);
gtk_preview_size (GTK_PREVIEW (wint.preview), preview_width, preview_height);
gtk_container_add (GTK_CONTAINER (pframe), wint.preview);
gtk_signal_connect (GTK_OBJECT (wint.preview), "button_press_event",
(GtkSignalFunc) preview_button_press_event,
NULL);
gtk_signal_connect (GTK_OBJECT (wint.preview), "button_release_event",
(GtkSignalFunc) preview_button_release_event,
NULL);
gtk_signal_connect (GTK_OBJECT (wint.preview), "motion_notify_event",
(GtkSignalFunc) preview_motion_notify_event,
NULL);
gtk_signal_connect (GTK_OBJECT (wint.preview), "leave_notify_event",
(GtkSignalFunc) preview_leave_notify_event,
NULL);
gtk_signal_connect (GTK_OBJECT (wint.preview), "enter_notify_event",
(GtkSignalFunc) preview_enter_notify_event,
NULL);
gtk_widget_set_events (wint.preview, (GDK_BUTTON_PRESS_MASK |
GDK_BUTTON_RELEASE_MASK |
GDK_POINTER_MOTION_MASK |
GDK_LEAVE_NOTIFY_MASK |
GDK_ENTER_NOTIFY_MASK));
gtk_widget_show (wint.preview);
toggle = gtk_check_button_new_with_label (_("Realtime Preview"));
gtk_box_pack_start (GTK_BOX (vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_toggle_update),
&wvals.alwayspreview);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), wvals.alwayspreview);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle, _("If you enable this option the preview "
"will be redrawn automatically"), NULL);
button = gtk_button_new_with_label (_("Redraw"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (dialog_redraw_callback),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Redraw preview"), NULL);
/* Zoom Options */
frame = gtk_frame_new (_("Zoom Options"));
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_end (GTK_BOX (left_vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
vbox = gtk_vbox_new(FALSE, 0);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
gtk_container_add (GTK_CONTAINER (frame), vbox);
gtk_widget_show (vbox);
button = gtk_button_new_with_label (_("Undo Zoom"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (dialog_undo_zoom_callback),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Undo last zoom"), NULL);
button = gtk_button_new_with_label (_("Redo Zoom"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (dialog_redo_zoom_callback),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Redo last zoom"), NULL);
button = gtk_button_new_with_label (_("Step In"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (dialog_step_in_callback),
dialog);
gtk_widget_show (button);
button = gtk_button_new_with_label (_("Step Out"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (dialog_step_out_callback),
dialog);
gtk_widget_show (button);
/* Create notebook */
notebook = gtk_notebook_new ();
gtk_box_pack_start (GTK_BOX (top_hbox), notebook, FALSE, FALSE, 0);
gtk_widget_show (notebook);
/* "Parameters" page */
vbox = gtk_vbox_new (FALSE, 4);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), vbox,
gtk_label_new (_("Parameters")));
gtk_widget_show (vbox);
frame = gtk_frame_new (_("Fractal Parameters"));
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
table = gtk_table_new (8, 3, FALSE);
gtk_container_set_border_width (GTK_CONTAINER (table), 4);
gtk_table_set_col_spacings (GTK_TABLE (table), 4);
gtk_table_set_row_spacings (GTK_TABLE (table), 2);
gtk_table_set_row_spacing (GTK_TABLE (table), 6, 4);
gtk_container_add (GTK_CONTAINER (frame), table);
gtk_widget_show (table);
elements->xmin =
gimp_scale_entry_new (GTK_TABLE (table), 0, 0,
_("XMIN:"), SCALE_WIDTH, 100,
wvals.xmin, -3, 3, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the first (minimal) x-coordinate "
"delimitation"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->xmin), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.xmin);
elements->xmax =
gimp_scale_entry_new (GTK_TABLE (table), 0, 1,
_("XMAX:"), SCALE_WIDTH, 100,
wvals.xmax, -3, 3, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the second (maximal) x-coordinate "
"delimitation"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->xmax), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.xmax);
elements->ymin =
gimp_scale_entry_new (GTK_TABLE (table), 0, 2,
_("YMIN:"), SCALE_WIDTH, 100,
wvals.ymin, -3, 3, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the first (minimal) y-coordinate "
"delimitation"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->ymin), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.ymin);
elements->ymax =
gimp_scale_entry_new (GTK_TABLE (table), 0, 3,
_("YMAX:"), SCALE_WIDTH, 100,
wvals.ymax, -3, 3, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the second (maximal) y-coordinate "
"delimitation"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->ymax), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.ymax);
elements->iter =
gimp_scale_entry_new (GTK_TABLE (table), 0, 4,
_("ITER:"), SCALE_WIDTH, 100,
wvals.iter, 0, 1000, 1, 10, 5,
TRUE, 0, 0,
_("Change the iteration value. The higher it "
"is, the more details will be calculated, "
"which will take more time"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->iter), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.iter);
elements->cx =
gimp_scale_entry_new (GTK_TABLE (table), 0, 5,
_("CX:"), SCALE_WIDTH, 100,
wvals.cx, -2.5, 2.5, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the CX value (changes aspect of "
"fractal, active with every fractal but "
"Mandelbrot and Sierpinski)"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->cx), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.cx);
elements->cy =
gimp_scale_entry_new (GTK_TABLE (table), 0, 6,
_("CY:"), SCALE_WIDTH, 100,
wvals.cy, -2.5, 2.5, 0.001, 0.01, 5,
TRUE, 0, 0,
_("Change the CY value (changes aspect of "
"fractal, active with every fractal but "
"Mandelbrot and Sierpinski)"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->cy), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.cy);
hbbox = gtk_hbox_new (FALSE, 4);
gtk_table_attach_defaults (GTK_TABLE (table), hbbox, 0, 3, 7, 8);
gtk_widget_show (hbbox);
button = gtk_button_new_with_label (_("Load"));
gtk_box_pack_start (GTK_BOX (hbbox), button, TRUE, TRUE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (create_load_file_selection),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Load a fractal from file"), NULL);
button = gtk_button_new_with_label (_("Reset"));
gtk_box_pack_start (GTK_BOX (hbbox), button, TRUE, TRUE, 0);
gtk_signal_connect (GTK_OBJECT(button), "clicked",
GTK_SIGNAL_FUNC (dialog_reset_callback),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Reset parameters to default values"),
NULL);
button = gtk_button_new_with_label (_("Save"));
gtk_box_pack_start (GTK_BOX (hbbox), button, TRUE, TRUE, 0);
gtk_signal_connect (GTK_OBJECT (button), "clicked",
GTK_SIGNAL_FUNC (create_file_selection),
dialog);
gtk_widget_show (button);
gimp_help_set_help_data (button, _("Save active fractal to file"), NULL);
/* Fractal type toggle box */
frame = gtk_frame_new (_("Fractal Type"));
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
hbox = gtk_hbox_new (FALSE, 6);
gtk_container_set_border_width (GTK_CONTAINER (hbox), 4);
gtk_container_add (GTK_CONTAINER (frame), hbox);
toggle_vbox =
gimp_radio_group_new2 (FALSE, NULL,
explorer_radio_update,
&wvals.fractaltype,
(gpointer) &wvals.fractaltype,
_("Mandelbrot"), (gpointer) TYPE_MANDELBROT,
&(elements->type[TYPE_MANDELBROT]),
_("Julia"), (gpointer) TYPE_JULIA,
&(elements->type[TYPE_JULIA]),
_("Barnsley 1"), (gpointer) TYPE_BARNSLEY_1,
&(elements->type[TYPE_BARNSLEY_1]),
_("Barnsley 2"), (gpointer) TYPE_BARNSLEY_2,
&(elements->type[TYPE_BARNSLEY_2]),
_("Barnsley 3"), (gpointer) TYPE_BARNSLEY_3,
&(elements->type[TYPE_BARNSLEY_3]),
_("Spider"), (gpointer) TYPE_SPIDER,
&(elements->type[TYPE_SPIDER]),
_("Man'o'war"), (gpointer) TYPE_MAN_O_WAR,
&(elements->type[TYPE_MAN_O_WAR]),
_("Lambda"), (gpointer) TYPE_LAMBDA,
&(elements->type[TYPE_LAMBDA]),
_("Sierpinski"), (gpointer) TYPE_SIERPINSKI,
&(elements->type[TYPE_SIERPINSKI]),
NULL);
toggle_vbox2 = gtk_vbox_new (FALSE, 1);
for (i = TYPE_BARNSLEY_2; i <= TYPE_SPIDER; i++)
{
gtk_object_ref (GTK_OBJECT (elements->type[i]));
gtk_widget_hide (elements->type[i]);
gtk_container_remove (GTK_CONTAINER (toggle_vbox), elements->type[i]);
gtk_box_pack_start (GTK_BOX (toggle_vbox2), elements->type[i],
FALSE, FALSE, 0);
gtk_widget_show (elements->type[i]);
gtk_object_unref (GTK_OBJECT (elements->type[i]));
}
toggle_vbox3 = gtk_vbox_new (FALSE, 1);
for (i = TYPE_MAN_O_WAR; i <= TYPE_SIERPINSKI; i++)
{
gtk_object_ref (GTK_OBJECT (elements->type[i]));
gtk_widget_hide (elements->type[i]);
gtk_container_remove (GTK_CONTAINER (toggle_vbox), elements->type[i]);
gtk_box_pack_start (GTK_BOX (toggle_vbox3), elements->type[i],
FALSE, FALSE, 0);
gtk_widget_show (elements->type[i]);
gtk_object_unref (GTK_OBJECT (elements->type[i]));
}
gtk_container_set_border_width (GTK_CONTAINER (toggle_vbox), 0);
gtk_box_pack_start (GTK_BOX (hbox), toggle_vbox, FALSE, FALSE, 0);
gtk_widget_show (toggle_vbox);
gtk_box_pack_start (GTK_BOX (hbox), toggle_vbox2, FALSE, FALSE, 0);
gtk_widget_show (toggle_vbox2);
gtk_box_pack_start (GTK_BOX (hbox), toggle_vbox3, FALSE, FALSE, 0);
gtk_widget_show (toggle_vbox3);
gtk_widget_show (hbox);
gtk_widget_show (frame);
/* Color page */
vbox = gtk_vbox_new (FALSE, 4);
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), vbox,
gtk_label_new (_("Colors")));
gtk_widget_show (vbox);
/* Number of Colors frame */
frame = gtk_frame_new (_("Number of Colors"));
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
table = gtk_table_new (2, 3, FALSE);
gtk_table_set_col_spacings (GTK_TABLE (table), 4);
gtk_table_set_row_spacings (GTK_TABLE (table), 2);
gtk_container_set_border_width (GTK_CONTAINER (table), 4);
gtk_container_add (GTK_CONTAINER (frame), table);
gtk_widget_show (table);
elements->ncol =
gimp_scale_entry_new (GTK_TABLE (table), 0, 0,
_("Number of Colors:"), SCALE_WIDTH, 0,
wvals.ncolors, 2, MAXNCOLORS, 1, 10, 0,
TRUE, 0, 0,
_("Change the number of colors in the mapping"),
NULL);
gtk_signal_connect (GTK_OBJECT (elements->ncol), "value_changed",
GTK_SIGNAL_FUNC (explorer_number_of_colors_callback),
&wvals.ncolors);
elements->useloglog = toggle =
gtk_check_button_new_with_label (_("Use loglog Smoothing"));
gtk_table_attach_defaults (GTK_TABLE (table), toggle, 0, 3, 1, 2);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
(GtkSignalFunc) explorer_toggle_update,
&wvals.useloglog);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), wvals.useloglog);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle, _("Use log log smoothing to eliminate "
"\"banding\" in the result"), NULL);
/* Color Density frame */
frame = gtk_frame_new (_("Color Density"));
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
table = gtk_table_new (3, 3, FALSE);
gtk_table_set_col_spacings (GTK_TABLE (table), 4);
gtk_table_set_row_spacings (GTK_TABLE (table), 2);
gtk_container_set_border_width (GTK_CONTAINER (table), 4);
gtk_container_add (GTK_CONTAINER (frame), table);
gtk_widget_show (table);
elements->red =
gimp_scale_entry_new (GTK_TABLE (table), 0, 0,
_("Red:"), SCALE_WIDTH, 0,
wvals.redstretch, 0, 1, 0.01, 0.1, 2,
TRUE, 0, 0,
_("Change the intensity of the red channel"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->red), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.redstretch);
elements->green =
gimp_scale_entry_new (GTK_TABLE (table), 0, 1,
_("Green:"), SCALE_WIDTH, 0,
wvals.greenstretch, 0, 1, 0.01, 0.1, 2,
TRUE, 0, 0,
_("Change the intensity of the green channel"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->green), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.greenstretch);
elements->blue =
gimp_scale_entry_new (GTK_TABLE (table), 0, 2,
_("Blue:"), SCALE_WIDTH, 0,
wvals.bluestretch, 0, 1, 0.01, 0.1, 2,
TRUE, 0, 0,
_("Change the intensity of the blue channel"), NULL);
gtk_signal_connect (GTK_OBJECT (elements->blue), "value_changed",
GTK_SIGNAL_FUNC (explorer_double_adjustment_update),
&wvals.bluestretch);
/* Color Function frame */
frame = gtk_frame_new (_("Color Function"));
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
hbox = gtk_hbox_new (FALSE, 6);
gtk_container_set_border_width (GTK_CONTAINER (hbox), 4);
gtk_container_add (GTK_CONTAINER (frame), hbox);
gtk_widget_show (hbox);
/* Redmode radio frame */
frame = gimp_radio_group_new2 (TRUE, _("Red"),
explorer_radio_update,
&wvals.redmode, (gpointer) wvals.redmode,
_("Sine"), (gpointer) SINUS,
&elements->redmode[SINUS],
_("Cosine"), (gpointer) COSINUS,
&elements->redmode[COSINUS],
_("None"), (gpointer) NONE,
&elements->redmode[NONE],
NULL);
gimp_help_set_help_data (elements->redmode[SINUS],
_("Use sine-function for this color component"),
NULL);
gimp_help_set_help_data (elements->redmode[COSINUS],
_("Use cosine-function for this color "
"component"), NULL);
gimp_help_set_help_data (elements->redmode[NONE],
_("Use linear mapping instead of any "
"trigonometrical function for this color "
"channel"), NULL);
gtk_box_pack_start (GTK_BOX (hbox), frame, TRUE, TRUE, 0);
gtk_widget_show (frame);
toggle_vbox = GTK_BIN (frame)->child;
elements->redinvert = toggle =
gtk_check_button_new_with_label (_("Inversion"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), wvals.redinvert);
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_toggle_update),
&wvals.redinvert);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle,
_("If you enable this option higher color values "
"will be swapped with lower ones and vice "
"versa"), NULL);
/* Greenmode radio frame */
frame = gimp_radio_group_new2 (TRUE, _("Green"),
explorer_radio_update,
&wvals.greenmode, (gpointer) wvals.greenmode,
_("Sine"), (gpointer) SINUS,
&elements->greenmode[SINUS],
_("Cosine"), (gpointer) COSINUS,
&elements->greenmode[COSINUS],
_("None"), (gpointer) NONE,
&elements->greenmode[NONE],
NULL);
gimp_help_set_help_data (elements->greenmode[SINUS],
_("Use sine-function for this color component"),
NULL);
gimp_help_set_help_data (elements->greenmode[COSINUS],
_("Use cosine-function for this color "
"component"), NULL);
gimp_help_set_help_data (elements->greenmode[NONE],
_("Use linear mapping instead of any "
"trigonometrical function for this color "
"channel"), NULL);
gtk_box_pack_start (GTK_BOX (hbox), frame, TRUE, TRUE, 0);
gtk_widget_show (frame);
toggle_vbox = GTK_BIN (frame)->child;
elements->greeninvert = toggle =
gtk_check_button_new_with_label (_("Inversion"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), wvals.greeninvert);
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_toggle_update),
&wvals.greeninvert);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle,
_("If you enable this option higher color values "
"will be swapped with lower ones and vice "
"versa"), NULL);
/* Bluemode radio frame */
frame = gimp_radio_group_new2 (TRUE, _("Blue"),
explorer_radio_update,
&wvals.bluemode, (gpointer) wvals.bluemode,
_("Sine"), (gpointer) SINUS,
&elements->bluemode[SINUS],
_("Cosine"), (gpointer) COSINUS,
&elements->bluemode[COSINUS],
_("None"), (gpointer) NONE,
&elements->bluemode[NONE],
NULL);
gimp_help_set_help_data (elements->bluemode[SINUS],
_("Use sine-function for this color component"),
NULL);
gimp_help_set_help_data (elements->bluemode[COSINUS],
_("Use cosine-function for this color "
"component"), NULL);
gimp_help_set_help_data (elements->bluemode[NONE],
_("Use linear mapping instead of any "
"trigonometrical function for this color "
"channel"), NULL);
gtk_box_pack_start (GTK_BOX (hbox), frame, TRUE, TRUE, 0);
gtk_widget_show (frame);
toggle_vbox = GTK_BIN (frame)->child;
elements->blueinvert = toggle =
gtk_check_button_new_with_label (_("Inversion"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON( toggle), wvals.blueinvert);
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_toggle_update),
&wvals.blueinvert);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle,
_("If you enable this option higher color values "
"will be swapped with lower ones and vice "
"versa"), NULL);
/* Colormode toggle box */
frame = gtk_frame_new (_("Color Mode"));
gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_ETCHED_IN);
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
toggle_vbox = gtk_vbox_new (FALSE, 2);
gtk_container_set_border_width (GTK_CONTAINER (toggle_vbox), 2);
gtk_container_add (GTK_CONTAINER (frame), toggle_vbox);
gtk_widget_show (toggle_vbox);
toggle = elements->colormode[0] =
gtk_radio_button_new_with_label (group, _("As Specified above"));
group = gtk_radio_button_group (GTK_RADIO_BUTTON (toggle));
gtk_box_pack_start (GTK_BOX (toggle_vbox), toggle, FALSE, FALSE, 0);
gtk_object_set_user_data (GTK_OBJECT (toggle), (gpointer) 0);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_radio_update),
&wvals.colormode);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
wvals.colormode == 0);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle,
_("Create a color-map with the options you "
"specified above (color density/function). The "
"result is visible in the preview image"), NULL);
hbox = gtk_hbox_new (FALSE, 6);
gtk_box_pack_start (GTK_BOX (toggle_vbox), hbox, FALSE, FALSE, 0);
gtk_widget_show (hbox);
toggle = elements->colormode[1] =
gtk_radio_button_new_with_label (group,
_("Apply Active Gradient to Final Image"));
group = gtk_radio_button_group (GTK_RADIO_BUTTON (toggle));
gtk_box_pack_start (GTK_BOX (hbox), toggle, TRUE, TRUE, 0);
gtk_object_set_user_data (GTK_OBJECT (toggle), (gpointer) 1);
gtk_signal_connect (GTK_OBJECT (toggle), "toggled",
GTK_SIGNAL_FUNC (explorer_radio_update),
&wvals.colormode);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle),
wvals.colormode == 1);
gtk_widget_show (toggle);
gimp_help_set_help_data (toggle,
_("Create a color-map using a gradient from "
"the gradient editor"), NULL);
gradient_name = gimp_gradients_get_active ();
gradient_samples = gimp_gradients_sample_uniform (wvals.ncolors);
gradient = gimp_gradient_select_widget (_("FractalExplorer Gradient"),
gradient_name,
explorer_gradient_select_callback,
NULL);
g_free (gradient_name);
gtk_box_pack_start (GTK_BOX (hbox), gradient, FALSE, FALSE, 0);
gtk_widget_show (gradient);
sep = gtk_hseparator_new ();
gtk_box_pack_start (GTK_BOX (toggle_vbox), sep, FALSE, FALSE, 1);
gtk_widget_show (sep);
abox = gtk_alignment_new (0.5, 0.5, 0.0, 0.0);
{
gint xsize, ysize;
for (ysize = 1; ysize * ysize * ysize < 8192; ysize++) /**/;
xsize = wvals.ncolors / ysize;
while (xsize * ysize < 8192) xsize++;
gtk_widget_set_usize (abox, xsize, ysize * 4);
}
gtk_box_pack_start (GTK_BOX (toggle_vbox), abox, FALSE, FALSE, 0);
gtk_widget_show (abox);
cmap_preview = gtk_preview_new (GTK_PREVIEW_COLOR);
gtk_preview_size (GTK_PREVIEW (cmap_preview), 32, 32);
gtk_container_add (GTK_CONTAINER (abox), cmap_preview);
gtk_widget_show (cmap_preview);
frame = add_objects_list ();
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), frame,
gtk_label_new (_("Fractals")));
gtk_widget_show (frame);
gtk_notebook_set_page (GTK_NOTEBOOK (notebook), 1);
/* Done */
/* Popup for list area: Not yet fully implemented
fractalexplorer_op_menu_create(maindlg);
*/
gtk_widget_show (dialog);
ready_now = TRUE;
set_cmap_preview ();
dialog_update_preview ();
gtk_main ();
gimp_help_free ();
gdk_flush ();
if (the_tile != NULL)
{
gimp_tile_unref (the_tile, FALSE);
the_tile = NULL;
}
g_free (wint.wimage);
return wint.run;
}
/**********************************************************************
FUNCTION: dialog_update_preview
*********************************************************************/
void
dialog_update_preview (void)
{
gdouble left;
gdouble right;
gdouble bottom;
gdouble top;
gdouble dx;
gdouble dy;
gdouble cx;
gdouble cy;
gint px;
gint py;
gint xcoord;
gint ycoord;
gint iteration;
guchar *p_ul;
guchar *p;
gdouble a;
gdouble b;
gdouble x;
gdouble y;
gdouble oldx;
gdouble oldy;
gdouble foldxinitx;
gdouble foldxinity;
gdouble tempsqrx;
gdouble tempsqry;
gdouble tmpx = 0;
gdouble tmpy = 0;
gdouble foldyinitx;
gdouble foldyinity;
gdouble adjust;
gdouble xx = 0;
gint zaehler;
gint color;
gint useloglog;
if (NULL == wint.preview)
return;
if ((ready_now) && (wvals.alwayspreview))
{
left = sel_x1;
right = sel_x2 - 1;
bottom = sel_y2 - 1;
top = sel_y1;
dx = (right - left) / (preview_width - 1);
dy = (bottom - top) / (preview_height - 1);
xmin = wvals.xmin;
xmax = wvals.xmax;
ymin = wvals.ymin;
ymax = wvals.ymax;
cx = wvals.cx;
cy = wvals.cy;
xbild = preview_width;
ybild = preview_height;
xdiff = (xmax - xmin) / xbild;
ydiff = (ymax - ymin) / ybild;
py = 0;
p_ul = wint.wimage;
iteration = (int) wvals.iter;
useloglog = wvals.useloglog;
for (ycoord = 0; ycoord < preview_height; ycoord++)
{
px = 0;
for (xcoord = 0; xcoord < preview_width; xcoord++)
{
a = (double) xmin + (double) xcoord *xdiff;
b = (double) ymin + (double) ycoord *ydiff;
if (wvals.fractaltype != TYPE_MANDELBROT)
{
tmpx = x = a;
tmpy = y = b;
}
else
{
x = 0;
y = 0;
}
for (zaehler = 0;
(zaehler < iteration) && ((x * x + y * y) < 4);
zaehler++)
{
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 (useloglog)
{
adjust = log (log (x * x + y * y) / 2) / log (2);
}
else
{
adjust = 0.0;
}
color = (int) (((zaehler - adjust) *
(wvals.ncolors - 1)) / iteration);
p_ul[0] = colormap[color][0];
p_ul[1] = colormap[color][1];
p_ul[2] = colormap[color][2];
p_ul += 3;
px += 1;
} /* for */
py += 1;
} /* for */
p = wint.wimage;
for (y = 0; y < preview_height; y++)
{
gtk_preview_draw_row (GTK_PREVIEW (wint.preview), p,
0, y, preview_width);
p += preview_width * 3;
}
gtk_widget_draw (wint.preview, NULL);
gdk_flush ();
}
}
/**********************************************************************
FUNCTION: set_cmap_preview()
*********************************************************************/
void
set_cmap_preview (void)
{
gint i;
gint x;
gint y;
gint j;
guchar *b;
guchar c[GR_WIDTH * 3];
gint xsize, ysize;
if (NULL == cmap_preview)
return;
make_color_map ();
for (ysize = 1; ysize * ysize * ysize < wvals.ncolors; ysize++) /**/;
xsize = wvals.ncolors / ysize;
while (xsize * ysize < wvals.ncolors) xsize++;
b = g_new (guchar, xsize * 3);
gtk_preview_size (GTK_PREVIEW (cmap_preview), xsize, ysize * 4);
gtk_widget_set_usize (GTK_WIDGET (cmap_preview), xsize, ysize * 4);
for (y = 0; y < ysize*4; y += 4)
{
for (x = 0; x < xsize; x++)
{
i = x + (y / 4) * xsize;
if (i > wvals.ncolors)
{
for (j = 0; j < 3; j++)
b[x * 3 + j] = 0;
}
else
{
for (j = 0; j < 3; j++)
b[x * 3 + j] = colormap[i][j];
}
}
gtk_preview_draw_row (GTK_PREVIEW (cmap_preview), b, 0, y, xsize);
gtk_preview_draw_row (GTK_PREVIEW (cmap_preview), b, 0, y + 1, xsize);
gtk_preview_draw_row (GTK_PREVIEW (cmap_preview), b, 0, y + 2, xsize);
gtk_preview_draw_row (GTK_PREVIEW (cmap_preview), b, 0, y + 3, xsize);
}
for (x = 0; x < GR_WIDTH; x++)
{
for (j = 0; j < 3; j++)
c[x * 3 + j] = colormap[(int)((float)x/(float)GR_WIDTH*256.0)][j];
}
gtk_widget_draw (cmap_preview, NULL);
g_free (b);
}
/**********************************************************************
FUNCTION: make_color_map()
*********************************************************************/
void
make_color_map (void)
{
gint i;
gint j;
gint r;
gint gr;
gint bl;
gdouble redstretch;
gdouble greenstretch;
gdouble bluestretch;
gdouble pi = atan (1) * 4;
redstretch = wvals.redstretch * 127.5;
greenstretch = wvals.greenstretch * 127.5;
bluestretch = wvals.bluestretch * 127.5;
for (i = 0; i < wvals.ncolors; i++)
if (wvals.colormode == 1)
{
for (j = 0; j < 3; j++)
colormap[i][j] = (int) (gradient_samples[i * 4 + j] * 255.0);
}
else
{
double x = (i*2.0) / wvals.ncolors;
r = gr = bl = 0;
switch (wvals.redmode)
{
case SINUS:
r = (int) redstretch *(1.0 + sin((x - 1) * pi));
break;
case COSINUS:
r = (int) redstretch *(1.0 + cos((x - 1) * pi));
break;
case NONE:
r = (int)(redstretch *(x));
break;
default:
break;
}
switch (wvals.greenmode)
{
case SINUS:
gr = (int) greenstretch *(1.0 + sin((x - 1) * pi));
break;
case COSINUS:
gr = (int) greenstretch *(1.0 + cos((x - 1) * pi));
break;
case NONE:
gr = (int)(greenstretch *(x));
break;
default:
break;
}
switch (wvals.bluemode)
{
case SINUS:
bl = (int) bluestretch * (1.0 + sin ((x - 1) * pi));
break;
case COSINUS:
bl = (int) bluestretch * (1.0 + cos ((x - 1) * pi));
break;
case NONE:
bl = (int) (bluestretch * x);
break;
default:
break;
}
r = MIN (r, 255);
gr = MIN (gr, 255);
bl = MIN (bl, 255);
if (wvals.redinvert)
r = 255 - r;
if (wvals.greeninvert)
gr = 255 - gr;
if (wvals.blueinvert)
bl = 255 - bl;
colormap[i][0] = r;
colormap[i][1] = gr;
colormap[i][2] = bl;
}
}
/**********************************************************************
FUNCTION: explorer_logo_dialog
*********************************************************************/
static void
explorer_logo_dialog (void)
{
GtkWidget *xdlg;
GtkWidget *xlabel = NULL;
GtkWidget *xlogo_box;
GtkWidget *xpreview;
GtkWidget *xframe;
GtkWidget *xframe2;
GtkWidget *xframe3;
GtkWidget *xvbox;
GtkWidget *xhbox;
GtkWidget *vpaned;
guchar *temp;
guchar *temp2;
guchar *datapointer;
gint y;
gint x;
if (logodlg)
{
gdk_window_raise (logodlg->window);
return;
}
xdlg = logodlg =
gimp_dialog_new (_("About"), "fractalexplorer",
gimp_standard_help_func, "filters/fractalexplorer.html",
GTK_WIN_POS_MOUSE,
FALSE, TRUE, FALSE,
_("OK"), gtk_widget_destroy,
NULL, 1, NULL, TRUE, TRUE,
NULL);
gtk_signal_connect (GTK_OBJECT (xdlg), "destroy",
GTK_SIGNAL_FUNC (gtk_widget_destroyed),
&logodlg);
xframe = gtk_frame_new (NULL);
gtk_frame_set_shadow_type (GTK_FRAME (xframe), GTK_SHADOW_ETCHED_IN);
gtk_container_set_border_width (GTK_CONTAINER (xframe), 6);
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (xdlg)->vbox), xframe, TRUE, TRUE, 0);
xvbox = gtk_vbox_new (FALSE, 4);
gtk_container_set_border_width (GTK_CONTAINER (xvbox), 4);
gtk_container_add (GTK_CONTAINER (xframe), xvbox);
/* The logo frame & drawing area */
xhbox = gtk_hbox_new (FALSE, 5);
gtk_box_pack_start (GTK_BOX (xvbox), xhbox, FALSE, TRUE, 0);
xlogo_box = gtk_vbox_new (FALSE, 0);
gtk_box_pack_start (GTK_BOX (xhbox), xlogo_box, FALSE, FALSE, 0);
xframe2 = gtk_frame_new (NULL);
gtk_frame_set_shadow_type(GTK_FRAME(xframe2), GTK_SHADOW_IN);
gtk_box_pack_start(GTK_BOX(xlogo_box), xframe2, FALSE, FALSE, 0);
xpreview = gtk_preview_new (GTK_PREVIEW_COLOR);
gtk_preview_size (GTK_PREVIEW (xpreview), logo_width, logo_height);
temp = g_malloc ((logo_width + 10) * 3);
datapointer = header_data + logo_width * logo_height - 1;
for (y = 0; y < logo_height; y++)
{
temp2 = temp;
for (x = 0; x < logo_width; x++)
{
HEADER_PIXEL(datapointer, temp2);
temp2 += 3;
}
gtk_preview_draw_row (GTK_PREVIEW (xpreview),
temp,
0, y, logo_width);
}
g_free (temp);
gtk_container_add (GTK_CONTAINER (xframe2), xpreview);
gtk_widget_show (xpreview);
gtk_widget_show (xframe2);
gtk_widget_show (xlogo_box);
gtk_widget_show (xhbox);
xhbox = gtk_hbox_new (FALSE, 5);
gtk_box_pack_start (GTK_BOX (xvbox), xhbox, TRUE, TRUE, 0);
vpaned = gtk_vpaned_new ();
gtk_box_pack_start (GTK_BOX (xhbox), vpaned, TRUE, TRUE, 0);
gtk_widget_show (vpaned);
xframe3 = gtk_frame_new (NULL);
gtk_paned_add1 (GTK_PANED (vpaned), xframe3);
gtk_widget_show (xframe3);
xlabel = gtk_label_new ("\nCotting Software Productions\n"
"Quellenstrasse 10\n"
"CH-8005 Zuerich (Switzerland)\n\n"
"cotting@multimania.com\n"
"http://www.multimania.com/cotting\n\n"
"Fractal Chaos Explorer\nPlug-In for the GIMP\n"
"Version 2.00 Beta 2 (Multilingual)\n");
gtk_container_add (GTK_CONTAINER (xframe3), xlabel);
gtk_widget_show (xlabel);
xframe3 = gtk_frame_new (NULL);
gtk_paned_add2 (GTK_PANED (vpaned), xframe3);
gtk_widget_show (xframe3);
xlabel = gtk_label_new ("\nContains code from:\n\n"
"Daniel Cotting\n<cotting@mygale.org>\n"
"Peter Kirchgessner\n<Pkirchg@aol.com>\n"
"Scott Draves\n<spot@cs.cmu.edu>\n"
"Andy Thomas\n<alt@picnic.demon.co.uk>\n"
"and the GIMP distribution.\n");
gtk_container_add (GTK_CONTAINER (xframe3), xlabel);
gtk_widget_show (xlabel);
gtk_widget_show (xhbox);
gtk_widget_show (xvbox);
gtk_widget_show (xframe);
gtk_widget_show (xdlg);
}
/**********************************************************************
FUNCTION: dialog_change_scale
*********************************************************************/
void
dialog_change_scale (void)
{
ready_now = FALSE;
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->xmin), wvals.xmin);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->xmax), wvals.xmax);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->ymin), wvals.ymin);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->ymax), wvals.ymax);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->iter), wvals.iter);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->cx), wvals.cx);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->cy), wvals.cy);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->red), wvals.redstretch);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->green),wvals.greenstretch);
gtk_adjustment_set_value (GTK_ADJUSTMENT (elements->blue), wvals.bluestretch);
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (elements->type[wvals.fractaltype]), TRUE);
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (elements->redmode[wvals.redmode]), TRUE);
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (elements->greenmode[wvals.greenmode]), TRUE);
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (elements->bluemode[wvals.bluemode]), TRUE);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (elements->redinvert),
wvals.redinvert);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (elements->greeninvert),
wvals.greeninvert);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (elements->blueinvert),
wvals.blueinvert);
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (elements->colormode[wvals.colormode]), TRUE);
ready_now = TRUE;
}
/**********************************************************************
FUNCTION: save_options
*********************************************************************/
void
save_options (FILE * fp)
{
/* Save options */
fprintf (fp, "fractaltype: %i\n", wvals.fractaltype);
fprintf (fp, "xmin: %0.15f\n", wvals.xmin);
fprintf (fp, "xmax: %0.15f\n", wvals.xmax);
fprintf (fp, "ymin: %0.15f\n", wvals.ymin);
fprintf (fp, "ymax: %0.15f\n", wvals.ymax);
fprintf (fp, "iter: %0.15f\n", wvals.iter);
fprintf (fp, "cx: %0.15f\n", wvals.cx);
fprintf (fp, "cy: %0.15f\n", wvals.cy);
fprintf (fp, "redstretch: %0.15f\n", wvals.redstretch * 128.0);
fprintf (fp, "greenstretch: %0.15f\n", wvals.greenstretch * 128.0);
fprintf (fp, "bluestretch: %0.15f\n", wvals.bluestretch * 128.0);
fprintf (fp, "redmode: %i\n", wvals.redmode);
fprintf (fp, "greenmode: %i\n", wvals.greenmode);
fprintf (fp, "bluemode: %i\n", wvals.bluemode);
fprintf (fp, "redinvert: %i\n", wvals.redinvert);
fprintf (fp, "greeninvert: %i\n", wvals.greeninvert);
fprintf (fp, "blueinvert: %i\n", wvals.blueinvert);
fprintf (fp, "colormode: %i\n", wvals.colormode);
fputs ("#**********************************************************************\n", fp);
fprintf(fp, "<EOF>\n");
fputs ("#**********************************************************************\n", fp);
}
/**********************************************************************
FUNCTION: save_callback
*********************************************************************/
void
save_callback (void)
{
FILE *fp;
gchar *savename;
gchar *message;
savename = filename;
fp = fopen (savename, "wt+");
if (!fp)
{
message = g_strconcat (_("Error opening: %s"),
"\n",
_("Could not save."),
savename);
g_message (message);
g_free (message);
return;
}
/* Write header out */
fputs (FRACTAL_HEADER, fp);
fputs ("#**********************************************************************\n", fp);
fputs ("# This is a data file for the Fractal Explorer plug-in for the GIMP *\n", fp);
fputs ("# Get the plug-in at http://www.multimania.com/cotting *\n", fp);
fputs ("#**********************************************************************\n", fp);
save_options (fp);
if (ferror (fp))
g_message (_("Failed to write file\n"));
fclose (fp);
}
/**********************************************************************
FUNCTION: file_selection_ok
*********************************************************************/
void
file_selection_ok (GtkWidget *w,
GtkFileSelection *fs,
gpointer data)
{
gchar *filenamebuf;
struct stat filestat;
gint err;
filenamebuf = gtk_file_selection_get_filename (GTK_FILE_SELECTION(fs));
/* Get the name */
if (strlen (filenamebuf) == 0)
{
g_message (_("Save: No filename given"));
return;
}
/* Check if directory exists */
err = stat (filenamebuf, &filestat);
if (!err && S_ISDIR (filestat.st_mode))
{
/* Can't save to directory */
g_message (_("Save: Can't save to a directory"));
return;
}
filename = g_strdup (filenamebuf);
save_callback ();
gtk_widget_destroy (GTK_WIDGET (fs));
}
/**********************************************************************
FUNCTION: load_file_selection_ok
**********************************************************************/
void
load_file_selection_ok (GtkWidget *w,
GtkFileSelection *fs,
gpointer data)
{
struct stat filestat;
gint err;
filename = g_strdup (gtk_file_selection_get_filename (GTK_FILE_SELECTION (fs)));
err = stat (filename, &filestat);
if (!err && S_ISREG (filestat.st_mode))
{
explorer_load ();
}
gtk_widget_destroy (GTK_WIDGET (fs));
gtk_widget_show (maindlg);
dialog_change_scale ();
set_cmap_preview ();
dialog_update_preview ();
}
/**********************************************************************
FUNCTION: create_load_file_selection
*********************************************************************/
void
create_load_file_selection (void)
{
static GtkWidget *window = NULL;
/* Load a single object */
if (!window)
{
window = gtk_file_selection_new (_("Load Fractal Parameters"));
gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_NONE);
gtk_signal_connect (GTK_OBJECT (window), "destroy",
GTK_SIGNAL_FUNC (gtk_widget_destroyed),
&window);
gtk_signal_connect (GTK_OBJECT (GTK_FILE_SELECTION (window)->ok_button),
"clicked",
GTK_SIGNAL_FUNC (load_file_selection_ok),
(gpointer) window);
gimp_help_set_help_data (GTK_FILE_SELECTION(window)->ok_button,
_("Click here to load your file"), NULL);
gtk_signal_connect_object (GTK_OBJECT (GTK_FILE_SELECTION (window)->cancel_button),
"clicked",
GTK_SIGNAL_FUNC (gtk_widget_destroy),
GTK_OBJECT (window));
gimp_help_set_help_data (GTK_FILE_SELECTION(window)->cancel_button,
_("Click here to cancel load procedure"), NULL);
}
if (!GTK_WIDGET_VISIBLE (window))
gtk_widget_show (window);
}
/**********************************************************************
FUNCTION: create_file_selection
*********************************************************************/
void
create_file_selection (void)
{
static GtkWidget *window = NULL;
if (!window)
{
window = gtk_file_selection_new (_("Save Fractal Parameters"));
gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_NONE);
gtk_signal_connect (GTK_OBJECT (window), "destroy",
GTK_SIGNAL_FUNC (gtk_widget_destroyed),
&window);
gtk_signal_connect (GTK_OBJECT (GTK_FILE_SELECTION (window)->ok_button),
"clicked",
GTK_SIGNAL_FUNC (file_selection_ok),
(gpointer) window);
gimp_help_set_help_data (GTK_FILE_SELECTION(window)->ok_button,
_("Click here to save your file"), NULL);
gtk_signal_connect_object (GTK_OBJECT (GTK_FILE_SELECTION(window)->cancel_button),
"clicked",
GTK_SIGNAL_FUNC (gtk_widget_destroy),
GTK_OBJECT (window));
gimp_help_set_help_data (GTK_FILE_SELECTION (window)->cancel_button,
_("Click here to cancel save procedure"), NULL);
}
if (tpath)
{
gtk_file_selection_set_filename (GTK_FILE_SELECTION (window), tpath);
}
else if (fractalexplorer_path_list)
{
gchar *dir;
dir = gimp_path_get_user_writable_dir (fractalexplorer_path_list);
if (!dir)
dir = g_strdup (gimp_directory ());
gtk_file_selection_set_filename (GTK_FILE_SELECTION (window), dir);
g_free (dir);
}
else
{
gtk_file_selection_set_filename (GTK_FILE_SELECTION (window), "/tmp");
}
if (!GTK_WIDGET_VISIBLE (window))
gtk_widget_show (window);
}
/**********************************************************************
FUNCTION: get_line
*********************************************************************/
gchar *
get_line (gchar *buf,
gint s,
FILE *from,
gint init)
{
gint slen;
gchar *ret;
if (init)
line_no = 1;
else
line_no++;
do
{
ret = fgets (buf, s, from);
}
while (!ferror (from) && buf[0] == '#');
slen = strlen (buf);
/* The last newline is a pain */
if (slen > 0)
buf[slen - 1] = '\0';
if (ferror (from))
{
g_warning ("Error reading file");
return NULL;
}
return ret;
}
/**********************************************************************
FUNCTION: load_options
*********************************************************************/
gint
load_options (fractalexplorerOBJ *xxx,
FILE *fp)
{
gchar load_buf[MAX_LOAD_LINE];
gchar str_buf[MAX_LOAD_LINE];
gchar opt_buf[MAX_LOAD_LINE];
/* default values */
xxx->opts.fractaltype = 0;
xxx->opts.xmin = -2.0;
xxx->opts.xmax = 2.0;
xxx->opts.ymin = -1.5;
xxx->opts.ymax = 1.5;
xxx->opts.iter = 50.0;
xxx->opts.cx = -0.75;
xxx->opts.cy = -0.2;
xxx->opts.colormode = 0;
xxx->opts.redstretch = 1.0;
xxx->opts.greenstretch = 1.0;
xxx->opts.bluestretch = 1.0;
xxx->opts.redmode = 1;
xxx->opts.greenmode = 1;
xxx->opts.bluemode = 1;
xxx->opts.redinvert = 0;
xxx->opts.greeninvert = 0;
xxx->opts.blueinvert = 0;
xxx->opts.alwayspreview = 1;
xxx->opts.ncolors = 256; /* not saved */
get_line (load_buf, MAX_LOAD_LINE, fp, 0);
while (!feof (fp) && strcmp (load_buf, "<EOF>"))
{
/* Get option name */
sscanf (load_buf, "%s %s", str_buf, opt_buf);
if (!strcmp (str_buf, "fractaltype:"))
{
gint sp = 0;
sp = atoi (opt_buf);
if (sp < 0)
return -1;
xxx->opts.fractaltype = sp;
}
else if (!strcmp (str_buf, "xmin:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.xmin = sp;
}
else if (!strcmp (str_buf, "xmax:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.xmax = sp;
}
else if (!strcmp(str_buf, "ymin:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.ymin = sp;
}
else if (!strcmp (str_buf, "ymax:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.ymax = sp;
}
else if (!strcmp(str_buf, "redstretch:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.redstretch = sp / 128.0;
}
else if (!strcmp(str_buf, "greenstretch:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.greenstretch = sp / 128.0;
}
else if (!strcmp (str_buf, "bluestretch:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.bluestretch = sp / 128.0;
}
else if (!strcmp (str_buf, "iter:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.iter = sp;
}
else if (!strcmp(str_buf, "cx:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.cx = sp;
}
else if (!strcmp (str_buf, "cy:"))
{
gdouble sp = 0;
sp = atof (opt_buf);
xxx->opts.cy = sp;
}
else if (!strcmp(str_buf, "redmode:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.redmode = sp;
}
else if (!strcmp(str_buf, "greenmode:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.greenmode = sp;
}
else if (!strcmp(str_buf, "bluemode:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.bluemode = sp;
}
else if (!strcmp (str_buf, "redinvert:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.redinvert = sp;
}
else if (!strcmp (str_buf, "greeninvert:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.greeninvert = sp;
}
else if (!strcmp(str_buf, "blueinvert:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.blueinvert = sp;
}
else if (!strcmp (str_buf, "colormode:"))
{
gint sp = 0;
sp = atoi (opt_buf);
xxx->opts.colormode = sp;
}
get_line (load_buf, MAX_LOAD_LINE, fp, 0);
}
return 0;
}
/**********************************************************************
FUNCTION: explorer_load
*********************************************************************/
void
explorer_load (void)
{
FILE *fp;
gchar load_buf[MAX_LOAD_LINE];
g_assert (filename != NULL);
fp = fopen (filename, "rt");
if (!fp)
{
g_warning ("Error opening: %s", filename);
return;
}
get_line (load_buf, MAX_LOAD_LINE, fp, 1);
if (strncmp (FRACTAL_HEADER, load_buf, strlen (load_buf)))
{
g_message (_("File '%s' is not a FractalExplorer file"), filename);
return;
}
if (load_options (current_obj,fp))
{
g_message (_("File '%s' is corrupt.\nLine %d Option section incorrect"),
filename, line_no);
return;
}
wvals = current_obj->opts;
fclose (fp);
}