gimp/app/paths_dialog.c

1584 lines
42 KiB
C

/* The GIMP -- an image manipulation program
* Copyright (C) 1999 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 2 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, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
* Some of this code is based on the layers_dialog box code.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "gdk/gdkkeysyms.h"
#include "appenv.h"
#include "draw_core.h"
#include "actionarea.h"
#include "buildmenu.h"
#include "colormaps.h"
#include "drawable.h"
#include "errors.h"
#include "floating_sel.h"
#include "gdisplay.h"
#include "gimage.h"
#include "gimage_mask.h"
#include "gimprc.h"
#include "gimpset.h"
#include "general.h"
#include "image_render.h"
#include "interface.h"
#include "layers_dialog.h"
#include "layers_dialogP.h"
#include "ops_buttons.h"
#include "paint_funcs.h"
#include "bezier_select.h"
#include "bezier_selectP.h"
#include "pathsP.h"
#include "paths_dialog.h"
#include "resize.h"
#include "session.h"
#include "undo.h"
#include "libgimp/gimpintl.h"
#include "tools/new.xpm"
#include "tools/duplicate.xpm"
#include "tools/delete.xpm"
#include "tools/pennorm.xpm"
#include "tools/penadd.xpm"
#include "tools/pendel.xpm"
#include "tools/penedit.xpm"
#include "tools/penstroke.xpm"
#include "tools/ptoselection.xpm"
#include "tools/path.xbm"
#define PREVIEW_EVENT_MASK GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK | GDK_ENTER_NOTIFY_MASK
#define PATHS_LIST_WIDTH 200
#define PATHS_LIST_HEIGHT 150
typedef struct {
GtkWidget *paths_list;
GtkWidget *vbox;
GtkWidget *ops_menu;
GtkAccelGroup *accel_group;
double ratio;
int image_width, image_height;
int gimage_width, gimage_height;
/* state information */
gint selsigid;
GimpImage * gimage;
GdkGC * gc;
GdkColor black;
GdkColor white;
gint selected_row_num;
gboolean been_selected;
PATHIMAGELISTP current_path_list;
} PATHSLIST, *PATHSLISTP;
static PATHSLISTP paths_dialog = NULL;
typedef struct {
GdkPixmap *paths_pixmap;
GString *text;
BZPATHP bzp;
} PATHWIDGET, *PATHWIDGETP;
static gint path_widget_preview_events (GtkWidget *, GdkEvent *);
static void paths_dialog_realized (GtkWidget *widget);
static void paths_select_row (GtkWidget *widget,gint row,gint column,GdkEventButton *event,gpointer data);
static void paths_unselect_row (GtkWidget *widget,gint row,gint column,GdkEventButton *event,gpointer data);
static gint paths_list_events (GtkWidget *widget,GdkEvent *event);
static void paths_dialog_new_path_callback (GtkWidget *, gpointer);
static void paths_dialog_delete_path_callback (GtkWidget *, gpointer);
static void paths_dialog_map_callback (GtkWidget *w,gpointer client_data);
static void paths_dialog_unmap_callback(GtkWidget *w,gpointer client_data);
static void paths_dialog_dup_path_callback(GtkWidget *w,gpointer client_data);
static void paths_dialog_stroke_path_callback(GtkWidget *w,gpointer client_data);
static void paths_dialog_path_to_sel_callback(GtkWidget *w,gpointer client_data);
static void paths_dialog_destroy_cb (GimpImage *image);
static void paths_update_paths(gpointer data,gint row);
static GSList * bzpoints_copy(GSList *list);
static void bzpoints_free(GSList *list);
static void paths_update_preview(BezierSelect *bezier_sel);
static void paths_dialog_preview_extents (void);
static void paths_dialog_new_point_callback (GtkWidget *, gpointer);
static void paths_dialog_add_point_callback (GtkWidget *, gpointer);
static void paths_dialog_delete_point_callback (GtkWidget *, gpointer);
static void paths_dialog_edit_point_callback (GtkWidget *, gpointer);
#define NEW_PATH_BUTTON 1
#define DUP_PATH_BUTTON 2
#define PATH_TO_SEL_BUTTON 3
#define STROKE_PATH_BUTTON 4
#define DEL_PATH_BUTTON 5
static MenuItem paths_ops[] =
{
{ N_("New Path"), 'N', GDK_CONTROL_MASK,
paths_dialog_new_path_callback, NULL, NULL, NULL },
{ N_("Duplicate Path"), 'C', GDK_CONTROL_MASK,
paths_dialog_dup_path_callback, NULL, NULL, NULL },
{ N_("Path to Selection"), 'S', GDK_CONTROL_MASK,
paths_dialog_path_to_sel_callback, NULL, NULL, NULL },
{ N_("Stroke Path"), 'T', GDK_CONTROL_MASK,
paths_dialog_stroke_path_callback, NULL, NULL, NULL },
{ N_("Delete Path"), 'D', GDK_CONTROL_MASK,
paths_dialog_delete_path_callback, NULL, NULL, NULL },
{ NULL, 0, 0, NULL, NULL, NULL, NULL }
};
static OpsButton paths_ops_buttons[] =
{
{ new_xpm, paths_dialog_new_path_callback, N_("New Path"), NULL },
{ duplicate_xpm, paths_dialog_dup_path_callback, N_("Duplicate Path"), NULL },
{ ptoselection_xpm, paths_dialog_path_to_sel_callback, N_("Path to Selection"), NULL },
{ penstroke_xpm, paths_dialog_stroke_path_callback, N_("Stroke Path"), NULL },
{ delete_xpm, paths_dialog_delete_path_callback, N_("Delete Path"), NULL },
{ NULL, NULL, NULL, NULL }
};
#define POINT_NEW_BUTTON 1
#define POINT_ADD_BUTTON 2
#define POINT_DEL_BUTTON 3
#define POINT_EDIT_BUTTON 4
static OpsButton point_ops_buttons[] =
{
{ pennorm_xpm, paths_dialog_new_point_callback, N_("New Point"), NULL },
{ penadd_xpm, paths_dialog_add_point_callback, N_("Add Point"), NULL },
{ pendel_xpm, paths_dialog_delete_point_callback, N_("Delete Point"), NULL },
{ penedit_xpm, paths_dialog_edit_point_callback, N_("Edit Point"), NULL },
{ NULL, NULL, NULL, NULL }
};
static void
paths_ops_button_set_sensitive(gint but,gboolean sensitive)
{
switch(but)
{
case NEW_PATH_BUTTON:
gtk_widget_set_sensitive(paths_ops[0].widget,sensitive);
gtk_widget_set_sensitive(paths_ops_buttons[0].widget,sensitive);
break;
case DUP_PATH_BUTTON:
gtk_widget_set_sensitive(paths_ops[1].widget,sensitive);
gtk_widget_set_sensitive(paths_ops_buttons[1].widget,sensitive);
break;
case PATH_TO_SEL_BUTTON:
gtk_widget_set_sensitive(paths_ops[2].widget,sensitive);
gtk_widget_set_sensitive(paths_ops_buttons[2].widget,sensitive);
break;
case STROKE_PATH_BUTTON:
gtk_widget_set_sensitive(paths_ops[3].widget,sensitive);
gtk_widget_set_sensitive(paths_ops_buttons[3].widget,sensitive);
break;
case DEL_PATH_BUTTON:
gtk_widget_set_sensitive(paths_ops[4].widget,sensitive);
gtk_widget_set_sensitive(paths_ops_buttons[4].widget,sensitive);
break;
default:
g_warning(_("paths_ops_button_set_sensitive:: invalid button specified"));
break;
}
}
static void
point_ops_button_set_sensitive(gint but,gboolean sensitive)
{
switch(but)
{
case POINT_NEW_BUTTON:
gtk_widget_set_sensitive(point_ops_buttons[0].widget,sensitive);
break;
case POINT_ADD_BUTTON:
gtk_widget_set_sensitive(point_ops_buttons[1].widget,sensitive);
break;
case POINT_DEL_BUTTON:
gtk_widget_set_sensitive(point_ops_buttons[2].widget,sensitive);
break;
case POINT_EDIT_BUTTON:
gtk_widget_set_sensitive(point_ops_buttons[3].widget,sensitive);
break;
default:
g_warning(_("point_ops_button_set_sensitive:: invalid button specified"));
break;
}
}
static void
paths_list_destroy (GtkWidget *w)
{
paths_dialog = NULL;
}
GtkWidget * paths_dialog_create()
{
GtkWidget *vbox;
GtkWidget *paths_list;
GtkWidget *scrolled_win;
GtkWidget *button_box;
if(!paths_dialog)
{
paths_dialog = g_new0(PATHSLIST,1);
/* The paths box */
paths_dialog->vbox = vbox = gtk_vbox_new (FALSE, 1);
/* The point operations */
button_box = ops_button_box_new (lc_shell,
tool_tips,
point_ops_buttons,
OPS_BUTTON_RADIO);
gtk_container_set_border_width(GTK_CONTAINER(button_box),7);
gtk_box_pack_start (GTK_BOX (vbox), button_box, FALSE, TRUE, 2);
gtk_widget_show (button_box);
gtk_container_border_width (GTK_CONTAINER (vbox), 2);
scrolled_win = gtk_scrolled_window_new (NULL, NULL);
gtk_box_pack_start(GTK_BOX(vbox), scrolled_win, TRUE, TRUE, 0);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_win),
GTK_POLICY_AUTOMATIC,
GTK_POLICY_ALWAYS);
paths_dialog->paths_list = paths_list = gtk_clist_new (1);
gtk_signal_connect (GTK_OBJECT (vbox), "destroy",
(GtkSignalFunc) paths_list_destroy, NULL);
/* gtk_clist_set_column_title(GTK_CLIST(paths_list), 0, "col1"); */
/* gtk_clist_column_titles_show(GTK_CLIST(paths_list)); */
gtk_container_add (GTK_CONTAINER (scrolled_win), paths_list);
gtk_clist_set_selection_mode (GTK_CLIST (paths_list), GTK_SELECTION_BROWSE);
gtk_signal_connect (GTK_OBJECT (paths_list), "event",
(GtkSignalFunc) paths_list_events,
paths_dialog);
gtk_container_set_focus_vadjustment (GTK_CONTAINER (paths_list),
gtk_scrolled_window_get_vadjustment (GTK_SCROLLED_WINDOW (scrolled_win)));
GTK_WIDGET_UNSET_FLAGS (GTK_SCROLLED_WINDOW (scrolled_win)->vscrollbar, GTK_CAN_FOCUS);
paths_dialog->selsigid = gtk_signal_connect(GTK_OBJECT(paths_list), "select_row",
GTK_SIGNAL_FUNC(paths_select_row),
(gpointer) NULL);
gtk_signal_connect(GTK_OBJECT(paths_list), "unselect_row",
GTK_SIGNAL_FUNC(paths_unselect_row),
(gpointer) NULL);
gtk_widget_show(scrolled_win);
gtk_widget_show(paths_list);
gtk_signal_connect(GTK_OBJECT(vbox),"realize",
(GtkSignalFunc)paths_dialog_realized,
(gpointer)NULL);
gtk_widget_show (vbox);
/* The ops buttons */
button_box = ops_button_box_new (lc_shell,
tool_tips,
paths_ops_buttons,
OPS_BUTTON_NORMAL);
gtk_box_pack_start (GTK_BOX (vbox), button_box, FALSE, FALSE, 2);
gtk_widget_show (button_box);
/* Set up signals for map/unmap for the accelerators */
paths_dialog->accel_group = gtk_accel_group_new ();
gtk_signal_connect (GTK_OBJECT (vbox), "map",
(GtkSignalFunc) paths_dialog_map_callback,
NULL);
gtk_signal_connect (GTK_OBJECT (vbox), "unmap",
(GtkSignalFunc) paths_dialog_unmap_callback,
NULL);
paths_dialog->ops_menu = build_menu (paths_ops,paths_dialog->accel_group);
paths_ops_button_set_sensitive(DUP_PATH_BUTTON,FALSE);
paths_ops_button_set_sensitive(DEL_PATH_BUTTON,FALSE);
paths_ops_button_set_sensitive(STROKE_PATH_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_ADD_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_DEL_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_NEW_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_EDIT_BUTTON,FALSE);
point_ops_button_set_sensitive(PATH_TO_SEL_BUTTON,FALSE);
}
return paths_dialog->vbox;
}
static void paths_dialog_realized(GtkWidget *widget)
{
GdkColormap *colormap;
gchar dash_list[2]= {3,3};
paths_dialog->gc = gdk_gc_new(widget->window);
gdk_gc_set_dashes(paths_dialog->gc,2,dash_list,2);
colormap = gtk_widget_get_colormap(paths_dialog->paths_list);
gdk_color_parse("black", &paths_dialog->black);
gdk_color_alloc(colormap, &paths_dialog->black);
gdk_color_parse("white", &paths_dialog->white);
gdk_color_alloc(colormap, &paths_dialog->white);
}
/* Clears out row when list element is deleted/destroyed */
static void
clear_pathwidget(gpointer data)
{
PATHWIDGETP pwidget = data;
if(pwidget)
{
g_free(pwidget);
}
}
static void
bzpoint_free(gpointer data,gpointer user_data)
{
BZPOINTP bzpoint = data;
g_free(bzpoint);
}
static void
bzpath_free(gpointer data,gpointer user_data)
{
BZPATHP bzp = data;
g_return_if_fail(bzp != NULL);
g_string_free(bzp->name,TRUE);
bzpoints_free(bzp->bezier_details);
g_free(bzp);
}
static BZPATHP
bzpath_dialog_new(gint name_seed, gpointer udata)
{
BZPATHP bzp = g_new0(BZPATH,1);
GString *s = g_string_new (NULL);
g_string_sprintf (s, "path %d",name_seed);
bzp->name = s;
bzp->bezier_details = (GSList *)udata; /* If called via button/menu this will be NULL */
return bzp;
}
static BZPATHP
bzpath_copy(BZPATHP bzp)
{
BZPATHP bzp_copy = g_new0(BZPATH,1);
GString *s = g_string_new (NULL);
g_string_sprintf (s, "%s copy",bzp->name->str);
bzp_copy->name = s;
bzp_copy->closed = bzp->closed;
bzp_copy->state = bzp->state;
bzp_copy->bezier_details = bzpoints_copy(bzp->bezier_details);
return bzp_copy;
}
static void
bzpath_close(BZPATHP bzp)
{
BZPOINTP pdata;
BZPOINTP bzpoint;
/* bzpaths are only really closed when converted to the BezierSelect ones */
bzp->closed = 1;
/* first point */
pdata = (BZPOINTP)bzp->bezier_details->data;
if(g_slist_length(bzp->bezier_details) < 5)
{
int i;
for (i = 0 ; i < 2 ; i++)
{
bzpoint = g_new0(BZPOINT,1);
bzpoint->type = (i & 1)?BEZIER_ANCHOR:BEZIER_CONTROL;
bzpoint->x = pdata->x+i;
bzpoint->y = pdata->y+i;
bzp->bezier_details = g_slist_append(bzp->bezier_details,bzpoint);
}
}
bzpoint = g_new0(BZPOINT,1);
pdata = (BZPOINTP)bzp->bezier_details->data;
bzpoint->type = BEZIER_CONTROL;
bzpoint->x = pdata->x;
bzpoint->y = pdata->y;
bzp->bezier_details = g_slist_append(bzp->bezier_details,bzpoint);
}
static void
beziersel_free(BezierSelect *bezier_sel)
{
bezier_select_reset (bezier_sel);
g_free(bezier_sel);
}
static BezierSelect *
bzpath_to_beziersel(BZPATHP bzp)
{
BezierSelect *bezier_sel;
GSList *list;
if(!bzp)
{
g_warning("bzpath_to_beziersel:: NULL bzp");
}
list = bzp->bezier_details;
bezier_sel = g_new0 (BezierSelect,1);
bezier_sel->num_points = 0;
bezier_sel->mask = NULL;
bezier_sel->core = NULL; /* not required will be reset in bezier code */
bezier_select_reset (bezier_sel);
bezier_sel->closed = bzp->closed;
/* bezier_sel->state = BEZIER_ADD; */
bezier_sel->state = bzp->state;
while(list)
{
BZPOINTP pdata;
pdata = (BZPOINTP)list->data;
bezier_add_point(bezier_sel,pdata->type,pdata->x,pdata->y);
list = g_slist_next(list);
}
if ( bezier_sel->closed )
{
bezier_sel->last_point->next = bezier_sel->points;
bezier_sel->points->prev = bezier_sel->last_point;
bezier_sel->cur_anchor = bezier_sel->points;
bezier_sel->cur_control = bezier_sel-> points->next;
}
return bezier_sel;
}
static void
pathimagelist_free(PATHIMAGELISTP iml)
{
g_return_if_fail(iml != NULL);
if(iml->bz_paths)
{
g_slist_foreach(iml->bz_paths,bzpath_free,NULL);
g_slist_free(iml->bz_paths);
}
g_free(iml);
}
static void
bz_change_name_row_to(gint row,gchar *text)
{
PATHWIDGETP pwidget;
pwidget = (PATHWIDGETP)gtk_clist_get_row_data(GTK_CLIST(paths_dialog->paths_list),row);
if(!pwidget)
return;
g_string_free(pwidget->bzp->name,TRUE);
pwidget->bzp->name = g_string_new(text);
}
static void
paths_set_dash_line(GdkGC *gc,gboolean state)
{
if(state)
{
gdk_gc_set_line_attributes(gc,0,GDK_LINE_ON_OFF_DASH,GDK_CAP_BUTT,GDK_JOIN_ROUND);
}
else
{
gdk_gc_set_line_attributes(gc,0,GDK_LINE_SOLID,GDK_CAP_BUTT,GDK_JOIN_ROUND);
}
}
static void
clear_pixmap_preview(PATHWIDGETP pwidget)
{
gchar *rgb_buf;
rgb_buf = g_new0(gchar,
(paths_dialog->image_width + 4)
*(paths_dialog->image_height + 4)*3);
memset(rgb_buf,0xFF,(paths_dialog->image_width + 4)
*(paths_dialog->image_height + 4)*3);
gdk_draw_rgb_image (pwidget->paths_pixmap,
paths_dialog->gc,
0,
0,
paths_dialog->image_width + 4,
paths_dialog->image_height + 4,
GDK_RGB_DITHER_NORMAL,
rgb_buf,
(paths_dialog->image_width + 4)*3);
gdk_gc_set_foreground(paths_dialog->gc, &paths_dialog->black);
paths_set_dash_line(paths_dialog->gc,FALSE);
gdk_draw_rectangle(pwidget->paths_pixmap,
paths_dialog->gc, FALSE, 0, 0,
paths_dialog->image_width+3,
paths_dialog->image_height+3);
gdk_draw_rectangle(pwidget->paths_pixmap,
paths_dialog->gc, FALSE, 1, 1,
paths_dialog->image_width+1,
paths_dialog->image_height+1);
}
/* insrow == -1 -> append else insert at insrow */
void paths_add_path(BZPATHP bzp,gint insrow)
{
/* Create a new entry in the list */
PATHWIDGETP pwidget;
gint row;
gchar *row_data[1];
pwidget = g_new0(PATHWIDGET,1);
if(!GTK_WIDGET_REALIZED(paths_dialog->vbox))
gtk_widget_realize(paths_dialog->vbox);
paths_dialog_preview_extents();
if(preview_size)
{
/* Need to add this to the list */
pwidget->paths_pixmap = gdk_pixmap_new(paths_dialog->vbox->window,
paths_dialog->image_width + 4,
paths_dialog->image_height + 4,
-1);
clear_pixmap_preview(pwidget);
}
else
{
pwidget->paths_pixmap =
gdk_pixmap_create_from_data (paths_dialog->vbox->window,
path_bits,
paths_dialog->image_width,
paths_dialog->image_height,
-1,
&paths_dialog->vbox->style->fg[GTK_STATE_SELECTED],
&paths_dialog->vbox->style->bg[GTK_STATE_SELECTED]);
}
gtk_clist_set_row_height(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->image_height + 6);
row_data[0] = "";
if(insrow == -1)
row = gtk_clist_append(GTK_CLIST(paths_dialog->paths_list),
row_data);
else
row = gtk_clist_insert(GTK_CLIST(paths_dialog->paths_list),
insrow,
row_data);
gtk_clist_set_pixtext(GTK_CLIST(paths_dialog->paths_list),
row,
0,
bzp->name->str,
2,
pwidget->paths_pixmap,
NULL);
gtk_clist_set_row_data_full(GTK_CLIST(paths_dialog->paths_list),
row,
(gpointer)pwidget,
clear_pathwidget);
gtk_signal_handler_block(GTK_OBJECT(paths_dialog->paths_list),paths_dialog->selsigid);
gtk_clist_select_row(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->current_path_list->last_selected_row,
0);
gtk_signal_handler_unblock(GTK_OBJECT(paths_dialog->paths_list),paths_dialog->selsigid);
pwidget->bzp = bzp;
}
static void
paths_dialog_preview_extents ()
{
GImage *gimage;
if (!paths_dialog)
return;
if (!(gimage = paths_dialog->gimage))
return;
gimage = paths_dialog->gimage;
paths_dialog->gimage_width = gimage->width;
paths_dialog->gimage_height = gimage->height;
/* Get the image width and height variables, based on the gimage */
if (gimage->width > gimage->height)
paths_dialog->ratio = (double) preview_size / (double) gimage->width;
else
paths_dialog->ratio = (double) preview_size / (double) gimage->height;
if (preview_size)
{
paths_dialog->image_width = (int) (paths_dialog->ratio * gimage->width);
paths_dialog->image_height = (int) (paths_dialog->ratio * gimage->height);
if (paths_dialog->image_width < 1) paths_dialog->image_width = 1;
if (paths_dialog->image_height < 1) paths_dialog->image_height = 1;
}
else
{
paths_dialog->image_width = path_width;
paths_dialog->image_height = path_height;
}
}
static gint
path_widget_preview_events (GtkWidget *widget,
GdkEvent *event)
{
GdkEventButton *bevent;
switch (event->type)
{
case GDK_BUTTON_PRESS:
/* Control-button press disables the application of the mask */
bevent = (GdkEventButton *) event;
break;
case GDK_EXPOSE:
if (preview_size)
{
/* layer_widget_preview_redraw (layer_widget, preview_type); */
/* gdk_draw_pixmap (widget->window, */
/* widget->style->black_gc, */
/* *pixmap, */
/* 0, 0, 2, 2, */
/* layersD->image_width, */
/* layersD->image_height); */
}
break;
default:
break;
}
return FALSE;
}
static void
paths_select_row(GtkWidget *widget,
gint row,
gint column,
GdkEventButton *event,
gpointer data)
{
PATHWIDGETP pwidget;
BZPATHP bzp;
BezierSelect * bsel;
GDisplay *gdisp;
pwidget = (PATHWIDGETP)gtk_clist_get_row_data(GTK_CLIST(widget),row);
if(!pwidget ||
(paths_dialog->current_path_list->last_selected_row == row &&
paths_dialog->been_selected == TRUE))
return;
paths_dialog->selected_row_num = row;
paths_dialog->current_path_list->last_selected_row = row;
paths_dialog->been_selected = TRUE;
bzp = (BZPATHP)g_slist_nth_data(paths_dialog->current_path_list->bz_paths,row);
g_return_if_fail(bzp != NULL);
bsel = bzpath_to_beziersel(bzp);
gdisp = gdisplays_check_valid(paths_dialog->current_path_list->gdisp,
paths_dialog->gimage);
if(!gdisp)
{
g_warning("Lost image which bezier curve belonged to");
return;
}
bezier_paste_bezierselect_to_current(gdisp,bsel);
paths_update_preview(bsel);
beziersel_free(bsel);
/* Draw white as the border */
/* gdk_gc_set_foreground(paths_dialog->gc, &paths_dialog->black); */
/* gdk_draw_rectangle(pwidget->paths_pixmap, */
/* paths_dialog->gc, FALSE, 0, 0, */
/* paths_dialog->image_width+3, */
/* paths_dialog->image_height+3); */
/* gdk_draw_rectangle(pwidget->paths_pixmap, */
/* paths_dialog->gc, FALSE, 1, 1, */
/* paths_dialog->image_width+1, */
/* paths_dialog->image_height+1); */
}
static void
paths_unselect_row(GtkWidget *widget,
gint row,
gint column,
GdkEventButton *event,
gpointer data)
{
PATHWIDGETP pwidget;
pwidget = (PATHWIDGETP)gtk_clist_get_row_data(GTK_CLIST(widget),row);
if(!pwidget)
return;
/* gdk_gc_set_foreground(paths_dialog->gc, &paths_dialog->white); */
/* gdk_draw_rectangle(pwidget->paths_pixmap, */
/* paths_dialog->gc, FALSE, 0, 0, */
/* paths_dialog->image_width+3, */
/* paths_dialog->image_height+3); */
/* gdk_draw_rectangle(pwidget->paths_pixmap, */
/* paths_dialog->gc, FALSE, 1, 1, */
/* paths_dialog->image_width+1, */
/* paths_dialog->image_height+1); */
}
void
paths_dialog_update (GimpImage* gimage)
{
PATHIMAGELISTP new_path_list;
GSList *plist;
gint loop;
gint tmprow;
if (!paths_dialog || !gimage)
return;
/* The last pointer comparison forces update if something has changed
* under our feet.
*/
if (paths_dialog->gimage == gimage &&
paths_dialog->current_path_list == (PATHIMAGELISTP)gimp_image_get_paths(gimage))
return;
paths_dialog->gimage=gimage;
paths_dialog_preview_extents ();
if(!GTK_WIDGET_REALIZED(paths_dialog->vbox))
gtk_widget_realize(paths_dialog->vbox);
/* ALT removed & replaced return;*/
/* clear clist out */
gtk_clist_freeze(GTK_CLIST(paths_dialog->paths_list));
gtk_clist_clear(GTK_CLIST(paths_dialog->paths_list));
gtk_clist_thaw(GTK_CLIST(paths_dialog->paths_list));
/* Find bz list */
new_path_list = (PATHIMAGELISTP)gimp_image_get_paths(gimage);
paths_dialog->current_path_list = new_path_list;
paths_dialog->been_selected = FALSE;
if(!new_path_list)
{
/* No list assoc with this image */
paths_ops_button_set_sensitive(DUP_PATH_BUTTON,FALSE);
paths_ops_button_set_sensitive(DEL_PATH_BUTTON,FALSE);
paths_ops_button_set_sensitive(STROKE_PATH_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_ADD_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_DEL_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_NEW_BUTTON,FALSE);
point_ops_button_set_sensitive(POINT_EDIT_BUTTON,FALSE);
point_ops_button_set_sensitive(PATH_TO_SEL_BUTTON,FALSE);
return;
}
else
{
paths_ops_button_set_sensitive(DUP_PATH_BUTTON,TRUE);
paths_ops_button_set_sensitive(DEL_PATH_BUTTON,TRUE);
paths_ops_button_set_sensitive(STROKE_PATH_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_ADD_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_DEL_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_NEW_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_EDIT_BUTTON,TRUE);
point_ops_button_set_sensitive(PATH_TO_SEL_BUTTON,TRUE);
}
/* update the clist to reflect this images bz list */
/* go around the image list populating the clist */
if(gimage != new_path_list->gimage)
{
g_warning(_("paths list: internal list error"));
}
plist = new_path_list->bz_paths;
loop = 0;
tmprow = paths_dialog->current_path_list->last_selected_row;
while(plist)
{
paths_update_paths(plist->data,loop);
loop++;
plist = g_slist_next(plist);
}
paths_dialog->current_path_list->last_selected_row = tmprow;
paths_dialog->selected_row_num = tmprow;
/* g_slist_foreach(plist,paths_update_paths,NULL); */
/* select last one */
gtk_signal_handler_block(GTK_OBJECT(paths_dialog->paths_list),paths_dialog->selsigid);
gtk_clist_select_row(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->current_path_list->last_selected_row,
0);
gtk_signal_handler_unblock(GTK_OBJECT(paths_dialog->paths_list),paths_dialog->selsigid);
gtk_clist_moveto(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->current_path_list->last_selected_row,
0,
0.5,
0.0);
}
static void
paths_update_paths(gpointer data,gint row)
{
BZPATHP bzp;
BezierSelect * bezier_sel;
paths_add_path((bzp = (BZPATHP)data),-1);
/* Now fudge the drawing....*/
bezier_sel = bzpath_to_beziersel(bzp);
paths_dialog->current_path_list->last_selected_row = row;
paths_update_preview(bezier_sel);
beziersel_free(bezier_sel);
}
static void
do_rename_paths_callback(GtkWidget *widget, gpointer call_data, gpointer client_data)
{
gchar *text;
GdkBitmap *mask;
guint8 spacing;
GdkPixmap *pixmap;
if(!(GTK_CLIST(call_data)->selection))
return;
text = g_strdup(client_data);
gtk_clist_get_pixtext(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->selected_row_num,
0,
NULL,
&spacing,
&pixmap,
&mask);
gtk_clist_set_pixtext(GTK_CLIST(call_data),
paths_dialog->selected_row_num,
0,
text,
spacing,
pixmap,
mask);
bz_change_name_row_to(paths_dialog->selected_row_num,text);
}
static void
paths_dialog_edit_path_query(GtkWidget *widget)
{
gchar *text;
gint ret;
GdkBitmap *mask;
/* Get the current name */
ret = gtk_clist_get_pixtext(GTK_CLIST(paths_dialog->paths_list),
paths_dialog->selected_row_num,
0,
&text,
NULL,
NULL,
&mask);
query_string_box(N_("Rename path"),
N_("Enter a new name for the path"),
text,
do_rename_paths_callback, widget);
}
static gint
paths_list_events (GtkWidget *widget,
GdkEvent *event)
{
GdkEventKey *kevent;
GdkEventButton *bevent;
static gint last_row = -1;
switch (event->type)
{
case GDK_BUTTON_PRESS:
bevent = (GdkEventButton *) event;
if(!gtk_clist_get_selection_info (GTK_CLIST(paths_dialog->paths_list),
bevent->x,
bevent->y,
&last_row,NULL))
last_row = -1;
else
{
if(paths_dialog->selected_row_num != last_row)
last_row = -1;
}
if (bevent->button == 3 || bevent->button == 2)
gtk_menu_popup (GTK_MENU (paths_dialog->ops_menu), NULL, NULL, NULL, NULL, bevent->button, bevent->time);
break;
case GDK_2BUTTON_PRESS:
bevent = (GdkEventButton *) event;
if(last_row != -1 &&
gtk_clist_get_selection_info (GTK_CLIST(paths_dialog->paths_list),
bevent->x,
bevent->y,
NULL,NULL))
{
paths_dialog_edit_path_query(widget);
return TRUE;
}
else
return FALSE;
case GDK_KEY_PRESS:
kevent = (GdkEventKey *) event;
switch (kevent->keyval)
{
case GDK_Up:
printf ("up arrow\n");
break;
case GDK_Down:
printf ("down arrow\n");
break;
default:
return FALSE;
}
return TRUE;
default:
break;
}
return FALSE;
}
static PATHIMAGELISTP
bzpath_add_to_current(PATHIMAGELISTP pip,BZPATHP bzp,GimpImage *gimage,gint pos)
{
/* add bzp to current list */
if(!pip)
{
/* This image does not have a list */
pip = pathsList_new(gimage,0,NULL);
/* add to gimage */
gimp_image_set_paths(gimage,pip);
}
if(pos < 0)
pip->bz_paths = g_slist_append(pip->bz_paths,bzp);
else
pip->bz_paths = g_slist_insert(pip->bz_paths,bzp,pos);
return pip;
}
static BZPATHP
paths_dialog_new_path(PATHIMAGELISTP *plp,gpointer points,GimpImage *gimage,gint pos)
{
static gint nseed = 0;
BZPATHP bzp = bzpath_dialog_new(nseed++,points);
*plp = bzpath_add_to_current(*plp,bzp,gimage,pos);
return(bzp);
}
static void
paths_dialog_new_path_callback (GtkWidget * widget, gpointer udata)
{
BZPATHP bzp = paths_dialog_new_path(&paths_dialog->current_path_list,
NULL,
paths_dialog->gimage,
paths_dialog->selected_row_num);
paths_add_path(bzp,paths_dialog->selected_row_num);
}
static void
paths_dialog_delete_path_callback (GtkWidget * widget, gpointer udata)
{
BZPATHP bzp;
PATHIMAGELISTP plp;
gboolean new_sz;
gint row = paths_dialog->selected_row_num;
g_return_if_fail(paths_dialog->current_path_list != NULL);
/* Get current selection... ignore if none */
if(paths_dialog->selected_row_num < 0)
return;
/* Get bzpath structure & delete its content */
plp = paths_dialog->current_path_list;
bzp = (BZPATHP)g_slist_nth_data(plp->bz_paths,row);
/* Remove from list */
plp->bz_paths = g_slist_remove(plp->bz_paths,bzp);
new_sz = (g_slist_length(plp->bz_paths) > 0);
bzpath_free(bzp,NULL);
/* If now empty free everything up */
if(!plp->bz_paths || g_slist_length(plp->bz_paths) == 0)
{
gtk_signal_disconnect(GTK_OBJECT (plp->gimage),
plp->sig_id);
gimp_image_set_paths(plp->gimage,NULL);
pathimagelist_free(plp);
paths_dialog->current_path_list = NULL;
}
/* Do this last since it might cause a new row to become selected */
/* Remove from the clist ... */
gtk_clist_remove(GTK_CLIST(paths_dialog->paths_list),row);
paths_ops_button_set_sensitive(DUP_PATH_BUTTON,new_sz);
paths_ops_button_set_sensitive(DEL_PATH_BUTTON,new_sz);
paths_ops_button_set_sensitive(STROKE_PATH_BUTTON,new_sz);
point_ops_button_set_sensitive(POINT_ADD_BUTTON,new_sz);
point_ops_button_set_sensitive(POINT_DEL_BUTTON,new_sz);
point_ops_button_set_sensitive(POINT_NEW_BUTTON,new_sz);
point_ops_button_set_sensitive(POINT_EDIT_BUTTON,new_sz);
point_ops_button_set_sensitive(PATH_TO_SEL_BUTTON,new_sz);
}
static void
paths_dialog_dup_path_callback (GtkWidget * widget, gpointer udata)
{
BZPATHP bzp;
PATHIMAGELISTP plp;
BezierSelect * bezier_sel;
gint row = paths_dialog->selected_row_num;
gint tmprow;
g_return_if_fail(paths_dialog->current_path_list != NULL);
/* Get current selection... ignore if none */
if(paths_dialog->selected_row_num < 0)
return;
/* Get bzpath structure */
plp = paths_dialog->current_path_list;
bzp = (BZPATHP)g_slist_nth_data(plp->bz_paths,row);
/* Insert at the current position */
bzp = bzpath_copy(bzp);
plp->bz_paths = g_slist_insert(plp->bz_paths,bzp,row);
paths_add_path(bzp,row);
/* Now fudge the drawing....*/
bezier_sel = bzpath_to_beziersel(bzp);
tmprow = paths_dialog->current_path_list->last_selected_row;
paths_dialog->current_path_list->last_selected_row = row;
paths_update_preview(bezier_sel);
beziersel_free(bezier_sel);
paths_dialog->current_path_list->last_selected_row = tmprow;
}
static void
paths_dialog_path_to_sel_callback (GtkWidget * widget, gpointer udata)
{
BZPATHP bzp;
PATHIMAGELISTP plp;
BezierSelect * bezier_sel;
GDisplay * gdisp;
gint row = paths_dialog->selected_row_num;
g_return_if_fail(paths_dialog->current_path_list != NULL);
/* Get current selection... ignore if none */
if(paths_dialog->selected_row_num < 0)
return;
/* Get bzpath structure */
plp = paths_dialog->current_path_list;
bzp = (BZPATHP)g_slist_nth_data(plp->bz_paths,row);
/* Now do the stroke....*/
gdisp = gdisplays_check_valid(paths_dialog->current_path_list->gdisp,
paths_dialog->gimage);
if(!bzp->closed)
{
BZPATHP bzpcopy = bzpath_copy(bzp);
/* Close it */
bzpath_close(bzpcopy);
bezier_sel = bzpath_to_beziersel(bzpcopy);
bzpath_free(bzpcopy,NULL);
bezier_to_selection (bezier_sel, gdisp);
beziersel_free(bezier_sel);
}
else
{
bezier_sel = bzpath_to_beziersel(bzp);
bezier_to_selection (bezier_sel, gdisp);
beziersel_free(bezier_sel);
}
}
static void
paths_dialog_stroke_path_callback (GtkWidget * widget, gpointer udata)
{
BZPATHP bzp;
PATHIMAGELISTP plp;
BezierSelect * bezier_sel;
GDisplay * gdisp;
gint row = paths_dialog->selected_row_num;
g_return_if_fail(paths_dialog->current_path_list != NULL);
/* Get current selection... ignore if none */
if(paths_dialog->selected_row_num < 0)
return;
/* Get bzpath structure */
plp = paths_dialog->current_path_list;
bzp = (BZPATHP)g_slist_nth_data(plp->bz_paths,row);
/* Now do the stroke....*/
gdisp = gdisplays_check_valid(paths_dialog->current_path_list->gdisp,
paths_dialog->gimage);
bezier_sel = bzpath_to_beziersel(bzp);
bezier_stroke (bezier_sel, gdisp, SUBDIVIDE, !bzp->closed);
beziersel_free(bezier_sel);
}
static void
paths_dialog_map_callback (GtkWidget *w,
gpointer client_data)
{
if (!paths_dialog)
return;
gtk_window_add_accel_group (GTK_WINDOW (gtk_widget_get_toplevel(paths_dialog->paths_list)),
paths_dialog->accel_group);
paths_dialog_preview_extents ();
}
static void
paths_dialog_unmap_callback (GtkWidget *w,
gpointer client_data)
{
if (!paths_dialog)
return;
gtk_window_remove_accel_group (GTK_WINDOW (gtk_widget_get_toplevel(paths_dialog->paths_list)),
paths_dialog->accel_group);
}
static void
paths_dialog_destroy_cb (GimpImage *gimage)
{
PATHIMAGELISTP new_path_list;
if(!paths_dialog)
return;
if(paths_dialog->current_path_list &&
gimage == paths_dialog->current_path_list->gimage)
{
/* showing could be last so remove here.. might get
done again if not the last one
*/
paths_dialog->current_path_list = NULL;
paths_dialog->been_selected = FALSE;
gtk_clist_freeze(GTK_CLIST(paths_dialog->paths_list));
gtk_clist_clear(GTK_CLIST(paths_dialog->paths_list));
gtk_clist_thaw(GTK_CLIST(paths_dialog->paths_list));
}
/* Find bz list */
new_path_list = (PATHIMAGELISTP)gimp_image_get_paths(gimage);
if(!new_path_list)
return; /* Already removed - signal handler jsut left in the air */
pathimagelist_free(new_path_list);
gimp_image_set_paths(gimage,NULL);
}
/* Functions used from the bezier code .. tie in with this code */
static void
bzpoints_free(GSList *list)
{
if(!list)
return;
g_slist_foreach(list,bzpoint_free,NULL);
g_slist_free(list);
}
static GSList *
bzpoints_create(BezierSelect *sel)
{
gint i;
GSList *list = NULL;
BZPOINTP bzpoint;
BezierPoint *pts = (BezierPoint *) sel->points;
for (i=0; i< sel->num_points; i++)
{
bzpoint = bzpoint_new(pts->type,pts->x,pts->y);
list = g_slist_append(list,bzpoint);
pts = pts->next;
}
return(list);
}
static GSList *
bzpoints_copy(GSList *list)
{
GSList *slcopy = NULL;
BZPOINTP pdata;
BZPOINTP bzpoint;
while(list)
{
bzpoint = g_new0(BZPOINT,1);
pdata = (BZPOINTP)list->data;
bzpoint->type = pdata->type;
bzpoint->x = pdata->x;
bzpoint->y = pdata->y;
slcopy = g_slist_append(slcopy,bzpoint);
list = g_slist_next(list);
}
return slcopy;
}
static void
paths_update_bzpath(PATHIMAGELISTP plp,BezierSelect *bezier_sel)
{
BZPATHP bzp;
bzp = (BZPATHP)g_slist_nth_data(plp->bz_paths,plp->last_selected_row);
if(bzp->bezier_details)
bzpoints_free(bzp->bezier_details);
bzp->bezier_details = bzpoints_create(bezier_sel);
bzp->closed = bezier_sel->closed;
bzp->state = bezier_sel->state;
}
static gboolean
paths_replaced_current(PATHIMAGELISTP plp,BezierSelect *bezier_sel)
{
/* Is there a currently selected path in this image? */
if(paths_dialog && plp &&
plp->last_selected_row >= 0)
{
paths_update_bzpath(plp,bezier_sel);
return TRUE;
}
return FALSE;
}
static void
paths_draw_segment_points(BezierSelect *bezier_sel,
GdkPoint *pnt,
int npoints)
{
/*
* hopefully the image points are already in image space co-ords.
* so just scale by ratio factor and draw 'em
*/
gint loop;
gint pcount = 0;
GdkPoint * copy_pnt = g_new(GdkPoint,npoints);
GdkPoint * cur_pnt = copy_pnt;
GdkPoint * last_pnt = NULL;
PATHWIDGETP pwidget;
gint row;
/* we could remove duplicate points here */
for(loop = 0; loop < npoints; loop++)
{
/* The "2" is because we have a boarder */
cur_pnt->x = 2+(int) (paths_dialog->ratio * pnt->x);
cur_pnt->y = 2+(int) (paths_dialog->ratio * pnt->y);
pnt++;
if(last_pnt &&
last_pnt->x == cur_pnt->x &&
last_pnt->y == cur_pnt->y)
{
/* same as last ... don't need this one */
continue;
}
/* printf("converting %d [%d,%d] => [%d,%d]\n", */
/* pcount,(int)pnt->x,(int)pnt->y,(int)cur_pnt->x,(int)cur_pnt->y); */
last_pnt = cur_pnt;
pcount++;
cur_pnt++;
}
row = paths_dialog->current_path_list->last_selected_row;
pwidget = (PATHWIDGETP)gtk_clist_get_row_data(GTK_CLIST(paths_dialog->paths_list),row);
if(pcount == 0)
return;
g_return_if_fail(pwidget != NULL);
paths_set_dash_line(paths_dialog->gc,!bezier_sel->closed);
gdk_draw_lines (pwidget->paths_pixmap,
paths_dialog->gc, copy_pnt, pcount);
g_free(copy_pnt);
}
static void
paths_update_preview(BezierSelect *bezier_sel)
{
gint row;
if(paths_dialog &&
paths_dialog->current_path_list &&
(row = paths_dialog->current_path_list->last_selected_row) >= 0 &&
preview_size)
{
PATHWIDGETP pwidget;
pwidget = (PATHWIDGETP)gtk_clist_get_row_data(GTK_CLIST(paths_dialog->paths_list),row);
/* Clear pixmap */
clear_pixmap_preview(pwidget);
/* update .. */
bezier_draw_curve (bezier_sel,paths_draw_segment_points,IMAGE_COORDS);
/* update the pixmap */
gtk_clist_set_pixtext(GTK_CLIST(paths_dialog->paths_list),
row,
0,
pwidget->bzp->name->str,
2,
pwidget->paths_pixmap,
NULL);
}
}
static void
paths_dialog_new_point_callback (GtkWidget * widget, gpointer udata)
{
bezier_select_mode(EXTEND_NEW);
}
static void
paths_dialog_add_point_callback (GtkWidget * widget, gpointer udata)
{
bezier_select_mode(EXTEND_ADD);
}
static void
paths_dialog_delete_point_callback (GtkWidget * widget, gpointer udata)
{
bezier_select_mode(EXTEND_REMOVE);
}
static void
paths_dialog_edit_point_callback (GtkWidget * widget, gpointer udata)
{
bezier_select_mode(EXTEND_EDIT);
}
void
paths_first_button_press(BezierSelect *bezier_sel,GDisplay * gdisp)
{
/* First time a button is pressed in this display */
/* We have two choices here
Either:-
1) We already have a paths item in the list.
=> In this case the new one replaces the current entry. We
need a callback into the bezier code to free things up.
2) We don't have an entry.
=> Create a new one and add this curve.
In either case we need to update the preview widget..
All this of course depends on the fact that gdisp is the same
as before.
*/
BZPATHP bzp;
PATHIMAGELISTP plp;
if(!paths_dialog)
return;
paths_dialog->been_selected = FALSE;
/* Button not pressed in this image...
* find which one it was pressed in if any.
*/
plp = (PATHIMAGELISTP)gimp_image_get_paths(gdisp->gimage);
/* Since beziers are part of the save format.. make the image dirty */
/* gimp_image_dirty(gdisp->gimage); */
if(!paths_replaced_current(plp,bezier_sel))
{
bzp = paths_dialog_new_path(&plp,bzpoints_create(bezier_sel),gdisp->gimage,-1);
bzp->closed = bezier_sel->closed;
bzp->state = bezier_sel->state;
if(paths_dialog->gimage == gdisp->gimage)
{
paths_dialog->current_path_list = plp;
paths_add_path(bzp,-1);
}
}
}
void
paths_newpoint_current(BezierSelect *bezier_sel,GDisplay * gdisp)
{
/* Check if currently showing the paths we are updating */
if(paths_dialog &&
gdisp->gimage == paths_dialog->gimage)
{
/* Enable the buttons!*/
paths_ops_button_set_sensitive(DUP_PATH_BUTTON,TRUE);
paths_ops_button_set_sensitive(DEL_PATH_BUTTON,TRUE);
paths_ops_button_set_sensitive(STROKE_PATH_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_NEW_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_DEL_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_ADD_BUTTON,TRUE);
point_ops_button_set_sensitive(POINT_EDIT_BUTTON,TRUE);
point_ops_button_set_sensitive(PATH_TO_SEL_BUTTON,TRUE);
paths_update_preview(bezier_sel);
}
paths_first_button_press(bezier_sel,gdisp);
}
void
paths_new_bezier_select_tool()
{
if(paths_dialog)
paths_dialog->been_selected = FALSE;
}
/**************************************************************/
/* Code to serialise the bezier curves.
* The curves will be saved out in XCF property format.
* The "save as XCF format" will prompt to save the curves away.
*
* Note the save should really used PDB function to get the
* curves etc. But I have yet to do those 8-)
*/
/**************************************************************/
BZPOINTP
bzpoint_new(gint type,
gint x,
gint y)
{
BZPOINTP bzpoint = g_new0(BZPOINT,1);
bzpoint->type = type;
bzpoint->x = x;
bzpoint->y = y;
return(bzpoint);
}
BZPATHP
bzpath_new(GSList * bezier_details,
gint closed,
gint state,
gint locked,
gchar * name)
{
BZPATHP bzpath = g_new0(BZPATH,1);
bzpath->bezier_details = bezier_details;
bzpath->closed = closed;
bzpath->state = state;
bzpath->locked = locked;
bzpath->name = g_string_new(name);
return bzpath;
}
PathsList *
pathsList_new(GimpImage * gimage,
gint last_selected_row,
GSList * bz_paths)
{
PATHIMAGELISTP pip = g_new0(PATHIMAGELIST,1);
pip->gimage = gimage;
pip->last_selected_row = last_selected_row;
/* add connector to image delete/destroy */
pip->sig_id = gtk_signal_connect(GTK_OBJECT (gimage),
"destroy",
GTK_SIGNAL_FUNC (paths_dialog_destroy_cb),
pip);
pip->bz_paths = bz_paths;
return (PathsList *)pip;
}