2003-03-13 19:52:28 +00:00
|
|
|
/*
|
|
|
|
* gtkdlg.c - GTK implementation of the PuTTY configuration box.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
2003-03-31 11:21:07 +00:00
|
|
|
#include <stdarg.h>
|
2003-03-16 12:37:48 +00:00
|
|
|
#include <ctype.h>
|
2003-04-09 18:46:45 +00:00
|
|
|
#include <time.h>
|
2015-08-31 12:05:51 +00:00
|
|
|
|
2003-03-13 19:52:28 +00:00
|
|
|
#include <gtk/gtk.h>
|
2015-08-08 14:06:15 +00:00
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
2003-03-16 12:37:48 +00:00
|
|
|
#include <gdk/gdkkeysyms.h>
|
2015-08-08 14:06:15 +00:00
|
|
|
#endif
|
2015-08-31 12:05:51 +00:00
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
#define MAY_REFER_TO_GTK_IN_HEADERS
|
|
|
|
|
2015-08-31 12:05:51 +00:00
|
|
|
#include "putty.h"
|
|
|
|
#include "gtkcompat.h"
|
|
|
|
#include "gtkcols.h"
|
|
|
|
#include "gtkfont.h"
|
2015-08-31 14:45:18 +00:00
|
|
|
#include "gtkmisc.h"
|
2015-08-31 12:05:51 +00:00
|
|
|
|
2015-08-08 15:35:19 +00:00
|
|
|
#ifndef NOT_X_WINDOWS
|
2003-03-18 19:06:51 +00:00
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
2015-08-08 15:35:19 +00:00
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
|
|
|
|
#ifdef TESTMODE
|
|
|
|
#define PUTTY_DO_GLOBALS /* actually _define_ globals */
|
|
|
|
#endif
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
#include "storage.h"
|
2003-03-13 19:52:28 +00:00
|
|
|
#include "dialog.h"
|
2003-03-15 15:50:42 +00:00
|
|
|
#include "tree234.h"
|
2015-12-22 12:43:31 +00:00
|
|
|
#include "licence.h"
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2015-08-08 16:34:25 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
/* Decide which of GtkFileChooserDialog and GtkFileSelection to use */
|
|
|
|
#define USE_GTK_FILE_CHOOSER_DIALOG
|
|
|
|
#endif
|
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
struct Shortcut {
|
|
|
|
GtkWidget *widget;
|
|
|
|
struct uctrl *uc;
|
|
|
|
int action;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Shortcuts {
|
|
|
|
struct Shortcut sc[128];
|
|
|
|
};
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
struct uctrl {
|
|
|
|
union control *ctrl;
|
|
|
|
GtkWidget *toplevel;
|
|
|
|
GtkWidget **buttons; int nbuttons; /* for radio buttons */
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkWidget *entry; /* for editbox, filesel, fontsel */
|
2008-04-04 10:16:24 +00:00
|
|
|
GtkWidget *button; /* for filesel, fontsel */
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
GtkWidget *list; /* for listbox (in GTK1), combobox (<=GTK2.3) */
|
2008-04-04 10:56:26 +00:00
|
|
|
GtkWidget *menu; /* for optionmenu (==droplist) */
|
|
|
|
GtkWidget *optmenu; /* also for optionmenu */
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkWidget *combo; /* for combo box (either editable or not) */
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkWidget *treeview; /* for listbox (GTK2), droplist+combo (>=2.4) */
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkListStore *listmodel; /* for all types of list box */
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
GtkWidget *text; /* for text */
|
2006-08-28 14:29:02 +00:00
|
|
|
GtkWidget *label; /* for dlg_label_change */
|
2003-03-18 19:06:51 +00:00
|
|
|
GtkAdjustment *adj; /* for the scrollbar in a list box */
|
2007-01-25 19:47:36 +00:00
|
|
|
guint entrysig;
|
2003-03-22 17:00:06 +00:00
|
|
|
guint textsig;
|
2007-03-27 18:16:36 +00:00
|
|
|
int nclicks;
|
2003-03-15 15:50:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct dlgparam {
|
|
|
|
tree234 *byctrl, *bywidget;
|
|
|
|
void *data;
|
|
|
|
struct { unsigned char r, g, b, ok; } coloursel_result; /* 0-255 */
|
|
|
|
/* `flags' are set to indicate when a GTK signal handler is being called
|
|
|
|
* due to automatic processing and should not flag a user event. */
|
|
|
|
int flags;
|
2003-03-16 12:37:48 +00:00
|
|
|
struct Shortcuts *shortcuts;
|
2007-01-25 19:33:29 +00:00
|
|
|
GtkWidget *window, *cancelbutton;
|
2003-03-18 19:06:51 +00:00
|
|
|
union control *currfocus, *lastfocus;
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkWidget *currtreeitem, **treeitems;
|
2003-03-18 19:06:51 +00:00
|
|
|
int ntreeitems;
|
2015-08-25 18:44:44 +00:00
|
|
|
#else
|
|
|
|
int nselparams;
|
|
|
|
struct selparam *selparams;
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
struct controlbox *ctrlbox;
|
2003-03-18 19:06:51 +00:00
|
|
|
int retval;
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
post_dialog_fn_t after;
|
|
|
|
void *afterctx;
|
2003-03-15 15:50:42 +00:00
|
|
|
};
|
|
|
|
#define FLAG_UPDATING_COMBO_LIST 1
|
2008-04-02 14:48:06 +00:00
|
|
|
#define FLAG_UPDATING_LISTBOX 2
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
enum { /* values for Shortcut.action */
|
|
|
|
SHORTCUT_EMPTY, /* no shortcut on this key */
|
2003-03-22 17:00:06 +00:00
|
|
|
SHORTCUT_TREE, /* focus a tree item */
|
2003-03-16 12:37:48 +00:00
|
|
|
SHORTCUT_FOCUS, /* focus the supplied widget */
|
|
|
|
SHORTCUT_UCTRL, /* do something sane with uctrl */
|
|
|
|
SHORTCUT_UCTRL_UP, /* uctrl is a draglist, move Up */
|
|
|
|
SHORTCUT_UCTRL_DOWN, /* uctrl is a draglist, move Down */
|
|
|
|
};
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
enum {
|
|
|
|
TREESTORE_PATH,
|
|
|
|
TREESTORE_PARAMS,
|
|
|
|
TREESTORE_NUM
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
/*
|
|
|
|
* Forward references.
|
|
|
|
*/
|
2003-03-18 19:06:51 +00:00
|
|
|
static gboolean widget_focus(GtkWidget *widget, GdkEventFocus *event,
|
|
|
|
gpointer data);
|
2003-03-16 12:37:48 +00:00
|
|
|
static void shortcut_add(struct Shortcuts *scs, GtkWidget *labelw,
|
|
|
|
int chr, int action, void *ptr);
|
2006-08-28 14:29:02 +00:00
|
|
|
static void shortcut_highlight(GtkWidget *label, int chr);
|
2008-04-04 10:56:26 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
static gboolean listitem_single_key(GtkWidget *item, GdkEventKey *event,
|
|
|
|
gpointer data);
|
|
|
|
static gboolean listitem_multi_key(GtkWidget *item, GdkEventKey *event,
|
|
|
|
gpointer data);
|
|
|
|
static gboolean listitem_button_press(GtkWidget *item, GdkEventButton *event,
|
|
|
|
gpointer data);
|
|
|
|
static gboolean listitem_button_release(GtkWidget *item, GdkEventButton *event,
|
|
|
|
gpointer data);
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
static void menuitem_activate(GtkMenuItem *item, gpointer data);
|
|
|
|
#endif
|
2015-08-22 13:49:02 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
static void colourchoose_response(GtkDialog *dialog,
|
|
|
|
gint response_id, gpointer data);
|
|
|
|
#else
|
2003-03-15 15:50:42 +00:00
|
|
|
static void coloursel_ok(GtkButton *button, gpointer data);
|
|
|
|
static void coloursel_cancel(GtkButton *button, gpointer data);
|
2015-08-22 13:49:02 +00:00
|
|
|
#endif
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
static void dlgparam_destroy(GtkWidget *widget, gpointer data);
|
2008-04-02 14:48:06 +00:00
|
|
|
int get_listitemheight(GtkWidget *widget);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
|
|
|
static int uctrl_cmp_byctrl(void *av, void *bv)
|
|
|
|
{
|
|
|
|
struct uctrl *a = (struct uctrl *)av;
|
|
|
|
struct uctrl *b = (struct uctrl *)bv;
|
|
|
|
if (a->ctrl < b->ctrl)
|
|
|
|
return -1;
|
|
|
|
else if (a->ctrl > b->ctrl)
|
|
|
|
return +1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int uctrl_cmp_byctrl_find(void *av, void *bv)
|
|
|
|
{
|
|
|
|
union control *a = (union control *)av;
|
|
|
|
struct uctrl *b = (struct uctrl *)bv;
|
|
|
|
if (a < b->ctrl)
|
|
|
|
return -1;
|
|
|
|
else if (a > b->ctrl)
|
|
|
|
return +1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int uctrl_cmp_bywidget(void *av, void *bv)
|
|
|
|
{
|
|
|
|
struct uctrl *a = (struct uctrl *)av;
|
|
|
|
struct uctrl *b = (struct uctrl *)bv;
|
|
|
|
if (a->toplevel < b->toplevel)
|
|
|
|
return -1;
|
|
|
|
else if (a->toplevel > b->toplevel)
|
|
|
|
return +1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int uctrl_cmp_bywidget_find(void *av, void *bv)
|
|
|
|
{
|
|
|
|
GtkWidget *a = (GtkWidget *)av;
|
|
|
|
struct uctrl *b = (struct uctrl *)bv;
|
|
|
|
if (a < b->toplevel)
|
|
|
|
return -1;
|
|
|
|
else if (a > b->toplevel)
|
|
|
|
return +1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dlg_init(struct dlgparam *dp)
|
|
|
|
{
|
|
|
|
dp->byctrl = newtree234(uctrl_cmp_byctrl);
|
|
|
|
dp->bywidget = newtree234(uctrl_cmp_bywidget);
|
|
|
|
dp->coloursel_result.ok = FALSE;
|
2007-01-25 19:33:29 +00:00
|
|
|
dp->window = dp->cancelbutton = NULL;
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2003-03-31 11:21:07 +00:00
|
|
|
dp->treeitems = NULL;
|
2007-01-25 19:33:29 +00:00
|
|
|
dp->currtreeitem = NULL;
|
|
|
|
#endif
|
2004-01-17 14:25:36 +00:00
|
|
|
dp->flags = 0;
|
|
|
|
dp->currfocus = NULL;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dlg_cleanup(struct dlgparam *dp)
|
|
|
|
{
|
|
|
|
struct uctrl *uc;
|
|
|
|
|
|
|
|
freetree234(dp->byctrl); /* doesn't free the uctrls inside */
|
2003-04-09 18:46:45 +00:00
|
|
|
dp->byctrl = NULL;
|
2003-03-15 15:50:42 +00:00
|
|
|
while ( (uc = index234(dp->bywidget, 0)) != NULL) {
|
|
|
|
del234(dp->bywidget, uc);
|
|
|
|
sfree(uc->buttons);
|
|
|
|
sfree(uc);
|
|
|
|
}
|
|
|
|
freetree234(dp->bywidget);
|
2003-04-09 18:46:45 +00:00
|
|
|
dp->bywidget = NULL;
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2003-03-18 19:06:51 +00:00
|
|
|
sfree(dp->treeitems);
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dlg_add_uctrl(struct dlgparam *dp, struct uctrl *uc)
|
|
|
|
{
|
|
|
|
add234(dp->byctrl, uc);
|
|
|
|
add234(dp->bywidget, uc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct uctrl *dlg_find_byctrl(struct dlgparam *dp, union control *ctrl)
|
|
|
|
{
|
2003-04-09 18:46:45 +00:00
|
|
|
if (!dp->byctrl)
|
|
|
|
return NULL;
|
2003-03-15 15:50:42 +00:00
|
|
|
return find234(dp->byctrl, ctrl, uctrl_cmp_byctrl_find);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct uctrl *dlg_find_bywidget(struct dlgparam *dp, GtkWidget *w)
|
|
|
|
{
|
|
|
|
struct uctrl *ret = NULL;
|
2003-04-09 18:46:45 +00:00
|
|
|
if (!dp->bywidget)
|
|
|
|
return NULL;
|
2003-03-15 15:50:42 +00:00
|
|
|
do {
|
|
|
|
ret = find234(dp->bywidget, w, uctrl_cmp_bywidget_find);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2015-08-08 14:53:43 +00:00
|
|
|
w = gtk_widget_get_parent(w);
|
2003-03-15 15:50:42 +00:00
|
|
|
} while (w);
|
|
|
|
return ret;
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2003-03-18 19:06:51 +00:00
|
|
|
union control *dlg_last_focused(union control *ctrl, void *dlg)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
2003-03-18 19:06:51 +00:00
|
|
|
if (dp->currfocus != ctrl)
|
|
|
|
return dp->currfocus;
|
|
|
|
else
|
|
|
|
return dp->lastfocus;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
void dlg_radiobutton_set(union control *ctrl, void *dlg, int which)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_RADIO);
|
|
|
|
assert(uc->buttons != NULL);
|
|
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(uc->buttons[which]), TRUE);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int dlg_radiobutton_get(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_RADIO);
|
|
|
|
assert(uc->buttons != NULL);
|
|
|
|
for (i = 0; i < uc->nbuttons; i++)
|
|
|
|
if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(uc->buttons[i])))
|
|
|
|
return i;
|
|
|
|
return 0; /* got to return something */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_checkbox_set(union control *ctrl, void *dlg, int checked)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_CHECKBOX);
|
|
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(uc->toplevel), checked);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int dlg_checkbox_get(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_CHECKBOX);
|
|
|
|
return gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(uc->toplevel));
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_editbox_set(union control *ctrl, void *dlg, char const *text)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkWidget *entry;
|
|
|
|
char *tmpstring;
|
2003-03-15 15:50:42 +00:00
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX);
|
2008-04-02 14:48:06 +00:00
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,4,0)
|
|
|
|
if (uc->combo)
|
2008-04-02 14:48:06 +00:00
|
|
|
entry = gtk_bin_get_child(GTK_BIN(uc->combo));
|
2008-05-31 19:23:45 +00:00
|
|
|
else
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
entry = uc->entry;
|
|
|
|
|
|
|
|
assert(entry != NULL);
|
2008-04-02 14:48:06 +00:00
|
|
|
|
2007-01-25 19:47:36 +00:00
|
|
|
/*
|
|
|
|
* GTK 2 implements gtk_entry_set_text by means of two separate
|
|
|
|
* operations: first delete the previous text leaving the empty
|
|
|
|
* string, then insert the new text. This causes two calls to
|
|
|
|
* the "changed" signal.
|
2008-04-02 14:48:06 +00:00
|
|
|
*
|
2007-01-25 19:47:36 +00:00
|
|
|
* The first call to "changed", if allowed to proceed normally,
|
|
|
|
* will cause an EVENT_VALCHANGE event on the edit box, causing
|
|
|
|
* a call to dlg_editbox_get() which will read the empty string
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
* out of the GtkEntry - and promptly write it straight into the
|
|
|
|
* Conf structure, which is precisely where our `text' pointer
|
|
|
|
* is probably pointing, so the second editing operation will
|
|
|
|
* insert that instead of the string we originally asked for.
|
2008-04-02 14:48:06 +00:00
|
|
|
*
|
|
|
|
* Hence, we must take our own copy of the text before we do
|
|
|
|
* this.
|
2007-01-25 19:47:36 +00:00
|
|
|
*/
|
2008-04-02 14:48:06 +00:00
|
|
|
tmpstring = dupstr(text);
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), tmpstring);
|
|
|
|
sfree(tmpstring);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
char *dlg_editbox_get(union control *ctrl, void *dlg)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,4,0)
|
|
|
|
if (uc->combo) {
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
return dupstr(gtk_entry_get_text
|
|
|
|
(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(uc->combo)))));
|
2008-04-02 14:48:06 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
|
|
|
|
if (uc->entry) {
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
return dupstr(gtk_entry_get_text(GTK_ENTRY(uc->entry)));
|
2008-05-31 19:23:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(!"We shouldn't get here");
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
static void container_remove_and_destroy(GtkWidget *w, gpointer data)
|
|
|
|
{
|
|
|
|
GtkContainer *cont = GTK_CONTAINER(data);
|
|
|
|
/* gtk_container_remove will unref the widget for us; we need not. */
|
|
|
|
gtk_container_remove(cont, w);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-03-13 19:52:28 +00:00
|
|
|
/* The `listbox' functions can also apply to combo boxes. */
|
|
|
|
void dlg_listbox_clear(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (uc->menu) {
|
|
|
|
gtk_container_foreach(GTK_CONTAINER(uc->menu),
|
|
|
|
container_remove_and_destroy,
|
|
|
|
GTK_CONTAINER(uc->menu));
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
gtk_list_clear_items(GTK_LIST(uc->list), 0, -1);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->listmodel) {
|
|
|
|
gtk_list_store_clear(uc->listmodel);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
assert(!"We shouldn't get here");
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_listbox_del(union control *ctrl, void *dlg, int index)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (uc->menu) {
|
|
|
|
gtk_container_remove
|
|
|
|
(GTK_CONTAINER(uc->menu),
|
|
|
|
g_list_nth_data(GTK_MENU_SHELL(uc->menu)->children, index));
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
gtk_list_clear_items(GTK_LIST(uc->list), index, index+1);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->listmodel) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
assert(uc->listmodel != NULL);
|
|
|
|
path = gtk_tree_path_new_from_indices(index, -1);
|
|
|
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(uc->listmodel), &iter, path);
|
|
|
|
gtk_list_store_remove(uc->listmodel, &iter);
|
|
|
|
gtk_tree_path_free(path);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_listbox_add(union control *ctrl, void *dlg, char const *text)
|
|
|
|
{
|
2003-03-25 23:45:56 +00:00
|
|
|
dlg_listbox_addwithid(ctrl, dlg, text, 0);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Each listbox entry may have a numeric id associated with it.
|
|
|
|
* Note that some front ends only permit a string to be stored at
|
|
|
|
* each position, which means that _if_ you put two identical
|
|
|
|
* strings in any listbox then you MUST not assign them different
|
|
|
|
* IDs and expect to get meaningful results back.
|
|
|
|
*/
|
2003-03-25 23:45:56 +00:00
|
|
|
void dlg_listbox_addwithid(union control *ctrl, void *dlg,
|
|
|
|
char const *text, int id)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
/*
|
2008-04-04 10:56:26 +00:00
|
|
|
* This routine is long and complicated in both GTK 1 and 2,
|
|
|
|
* and completely different. Sigh.
|
2008-04-02 14:48:06 +00:00
|
|
|
*/
|
2008-04-04 10:56:26 +00:00
|
|
|
dp->flags |= FLAG_UPDATING_COMBO_LIST;
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (uc->menu) {
|
|
|
|
/*
|
|
|
|
* List item in a drop-down (but non-combo) list. Tabs are
|
|
|
|
* ignored; we just provide a standard menu item with the
|
|
|
|
* text.
|
|
|
|
*/
|
|
|
|
GtkWidget *menuitem = gtk_menu_item_new_with_label(text);
|
|
|
|
|
|
|
|
gtk_container_add(GTK_CONTAINER(uc->menu), menuitem);
|
|
|
|
gtk_widget_show(menuitem);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(menuitem), "user-data",
|
|
|
|
GINT_TO_POINTER(id));
|
|
|
|
g_signal_connect(G_OBJECT(menuitem), "activate",
|
|
|
|
G_CALLBACK(menuitem_activate), dp);
|
2008-05-31 19:23:45 +00:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (uc->list && uc->entry) {
|
|
|
|
/*
|
|
|
|
* List item in a combo-box list, which means the sensible
|
|
|
|
* thing to do is make it a perfectly normal label. Hence
|
|
|
|
* tabs are disregarded.
|
|
|
|
*/
|
|
|
|
GtkWidget *listitem = gtk_list_item_new_with_label(text);
|
|
|
|
|
|
|
|
gtk_container_add(GTK_CONTAINER(uc->list), listitem);
|
|
|
|
gtk_widget_show(listitem);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(listitem), "user-data",
|
|
|
|
GINT_TO_POINTER(id));
|
2008-05-31 19:23:45 +00:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
/*
|
|
|
|
* List item in a non-combo-box list box. We make all of
|
|
|
|
* these Columns containing GtkLabels. This allows us to do
|
|
|
|
* the nasty force_left hack irrespective of whether there
|
|
|
|
* are tabs in the thing.
|
|
|
|
*/
|
|
|
|
GtkWidget *listitem = gtk_list_item_new();
|
|
|
|
GtkWidget *cols = columns_new(10);
|
|
|
|
gint *percents;
|
|
|
|
int i, ncols;
|
|
|
|
|
|
|
|
/* Count the tabs in the text, and hence determine # of columns. */
|
|
|
|
ncols = 1;
|
|
|
|
for (i = 0; text[i]; i++)
|
|
|
|
if (text[i] == '\t')
|
|
|
|
ncols++;
|
|
|
|
|
|
|
|
assert(ncols <=
|
|
|
|
(uc->ctrl->listbox.ncols ? uc->ctrl->listbox.ncols : 1));
|
|
|
|
percents = snewn(ncols, gint);
|
|
|
|
percents[ncols-1] = 100;
|
|
|
|
for (i = 0; i < ncols-1; i++) {
|
|
|
|
percents[i] = uc->ctrl->listbox.percentages[i];
|
|
|
|
percents[ncols-1] -= percents[i];
|
|
|
|
}
|
|
|
|
columns_set_cols(COLUMNS(cols), ncols, percents);
|
|
|
|
sfree(percents);
|
|
|
|
|
|
|
|
for (i = 0; i < ncols; i++) {
|
|
|
|
int len = strcspn(text, "\t");
|
|
|
|
char *dup = dupprintf("%.*s", len, text);
|
|
|
|
GtkWidget *label;
|
|
|
|
|
|
|
|
text += len;
|
|
|
|
if (*text) text++;
|
|
|
|
label = gtk_label_new(dup);
|
|
|
|
sfree(dup);
|
|
|
|
|
|
|
|
columns_add(COLUMNS(cols), label, i, 1);
|
|
|
|
columns_force_left_align(COLUMNS(cols), label);
|
|
|
|
gtk_widget_show(label);
|
|
|
|
}
|
|
|
|
gtk_container_add(GTK_CONTAINER(listitem), cols);
|
|
|
|
gtk_widget_show(cols);
|
|
|
|
gtk_container_add(GTK_CONTAINER(uc->list), listitem);
|
|
|
|
gtk_widget_show(listitem);
|
|
|
|
|
|
|
|
if (ctrl->listbox.multisel) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(listitem), "key_press_event",
|
|
|
|
G_CALLBACK(listitem_multi_key), uc->adj);
|
2008-04-04 10:56:26 +00:00
|
|
|
} else {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(listitem), "key_press_event",
|
|
|
|
G_CALLBACK(listitem_single_key), uc->adj);
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(listitem), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
|
|
|
g_signal_connect(G_OBJECT(listitem), "button_press_event",
|
|
|
|
G_CALLBACK(listitem_button_press), dp);
|
|
|
|
g_signal_connect(G_OBJECT(listitem), "button_release_event",
|
|
|
|
G_CALLBACK(listitem_button_release), dp);
|
|
|
|
g_object_set_data(G_OBJECT(listitem), "user-data",
|
|
|
|
GINT_TO_POINTER(id));
|
2008-05-31 19:23:45 +00:00
|
|
|
goto done;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#else
|
2008-05-31 19:23:45 +00:00
|
|
|
if (uc->listmodel) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GtkTreeIter iter;
|
|
|
|
int i, cols;
|
|
|
|
|
|
|
|
dp->flags |= FLAG_UPDATING_LISTBOX;/* inhibit drag-list update */
|
|
|
|
gtk_list_store_append(uc->listmodel, &iter);
|
|
|
|
dp->flags &= ~FLAG_UPDATING_LISTBOX;
|
|
|
|
gtk_list_store_set(uc->listmodel, &iter, 0, id, -1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now go through text and divide it into columns at the tabs,
|
|
|
|
* as necessary.
|
|
|
|
*/
|
|
|
|
cols = (uc->ctrl->generic.type == CTRL_LISTBOX ? ctrl->listbox.ncols : 1);
|
|
|
|
cols = cols ? cols : 1;
|
|
|
|
for (i = 0; i < cols; i++) {
|
|
|
|
int collen = strcspn(text, "\t");
|
|
|
|
char *tmpstr = snewn(collen+1, char);
|
|
|
|
memcpy(tmpstr, text, collen);
|
|
|
|
tmpstr[collen] = '\0';
|
|
|
|
gtk_list_store_set(uc->listmodel, &iter, i+1, tmpstr, -1);
|
|
|
|
sfree(tmpstr);
|
|
|
|
text += collen;
|
|
|
|
if (*text) text++;
|
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
goto done;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
|
|
|
done:
|
|
|
|
dp->flags &= ~FLAG_UPDATING_COMBO_LIST;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int dlg_listbox_getid(union control *ctrl, void *dlg, int index)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
if (uc->menu || uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GList *children;
|
2015-08-08 16:29:02 +00:00
|
|
|
GObject *item;
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
children = gtk_container_children(GTK_CONTAINER(uc->menu ? uc->menu :
|
|
|
|
uc->list));
|
2015-08-08 16:29:02 +00:00
|
|
|
item = G_OBJECT(g_list_nth_data(children, index));
|
2008-04-04 10:56:26 +00:00
|
|
|
g_list_free(children);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
return GPOINTER_TO_INT(g_object_get_data(G_OBJECT(item), "user-data"));
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->listmodel) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
path = gtk_tree_path_new_from_indices(index, -1);
|
|
|
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(uc->listmodel), &iter, path);
|
|
|
|
gtk_tree_model_get(GTK_TREE_MODEL(uc->listmodel), &iter, 0, &ret, -1);
|
|
|
|
gtk_tree_path_free(path);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
|
|
|
return -1; /* placate dataflow analysis */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* dlg_listbox_index returns <0 if no single element is selected. */
|
|
|
|
int dlg_listbox_index(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
if (uc->menu || uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GList *children;
|
|
|
|
GtkWidget *item, *activeitem;
|
|
|
|
int i;
|
|
|
|
int selected = -1;
|
|
|
|
|
|
|
|
if (uc->menu)
|
|
|
|
activeitem = gtk_menu_get_active(GTK_MENU(uc->menu));
|
|
|
|
else
|
|
|
|
activeitem = NULL; /* unnecessarily placate gcc */
|
|
|
|
|
|
|
|
children = gtk_container_children(GTK_CONTAINER(uc->menu ? uc->menu :
|
|
|
|
uc->list));
|
|
|
|
for (i = 0; children!=NULL && (item = GTK_WIDGET(children->data))!=NULL;
|
|
|
|
i++, children = children->next) {
|
|
|
|
if (uc->menu ? activeitem == item :
|
|
|
|
GTK_WIDGET_STATE(item) == GTK_STATE_SELECTED) {
|
|
|
|
if (selected == -1)
|
|
|
|
selected = i;
|
|
|
|
else
|
|
|
|
selected = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_list_free(children);
|
|
|
|
return selected < 0 ? -1 : selected;
|
|
|
|
}
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
if (uc->combo) {
|
|
|
|
/*
|
|
|
|
* This API function already does the right thing in the
|
|
|
|
* case of no current selection.
|
|
|
|
*/
|
|
|
|
return gtk_combo_box_get_active(GTK_COMBO_BOX(uc->combo));
|
2008-05-31 19:23:45 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->treeview) {
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkTreeSelection *treesel;
|
|
|
|
GtkTreePath *path;
|
2008-06-11 18:03:35 +00:00
|
|
|
GtkTreeModel *model;
|
2008-04-02 14:48:06 +00:00
|
|
|
GList *sellist;
|
|
|
|
gint *indices;
|
|
|
|
int ret;
|
|
|
|
|
2008-04-04 10:16:24 +00:00
|
|
|
assert(uc->treeview != NULL);
|
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(uc->treeview));
|
2008-04-02 14:48:06 +00:00
|
|
|
|
|
|
|
if (gtk_tree_selection_count_selected_rows(treesel) != 1)
|
|
|
|
return -1;
|
|
|
|
|
2008-06-11 18:03:35 +00:00
|
|
|
sellist = gtk_tree_selection_get_selected_rows(treesel, &model);
|
2008-04-02 14:48:06 +00:00
|
|
|
|
|
|
|
assert(sellist && sellist->data);
|
|
|
|
path = sellist->data;
|
|
|
|
|
|
|
|
if (gtk_tree_path_get_depth(path) != 1) {
|
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
indices = gtk_tree_path_get_indices(path);
|
|
|
|
if (!indices) {
|
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
ret = indices[0];
|
|
|
|
}
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
2008-04-02 14:48:06 +00:00
|
|
|
|
|
|
|
g_list_foreach(sellist, (GFunc)gtk_tree_path_free, NULL);
|
|
|
|
g_list_free(sellist);
|
|
|
|
|
|
|
|
return ret;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
|
|
|
return -1; /* placate dataflow analysis */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int dlg_listbox_issel(union control *ctrl, void *dlg, int index)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
if (uc->menu || uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
GList *children;
|
|
|
|
GtkWidget *item, *activeitem;
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
assert(uc->menu != NULL || uc->list != NULL);
|
|
|
|
|
|
|
|
children = gtk_container_children(GTK_CONTAINER(uc->menu ? uc->menu :
|
|
|
|
uc->list));
|
|
|
|
item = GTK_WIDGET(g_list_nth_data(children, index));
|
|
|
|
g_list_free(children);
|
|
|
|
|
|
|
|
if (uc->menu) {
|
|
|
|
activeitem = gtk_menu_get_active(GTK_MENU(uc->menu));
|
|
|
|
return item == activeitem;
|
|
|
|
} else {
|
|
|
|
return GTK_WIDGET_STATE(item) == GTK_STATE_SELECTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
if (uc->combo) {
|
|
|
|
/*
|
|
|
|
* This API function already does the right thing in the
|
|
|
|
* case of no current selection.
|
|
|
|
*/
|
|
|
|
return gtk_combo_box_get_active(GTK_COMBO_BOX(uc->combo)) == index;
|
2008-05-31 19:23:45 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->treeview) {
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkTreeSelection *treesel;
|
|
|
|
GtkTreePath *path;
|
|
|
|
int ret;
|
|
|
|
|
2008-04-04 10:16:24 +00:00
|
|
|
assert(uc->treeview != NULL);
|
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(uc->treeview));
|
2008-04-02 14:48:06 +00:00
|
|
|
|
|
|
|
path = gtk_tree_path_new_from_indices(index, -1);
|
|
|
|
ret = gtk_tree_selection_path_is_selected(treesel, path);
|
|
|
|
gtk_tree_path_free(path);
|
|
|
|
|
|
|
|
return ret;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
|
|
|
return -1; /* placate dataflow analysis */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_listbox_select(union control *ctrl, void *dlg, int index)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_EDITBOX ||
|
|
|
|
uc->ctrl->generic.type == CTRL_LISTBOX);
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (uc->optmenu) {
|
|
|
|
gtk_option_menu_set_history(GTK_OPTION_MENU(uc->optmenu), index);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (uc->list) {
|
2008-04-04 10:56:26 +00:00
|
|
|
int nitems;
|
|
|
|
GList *items;
|
|
|
|
gdouble newtop, newbot;
|
|
|
|
|
|
|
|
gtk_list_select_item(GTK_LIST(uc->list), index);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Scroll the list box if necessary to ensure the newly
|
|
|
|
* selected item is visible.
|
|
|
|
*/
|
|
|
|
items = gtk_container_children(GTK_CONTAINER(uc->list));
|
|
|
|
nitems = g_list_length(items);
|
|
|
|
if (nitems > 0) {
|
|
|
|
int modified = FALSE;
|
|
|
|
g_list_free(items);
|
|
|
|
newtop = uc->adj->lower +
|
|
|
|
(uc->adj->upper - uc->adj->lower) * index / nitems;
|
|
|
|
newbot = uc->adj->lower +
|
|
|
|
(uc->adj->upper - uc->adj->lower) * (index+1) / nitems;
|
|
|
|
if (uc->adj->value > newtop) {
|
|
|
|
modified = TRUE;
|
|
|
|
uc->adj->value = newtop;
|
|
|
|
} else if (uc->adj->value < newbot - uc->adj->page_size) {
|
|
|
|
modified = TRUE;
|
|
|
|
uc->adj->value = newbot - uc->adj->page_size;
|
|
|
|
}
|
|
|
|
if (modified)
|
|
|
|
gtk_adjustment_value_changed(uc->adj);
|
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
if (uc->combo) {
|
|
|
|
gtk_combo_box_set_active(GTK_COMBO_BOX(uc->combo), index);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->treeview) {
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkTreeSelection *treesel;
|
|
|
|
GtkTreePath *path;
|
2005-09-21 17:09:07 +00:00
|
|
|
|
2008-04-04 10:16:24 +00:00
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(uc->treeview));
|
2005-09-21 17:09:07 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
path = gtk_tree_path_new_from_indices(index, -1);
|
|
|
|
gtk_tree_selection_select_path(treesel, path);
|
2008-04-04 10:16:24 +00:00
|
|
|
gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(uc->treeview),
|
2008-04-02 14:48:06 +00:00
|
|
|
path, NULL, FALSE, 0.0, 0.0);
|
|
|
|
gtk_tree_path_free(path);
|
2008-05-31 19:23:45 +00:00
|
|
|
return;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_text_set(union control *ctrl, void *dlg, char const *text)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_TEXT);
|
|
|
|
assert(uc->text != NULL);
|
|
|
|
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->text), text);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2006-08-28 10:35:12 +00:00
|
|
|
void dlg_label_change(union control *ctrl, void *dlg, char const *text)
|
|
|
|
{
|
2006-08-28 14:29:02 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
switch (uc->ctrl->generic.type) {
|
|
|
|
case CTRL_BUTTON:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->toplevel), text);
|
|
|
|
shortcut_highlight(uc->toplevel, ctrl->button.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_CHECKBOX:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->toplevel), text);
|
|
|
|
shortcut_highlight(uc->toplevel, ctrl->checkbox.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_RADIO:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->label), text);
|
|
|
|
shortcut_highlight(uc->label, ctrl->radio.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_EDITBOX:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->label), text);
|
|
|
|
shortcut_highlight(uc->label, ctrl->editbox.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_FILESELECT:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->label), text);
|
|
|
|
shortcut_highlight(uc->label, ctrl->fileselect.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_FONTSELECT:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->label), text);
|
|
|
|
shortcut_highlight(uc->label, ctrl->fontselect.shortcut);
|
|
|
|
break;
|
|
|
|
case CTRL_LISTBOX:
|
|
|
|
gtk_label_set_text(GTK_LABEL(uc->label), text);
|
|
|
|
shortcut_highlight(uc->label, ctrl->listbox.shortcut);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(!"This shouldn't happen");
|
|
|
|
break;
|
|
|
|
}
|
2006-08-28 10:35:12 +00:00
|
|
|
}
|
|
|
|
|
2011-10-02 11:01:57 +00:00
|
|
|
void dlg_filesel_set(union control *ctrl, void *dlg, Filename *fn)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
2012-04-13 18:02:30 +00:00
|
|
|
/* We must copy fn->path before passing it to gtk_entry_set_text.
|
|
|
|
* See comment in dlg_editbox_set() for the reasons. */
|
|
|
|
char *duppath = dupstr(fn->path);
|
2003-03-15 15:50:42 +00:00
|
|
|
assert(uc->ctrl->generic.type == CTRL_FILESELECT);
|
|
|
|
assert(uc->entry != NULL);
|
2012-04-13 18:02:30 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), duppath);
|
|
|
|
sfree(duppath);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2011-10-02 11:01:57 +00:00
|
|
|
Filename *dlg_filesel_get(union control *ctrl, void *dlg)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_FILESELECT);
|
|
|
|
assert(uc->entry != NULL);
|
2011-10-02 11:01:57 +00:00
|
|
|
return filename_from_str(gtk_entry_get_text(GTK_ENTRY(uc->entry)));
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2011-10-01 17:38:59 +00:00
|
|
|
void dlg_fontsel_set(union control *ctrl, void *dlg, FontSpec *fs)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
2012-04-13 18:02:30 +00:00
|
|
|
/* We must copy fs->name before passing it to gtk_entry_set_text.
|
|
|
|
* See comment in dlg_editbox_set() for the reasons. */
|
|
|
|
char *dupname = dupstr(fs->name);
|
2003-03-15 15:50:42 +00:00
|
|
|
assert(uc->ctrl->generic.type == CTRL_FONTSELECT);
|
|
|
|
assert(uc->entry != NULL);
|
2012-04-13 18:02:30 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), dupname);
|
|
|
|
sfree(dupname);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2011-10-01 17:38:59 +00:00
|
|
|
FontSpec *dlg_fontsel_get(union control *ctrl, void *dlg)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
assert(uc->ctrl->generic.type == CTRL_FONTSELECT);
|
|
|
|
assert(uc->entry != NULL);
|
2011-10-01 17:38:59 +00:00
|
|
|
return fontspec_new(gtk_entry_get_text(GTK_ENTRY(uc->entry)));
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Bracketing a large set of updates in these two functions will
|
|
|
|
* cause the front end (if possible) to delay updating the screen
|
|
|
|
* until it's all complete, thus avoiding flicker.
|
|
|
|
*/
|
|
|
|
void dlg_update_start(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
/*
|
|
|
|
* Apparently we can't do this at all in GTK. GtkCList supports
|
|
|
|
* freeze and thaw, but not GtkList. Bah.
|
|
|
|
*/
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_update_done(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
/*
|
|
|
|
* Apparently we can't do this at all in GTK. GtkCList supports
|
|
|
|
* freeze and thaw, but not GtkList. Bah.
|
|
|
|
*/
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_set_focus(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
2003-03-18 19:06:51 +00:00
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
|
|
|
switch (ctrl->generic.type) {
|
|
|
|
case CTRL_CHECKBOX:
|
|
|
|
case CTRL_BUTTON:
|
|
|
|
/* Check boxes and buttons get the focus _and_ get toggled. */
|
|
|
|
gtk_widget_grab_focus(uc->toplevel);
|
|
|
|
break;
|
|
|
|
case CTRL_FILESELECT:
|
|
|
|
case CTRL_FONTSELECT:
|
|
|
|
case CTRL_EDITBOX:
|
2008-04-02 14:48:06 +00:00
|
|
|
if (uc->entry) {
|
|
|
|
/* Anything containing an edit box gets that focused. */
|
|
|
|
gtk_widget_grab_focus(uc->entry);
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
else if (uc->combo) {
|
2008-04-02 14:48:06 +00:00
|
|
|
/* Failing that, there'll be a combo box. */
|
|
|
|
gtk_widget_grab_focus(uc->combo);
|
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2003-03-18 19:06:51 +00:00
|
|
|
break;
|
|
|
|
case CTRL_RADIO:
|
|
|
|
/*
|
|
|
|
* Radio buttons: we find the currently selected button and
|
|
|
|
* focus it.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < ctrl->radio.nbuttons; i++)
|
|
|
|
if (gtk_toggle_button_get_active
|
|
|
|
(GTK_TOGGLE_BUTTON(uc->buttons[i]))) {
|
|
|
|
gtk_widget_grab_focus(uc->buttons[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CTRL_LISTBOX:
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (uc->optmenu) {
|
|
|
|
gtk_widget_grab_focus(uc->optmenu);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (uc->combo) {
|
|
|
|
gtk_widget_grab_focus(uc->combo);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (uc->list) {
|
|
|
|
/*
|
|
|
|
* For GTK-1 style list boxes, we tell it to focus one
|
|
|
|
* of its children, which appears to do the Right
|
|
|
|
* Thing.
|
|
|
|
*/
|
2008-04-04 10:56:26 +00:00
|
|
|
gtk_container_focus(GTK_CONTAINER(uc->list), GTK_DIR_TAB_FORWARD);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#else
|
2008-04-04 10:16:24 +00:00
|
|
|
if (uc->treeview) {
|
|
|
|
gtk_widget_grab_focus(uc->treeview);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
2008-04-02 14:48:06 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
2003-03-18 19:06:51 +00:00
|
|
|
break;
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* During event processing, you might well want to give an error
|
|
|
|
* indication to the user. dlg_beep() is a quick and easy generic
|
|
|
|
* error; dlg_error() puts up a message-box or equivalent.
|
|
|
|
*/
|
|
|
|
void dlg_beep(void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
gdk_beep();
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2003-04-10 08:53:43 +00:00
|
|
|
static void set_transient_window_pos(GtkWidget *parent, GtkWidget *child)
|
|
|
|
{
|
2015-08-08 17:02:01 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2003-04-10 08:53:43 +00:00
|
|
|
gint x, y, w, h, dx, dy;
|
|
|
|
GtkRequisition req;
|
|
|
|
gtk_window_set_position(GTK_WINDOW(child), GTK_WIN_POS_NONE);
|
|
|
|
gtk_widget_size_request(GTK_WIDGET(child), &req);
|
|
|
|
|
2015-08-08 14:53:43 +00:00
|
|
|
gdk_window_get_origin(gtk_widget_get_window(GTK_WIDGET(parent)), &x, &y);
|
|
|
|
gdk_window_get_size(gtk_widget_get_window(GTK_WIDGET(parent)), &w, &h);
|
2003-04-10 08:53:43 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* One corner of the transient will be offset inwards, by 1/4
|
|
|
|
* of the parent window's size, from the corresponding corner
|
|
|
|
* of the parent window. The corner will be chosen so as to
|
|
|
|
* place the transient closer to the centre of the screen; this
|
|
|
|
* should avoid transients going off the edge of the screen on
|
|
|
|
* a regular basis.
|
|
|
|
*/
|
|
|
|
if (x + w/2 < gdk_screen_width() / 2)
|
|
|
|
dx = x + w/4; /* work from left edges */
|
|
|
|
else
|
|
|
|
dx = x + 3*w/4 - req.width; /* work from right edges */
|
|
|
|
if (y + h/2 < gdk_screen_height() / 2)
|
|
|
|
dy = y + h/4; /* work from top edges */
|
|
|
|
else
|
|
|
|
dy = y + 3*h/4 - req.height; /* work from bottom edges */
|
|
|
|
gtk_widget_set_uposition(GTK_WIDGET(child), dx, dy);
|
2015-08-08 17:02:01 +00:00
|
|
|
#endif
|
2003-04-10 08:53:43 +00:00
|
|
|
}
|
|
|
|
|
2017-11-26 17:07:47 +00:00
|
|
|
void trivial_post_dialog_fn(void *vctx, int result)
|
2017-11-26 17:05:51 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-05-15 10:15:42 +00:00
|
|
|
void dlg_error_msg(void *dlg, const char *msg)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
2017-11-26 17:05:51 +00:00
|
|
|
create_message_box(
|
|
|
|
dp->window, "Error", msg,
|
|
|
|
string_width("Some sort of text about a config-box error message"),
|
|
|
|
FALSE, &buttons_ok, trivial_post_dialog_fn, NULL);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function signals to the front end that the dialog's
|
|
|
|
* processing is completed, and passes an integer value (typically
|
|
|
|
* a success status).
|
|
|
|
*/
|
|
|
|
void dlg_end(void *dlg, int value)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
2003-03-18 19:06:51 +00:00
|
|
|
dp->retval = value;
|
2003-03-29 19:52:50 +00:00
|
|
|
gtk_widget_destroy(dp->window);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_refresh(union control *ctrl, void *dlg)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc;
|
|
|
|
|
|
|
|
if (ctrl) {
|
|
|
|
if (ctrl->generic.handler != NULL)
|
|
|
|
ctrl->generic.handler(ctrl, dp, dp->data, EVENT_REFRESH);
|
|
|
|
} else {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; (uc = index234(dp->byctrl, i)) != NULL; i++) {
|
|
|
|
assert(uc->ctrl != NULL);
|
|
|
|
if (uc->ctrl->generic.handler != NULL)
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp,
|
|
|
|
dp->data, EVENT_REFRESH);
|
|
|
|
}
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void dlg_coloursel_start(union control *ctrl, void *dlg, int r, int g, int b)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl);
|
|
|
|
|
2015-08-22 13:49:02 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
GtkWidget *coloursel =
|
|
|
|
gtk_color_chooser_dialog_new("Select a colour",
|
|
|
|
GTK_WINDOW(dp->window));
|
|
|
|
gtk_color_chooser_set_use_alpha(GTK_COLOR_CHOOSER(coloursel), FALSE);
|
|
|
|
#else
|
|
|
|
GtkWidget *okbutton, *cancelbutton;
|
2003-03-15 15:50:42 +00:00
|
|
|
GtkWidget *coloursel =
|
|
|
|
gtk_color_selection_dialog_new("Select a colour");
|
|
|
|
GtkColorSelectionDialog *ccs = GTK_COLOR_SELECTION_DIALOG(coloursel);
|
2015-08-08 14:53:43 +00:00
|
|
|
GtkColorSelection *cs = GTK_COLOR_SELECTION
|
|
|
|
(gtk_color_selection_dialog_get_color_selection(ccs));
|
2015-08-22 13:49:02 +00:00
|
|
|
gtk_color_selection_set_has_opacity_control(cs, FALSE);
|
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
|
|
|
|
dp->coloursel_result.ok = FALSE;
|
|
|
|
|
|
|
|
gtk_window_set_modal(GTK_WINDOW(coloursel), TRUE);
|
2015-08-08 16:32:15 +00:00
|
|
|
|
2015-08-22 13:49:02 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
{
|
|
|
|
GdkRGBA rgba;
|
|
|
|
rgba.red = r / 255.0;
|
|
|
|
rgba.green = g / 255.0;
|
|
|
|
rgba.blue = b / 255.0;
|
|
|
|
rgba.alpha = 1.0; /* fully opaque! */
|
|
|
|
gtk_color_chooser_set_rgba(GTK_COLOR_CHOOSER(coloursel), &rgba);
|
|
|
|
}
|
|
|
|
#elif GTK_CHECK_VERSION(2,0,0)
|
2015-08-08 16:32:15 +00:00
|
|
|
{
|
|
|
|
GdkColor col;
|
|
|
|
col.red = r * 0x0101;
|
|
|
|
col.green = g * 0x0101;
|
|
|
|
col.blue = b * 0x0101;
|
|
|
|
gtk_color_selection_set_current_color(cs, &col);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
{
|
|
|
|
gdouble cvals[4];
|
|
|
|
cvals[0] = r / 255.0;
|
|
|
|
cvals[1] = g / 255.0;
|
|
|
|
cvals[2] = b / 255.0;
|
|
|
|
cvals[3] = 1.0; /* fully opaque! */
|
|
|
|
gtk_color_selection_set_color(cs, cvals);
|
|
|
|
}
|
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2015-08-22 13:49:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(coloursel), "user-data", (gpointer)uc);
|
|
|
|
|
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
g_signal_connect(G_OBJECT(coloursel), "response",
|
|
|
|
G_CALLBACK(colourchoose_response), (gpointer)dp);
|
|
|
|
#else
|
|
|
|
|
2015-08-08 14:53:43 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
g_object_get(G_OBJECT(ccs),
|
|
|
|
"ok-button", &okbutton,
|
|
|
|
"cancel-button", &cancelbutton,
|
|
|
|
(const char *)NULL);
|
|
|
|
#else
|
|
|
|
okbutton = ccs->ok_button;
|
|
|
|
cancelbutton = ccs->cancel_button;
|
|
|
|
#endif
|
|
|
|
g_object_set_data(G_OBJECT(okbutton), "user-data",
|
2003-03-15 15:50:42 +00:00
|
|
|
(gpointer)coloursel);
|
2015-08-08 14:53:43 +00:00
|
|
|
g_object_set_data(G_OBJECT(cancelbutton), "user-data",
|
2003-03-15 15:50:42 +00:00
|
|
|
(gpointer)coloursel);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(okbutton), "clicked",
|
|
|
|
G_CALLBACK(coloursel_ok), (gpointer)dp);
|
|
|
|
g_signal_connect(G_OBJECT(cancelbutton), "clicked",
|
|
|
|
G_CALLBACK(coloursel_cancel), (gpointer)dp);
|
|
|
|
g_signal_connect_swapped(G_OBJECT(okbutton), "clicked",
|
|
|
|
G_CALLBACK(gtk_widget_destroy),
|
|
|
|
(gpointer)coloursel);
|
|
|
|
g_signal_connect_swapped(G_OBJECT(cancelbutton), "clicked",
|
|
|
|
G_CALLBACK(gtk_widget_destroy),
|
|
|
|
(gpointer)coloursel);
|
2015-08-22 13:49:02 +00:00
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
gtk_widget_show(coloursel);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int dlg_coloursel_results(union control *ctrl, void *dlg,
|
|
|
|
int *r, int *g, int *b)
|
|
|
|
{
|
2003-03-15 15:50:42 +00:00
|
|
|
struct dlgparam *dp = (struct dlgparam *)dlg;
|
|
|
|
if (dp->coloursel_result.ok) {
|
|
|
|
*r = dp->coloursel_result.r;
|
|
|
|
*g = dp->coloursel_result.g;
|
|
|
|
*b = dp->coloursel_result.b;
|
|
|
|
return 1;
|
|
|
|
} else
|
|
|
|
return 0;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
/* ----------------------------------------------------------------------
|
|
|
|
* Signal handlers while the dialog box is active.
|
|
|
|
*/
|
|
|
|
|
2003-03-18 19:06:51 +00:00
|
|
|
static gboolean widget_focus(GtkWidget *widget, GdkEventFocus *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, widget);
|
|
|
|
union control *focus;
|
|
|
|
|
|
|
|
if (uc && uc->ctrl)
|
|
|
|
focus = uc->ctrl;
|
|
|
|
else
|
|
|
|
focus = NULL;
|
|
|
|
|
|
|
|
if (focus != dp->currfocus) {
|
|
|
|
dp->lastfocus = dp->currfocus;
|
|
|
|
dp->currfocus = focus;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
static void button_clicked(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(button));
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_ACTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void button_toggled(GtkToggleButton *tb, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(tb));
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_VALCHANGE);
|
|
|
|
}
|
|
|
|
|
2007-01-25 19:47:36 +00:00
|
|
|
static gboolean editbox_key(GtkWidget *widget, GdkEventKey *event,
|
|
|
|
gpointer data)
|
2003-03-16 12:37:48 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* GtkEntry has a nasty habit of eating the Return key, which
|
|
|
|
* is unhelpful since it doesn't actually _do_ anything with it
|
|
|
|
* (it calls gtk_widget_activate, but our edit boxes never need
|
|
|
|
* activating). So I catch Return before GtkEntry sees it, and
|
|
|
|
* pass it straight on to the parent widget. Effect: hitting
|
|
|
|
* Return in an edit box will now activate the default button
|
|
|
|
* in the dialog just like it will everywhere else.
|
|
|
|
*/
|
2015-08-08 14:53:43 +00:00
|
|
|
GtkWidget *parent = gtk_widget_get_parent(widget);
|
2015-08-08 15:23:54 +00:00
|
|
|
if (event->keyval == GDK_KEY_Return && parent != NULL) {
|
2007-01-25 19:47:36 +00:00
|
|
|
gboolean return_val;
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(widget), "key_press_event");
|
|
|
|
g_signal_emit_by_name(G_OBJECT(parent), "key_press_event",
|
|
|
|
event, &return_val);
|
2003-03-16 12:37:48 +00:00
|
|
|
return return_val;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
static void editbox_changed(GtkEditable *ed, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
if (!(dp->flags & FLAG_UPDATING_COMBO_LIST)) {
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(ed));
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_VALCHANGE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-25 19:47:36 +00:00
|
|
|
static gboolean editbox_lostfocus(GtkWidget *ed, GdkEventFocus *event,
|
|
|
|
gpointer data)
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(ed));
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_REFRESH);
|
2007-01-25 19:47:36 +00:00
|
|
|
return FALSE;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
2008-04-04 10:56:26 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GTK 1 list box event handlers.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static gboolean listitem_key(GtkWidget *item, GdkEventKey *event,
|
|
|
|
gpointer data, int multiple)
|
|
|
|
{
|
|
|
|
GtkAdjustment *adj = GTK_ADJUSTMENT(data);
|
|
|
|
|
|
|
|
if (event->keyval == GDK_Up || event->keyval == GDK_KP_Up ||
|
|
|
|
event->keyval == GDK_Down || event->keyval == GDK_KP_Down ||
|
|
|
|
event->keyval == GDK_Page_Up || event->keyval == GDK_KP_Page_Up ||
|
|
|
|
event->keyval == GDK_Page_Down || event->keyval == GDK_KP_Page_Down) {
|
|
|
|
/*
|
|
|
|
* Up, Down, PgUp or PgDn have been pressed on a ListItem
|
|
|
|
* in a list box. So, if the list box is single-selection:
|
|
|
|
*
|
|
|
|
* - if the list item in question isn't already selected,
|
|
|
|
* we simply select it.
|
|
|
|
* - otherwise, we find the next one (or next
|
|
|
|
* however-far-away) in whichever direction we're going,
|
|
|
|
* and select that.
|
|
|
|
* + in this case, we must also fiddle with the
|
|
|
|
* scrollbar to ensure the newly selected item is
|
|
|
|
* actually visible.
|
|
|
|
*
|
|
|
|
* If it's multiple-selection, we do all of the above
|
|
|
|
* except actually selecting anything, so we move the focus
|
|
|
|
* and fiddle the scrollbar to follow it.
|
|
|
|
*/
|
|
|
|
GtkWidget *list = item->parent;
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(item), "key_press_event");
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
if (!multiple &&
|
|
|
|
GTK_WIDGET_STATE(item) != GTK_STATE_SELECTED) {
|
|
|
|
gtk_list_select_child(GTK_LIST(list), item);
|
|
|
|
} else {
|
|
|
|
int direction =
|
|
|
|
(event->keyval==GDK_Up || event->keyval==GDK_KP_Up ||
|
|
|
|
event->keyval==GDK_Page_Up || event->keyval==GDK_KP_Page_Up)
|
|
|
|
? -1 : +1;
|
|
|
|
int step =
|
|
|
|
(event->keyval==GDK_Page_Down ||
|
|
|
|
event->keyval==GDK_KP_Page_Down ||
|
|
|
|
event->keyval==GDK_Page_Up || event->keyval==GDK_KP_Page_Up)
|
|
|
|
? 2 : 1;
|
|
|
|
int i, n;
|
|
|
|
GList *children, *chead;
|
|
|
|
|
|
|
|
chead = children = gtk_container_children(GTK_CONTAINER(list));
|
|
|
|
|
|
|
|
n = g_list_length(children);
|
|
|
|
|
|
|
|
if (step == 2) {
|
|
|
|
/*
|
|
|
|
* Figure out how many list items to a screenful,
|
|
|
|
* and adjust the step appropriately.
|
|
|
|
*/
|
|
|
|
step = 0.5 + adj->page_size * n / (adj->upper - adj->lower);
|
|
|
|
step--; /* go by one less than that */
|
|
|
|
}
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
while (children != NULL) {
|
|
|
|
if (item == children->data)
|
|
|
|
break;
|
|
|
|
children = children->next;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (step > 0) {
|
|
|
|
if (direction < 0 && i > 0)
|
|
|
|
children = children->prev, i--;
|
|
|
|
else if (direction > 0 && i < n-1)
|
|
|
|
children = children->next, i++;
|
|
|
|
step--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (children && children->data) {
|
|
|
|
if (!multiple)
|
|
|
|
gtk_list_select_child(GTK_LIST(list),
|
|
|
|
GTK_WIDGET(children->data));
|
|
|
|
gtk_widget_grab_focus(GTK_WIDGET(children->data));
|
|
|
|
gtk_adjustment_clamp_page
|
|
|
|
(adj,
|
|
|
|
adj->lower + (adj->upper-adj->lower) * i / n,
|
|
|
|
adj->lower + (adj->upper-adj->lower) * (i+1) / n);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free(chead);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean listitem_single_key(GtkWidget *item, GdkEventKey *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
return listitem_key(item, event, data, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean listitem_multi_key(GtkWidget *item, GdkEventKey *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
return listitem_key(item, event, data, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean listitem_button_press(GtkWidget *item, GdkEventButton *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(item));
|
|
|
|
switch (event->type) {
|
|
|
|
default:
|
|
|
|
case GDK_BUTTON_PRESS: uc->nclicks = 1; break;
|
|
|
|
case GDK_2BUTTON_PRESS: uc->nclicks = 2; break;
|
|
|
|
case GDK_3BUTTON_PRESS: uc->nclicks = 3; break;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean listitem_button_release(GtkWidget *item, GdkEventButton *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(item));
|
|
|
|
if (uc->nclicks>1) {
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_ACTION);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void list_selchange(GtkList *list, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(list));
|
|
|
|
if (!uc) return;
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_SELCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void draglist_move(struct dlgparam *dp, struct uctrl *uc, int direction)
|
|
|
|
{
|
|
|
|
int index = dlg_listbox_index(uc->ctrl, dp);
|
|
|
|
GList *children = gtk_container_children(GTK_CONTAINER(uc->list));
|
|
|
|
GtkWidget *child;
|
|
|
|
|
|
|
|
if ((index < 0) ||
|
|
|
|
(index == 0 && direction < 0) ||
|
|
|
|
(index == g_list_length(children)-1 && direction > 0)) {
|
|
|
|
gdk_beep();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
child = g_list_nth_data(children, index);
|
|
|
|
gtk_widget_ref(child);
|
|
|
|
gtk_list_clear_items(GTK_LIST(uc->list), index, index+1);
|
|
|
|
g_list_free(children);
|
|
|
|
|
|
|
|
children = NULL;
|
|
|
|
children = g_list_append(children, child);
|
|
|
|
gtk_list_insert_items(GTK_LIST(uc->list), children, index + direction);
|
|
|
|
gtk_list_select_item(GTK_LIST(uc->list), index + direction);
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_VALCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void draglist_up(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(button));
|
|
|
|
draglist_move(dp, uc, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void draglist_down(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(button));
|
|
|
|
draglist_move(dp, uc, +1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* !GTK_CHECK_VERSION(2,0,0) */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GTK 2 list box event handlers.
|
|
|
|
*/
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
static void listbox_doubleclick(GtkTreeView *treeview, GtkTreePath *path,
|
|
|
|
GtkTreeViewColumn *column, gpointer data)
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2008-04-02 14:48:06 +00:00
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(treeview));
|
|
|
|
if (uc)
|
2003-03-15 15:50:42 +00:00
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_ACTION);
|
|
|
|
}
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
static void listbox_selchange(GtkTreeSelection *treeselection,
|
|
|
|
gpointer data)
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2008-04-02 14:48:06 +00:00
|
|
|
GtkTreeView *tree = gtk_tree_selection_get_tree_view(treeselection);
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(tree));
|
|
|
|
if (uc)
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_SELCHANGE);
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
struct draglist_valchange_ctx {
|
|
|
|
struct uctrl *uc;
|
|
|
|
struct dlgparam *dp;
|
|
|
|
};
|
|
|
|
|
|
|
|
static gboolean draglist_valchange(gpointer data)
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
2008-04-02 14:48:06 +00:00
|
|
|
struct draglist_valchange_ctx *ctx =
|
|
|
|
(struct draglist_valchange_ctx *)data;
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
ctx->uc->ctrl->generic.handler(ctx->uc->ctrl, ctx->dp,
|
|
|
|
ctx->dp->data, EVENT_VALCHANGE);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
sfree(ctx);
|
2003-03-18 19:06:51 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
return FALSE;
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
static void listbox_reorder(GtkTreeModel *treemodel, GtkTreePath *path,
|
|
|
|
GtkTreeIter *iter, gpointer data)
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2008-04-02 14:48:06 +00:00
|
|
|
gpointer tree;
|
|
|
|
struct uctrl *uc;
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
if (dp->flags & FLAG_UPDATING_LISTBOX)
|
|
|
|
return; /* not a user drag operation */
|
|
|
|
|
|
|
|
tree = g_object_get_data(G_OBJECT(treemodel), "user-data");
|
|
|
|
uc = dlg_find_bywidget(dp, GTK_WIDGET(tree));
|
|
|
|
if (uc) {
|
|
|
|
/*
|
|
|
|
* We should cause EVENT_VALCHANGE on the list box, now
|
|
|
|
* that its rows have been reordered. However, the GTK 2
|
|
|
|
* docs say that at the point this signal is received the
|
|
|
|
* new row might not have actually been filled in yet.
|
|
|
|
*
|
|
|
|
* (So what smegging use is it then, eh? Don't suppose it
|
|
|
|
* occurred to you at any point that letting the
|
|
|
|
* application know _after_ the reordering was compelete
|
|
|
|
* might be helpful to someone?)
|
|
|
|
*
|
|
|
|
* To get round this, I schedule an idle function, which I
|
|
|
|
* hope won't be called until the main event loop is
|
|
|
|
* re-entered after the drag-and-drop handler has finished
|
|
|
|
* furtling with the list store.
|
|
|
|
*/
|
|
|
|
struct draglist_valchange_ctx *ctx =
|
|
|
|
snew(struct draglist_valchange_ctx);
|
|
|
|
ctx->uc = uc;
|
|
|
|
ctx->dp = dp;
|
|
|
|
g_idle_add(draglist_valchange, ctx);
|
|
|
|
}
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif /* !GTK_CHECK_VERSION(2,0,0) */
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
|
|
|
|
static void menuitem_activate(GtkMenuItem *item, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
GtkWidget *menushell = GTK_WIDGET(item)->parent;
|
2015-08-08 16:29:02 +00:00
|
|
|
gpointer optmenu = g_object_get_data(G_OBJECT(menushell), "user-data");
|
2008-05-31 19:23:45 +00:00
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(optmenu));
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_SELCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
static void droplist_selchange(GtkComboBox *combo, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(combo));
|
|
|
|
if (uc)
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_SELCHANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* !GTK_CHECK_VERSION(2,4,0) */
|
|
|
|
|
2015-08-08 16:34:25 +00:00
|
|
|
#ifdef USE_GTK_FILE_CHOOSER_DIALOG
|
|
|
|
static void filechoose_response(GtkDialog *dialog, gint response,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
/* struct dlgparam *dp = (struct dlgparam *)data; */
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(dialog), "user-data");
|
|
|
|
if (response == GTK_RESPONSE_ACCEPT) {
|
|
|
|
gchar *name = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), name);
|
|
|
|
g_free(name);
|
|
|
|
}
|
|
|
|
gtk_widget_destroy(GTK_WIDGET(dialog));
|
|
|
|
}
|
|
|
|
#else
|
2003-03-15 15:50:42 +00:00
|
|
|
static void filesel_ok(GtkButton *button, gpointer data)
|
|
|
|
{
|
2003-03-29 19:52:50 +00:00
|
|
|
/* struct dlgparam *dp = (struct dlgparam *)data; */
|
2015-08-08 16:29:02 +00:00
|
|
|
gpointer filesel = g_object_get_data(G_OBJECT(button), "user-data");
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(filesel), "user-data");
|
2007-01-25 19:33:29 +00:00
|
|
|
const char *name = gtk_file_selection_get_filename
|
|
|
|
(GTK_FILE_SELECTION(filesel));
|
2003-03-15 15:50:42 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), name);
|
|
|
|
}
|
2015-08-08 16:34:25 +00:00
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
|
|
|
|
static void fontsel_ok(GtkButton *button, gpointer data)
|
|
|
|
{
|
2003-03-29 19:52:50 +00:00
|
|
|
/* struct dlgparam *dp = (struct dlgparam *)data; */
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
gpointer fontsel = g_object_get_data(G_OBJECT(button), "user-data");
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(fontsel), "user-data");
|
2008-04-04 10:56:26 +00:00
|
|
|
const char *name = gtk_font_selection_dialog_get_font_name
|
|
|
|
(GTK_FONT_SELECTION_DIALOG(fontsel));
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), name);
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
unifontsel *fontsel = (unifontsel *)g_object_get_data
|
|
|
|
(G_OBJECT(button), "user-data");
|
2008-03-25 21:49:14 +00:00
|
|
|
struct uctrl *uc = (struct uctrl *)fontsel->user_data;
|
|
|
|
char *name = unifontsel_get_name(fontsel);
|
2008-03-29 14:21:25 +00:00
|
|
|
assert(name); /* should always be ok after OK pressed */
|
2003-03-15 15:50:42 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(uc->entry), name);
|
2008-03-25 21:49:14 +00:00
|
|
|
sfree(name);
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-22 13:49:02 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
|
|
|
|
static void colourchoose_response(GtkDialog *dialog,
|
|
|
|
gint response_id, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(dialog), "user-data");
|
|
|
|
|
|
|
|
if (response_id == GTK_RESPONSE_OK) {
|
|
|
|
GdkRGBA rgba;
|
|
|
|
gtk_color_chooser_get_rgba(GTK_COLOR_CHOOSER(dialog), &rgba);
|
|
|
|
dp->coloursel_result.r = (int) (255 * rgba.red);
|
|
|
|
dp->coloursel_result.g = (int) (255 * rgba.green);
|
|
|
|
dp->coloursel_result.b = (int) (255 * rgba.blue);
|
|
|
|
dp->coloursel_result.ok = TRUE;
|
|
|
|
} else {
|
|
|
|
dp->coloursel_result.ok = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_CALLBACK);
|
|
|
|
|
|
|
|
gtk_widget_destroy(GTK_WIDGET(dialog));
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* GTK 1/2 coloursel response handlers */
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
static void coloursel_ok(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2015-08-08 16:29:02 +00:00
|
|
|
gpointer coloursel = g_object_get_data(G_OBJECT(button), "user-data");
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(coloursel), "user-data");
|
2015-08-22 13:49:02 +00:00
|
|
|
|
2015-08-08 16:32:15 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
{
|
2015-08-22 13:49:02 +00:00
|
|
|
GtkColorSelection *cs = GTK_COLOR_SELECTION
|
|
|
|
(gtk_color_selection_dialog_get_color_selection
|
|
|
|
(GTK_COLOR_SELECTION_DIALOG(coloursel)));
|
2015-08-08 16:32:15 +00:00
|
|
|
GdkColor col;
|
|
|
|
gtk_color_selection_get_current_color(cs, &col);
|
|
|
|
dp->coloursel_result.r = col.red / 0x0100;
|
|
|
|
dp->coloursel_result.g = col.green / 0x0100;
|
|
|
|
dp->coloursel_result.b = col.blue / 0x0100;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
{
|
2015-08-22 13:49:02 +00:00
|
|
|
GtkColorSelection *cs = GTK_COLOR_SELECTION
|
|
|
|
(gtk_color_selection_dialog_get_color_selection
|
|
|
|
(GTK_COLOR_SELECTION_DIALOG(coloursel)));
|
2015-08-08 16:32:15 +00:00
|
|
|
gdouble cvals[4];
|
|
|
|
gtk_color_selection_get_color(cs, cvals);
|
|
|
|
dp->coloursel_result.r = (int) (255 * cvals[0]);
|
|
|
|
dp->coloursel_result.g = (int) (255 * cvals[1]);
|
|
|
|
dp->coloursel_result.b = (int) (255 * cvals[2]);
|
|
|
|
}
|
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
dp->coloursel_result.ok = TRUE;
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_CALLBACK);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void coloursel_cancel(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2015-08-08 16:29:02 +00:00
|
|
|
gpointer coloursel = g_object_get_data(G_OBJECT(button), "user-data");
|
|
|
|
struct uctrl *uc = g_object_get_data(G_OBJECT(coloursel), "user-data");
|
2003-03-15 15:50:42 +00:00
|
|
|
dp->coloursel_result.ok = FALSE;
|
|
|
|
uc->ctrl->generic.handler(uc->ctrl, dp, dp->data, EVENT_CALLBACK);
|
|
|
|
}
|
|
|
|
|
2015-08-22 13:49:02 +00:00
|
|
|
#endif /* end of coloursel response handlers */
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
static void filefont_clicked(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, GTK_WIDGET(button));
|
|
|
|
|
|
|
|
if (uc->ctrl->generic.type == CTRL_FILESELECT) {
|
2015-08-08 16:34:25 +00:00
|
|
|
#ifdef USE_GTK_FILE_CHOOSER_DIALOG
|
|
|
|
GtkWidget *filechoose = gtk_file_chooser_dialog_new
|
|
|
|
(uc->ctrl->fileselect.title, GTK_WINDOW(dp->window),
|
|
|
|
(uc->ctrl->fileselect.for_writing ?
|
|
|
|
GTK_FILE_CHOOSER_ACTION_SAVE :
|
|
|
|
GTK_FILE_CHOOSER_ACTION_OPEN),
|
2015-08-22 12:50:56 +00:00
|
|
|
STANDARD_CANCEL_LABEL, GTK_RESPONSE_CANCEL,
|
|
|
|
STANDARD_OPEN_LABEL, GTK_RESPONSE_ACCEPT,
|
2015-08-08 16:34:25 +00:00
|
|
|
(const gchar *)NULL);
|
|
|
|
gtk_window_set_modal(GTK_WINDOW(filechoose), TRUE);
|
|
|
|
g_object_set_data(G_OBJECT(filechoose), "user-data", (gpointer)uc);
|
|
|
|
g_signal_connect(G_OBJECT(filechoose), "response",
|
|
|
|
G_CALLBACK(filechoose_response), (gpointer)dp);
|
|
|
|
gtk_widget_show(filechoose);
|
|
|
|
#else
|
2003-03-15 15:50:42 +00:00
|
|
|
GtkWidget *filesel =
|
|
|
|
gtk_file_selection_new(uc->ctrl->fileselect.title);
|
|
|
|
gtk_window_set_modal(GTK_WINDOW(filesel), TRUE);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data
|
|
|
|
(G_OBJECT(GTK_FILE_SELECTION(filesel)->ok_button), "user-data",
|
2003-03-15 15:50:42 +00:00
|
|
|
(gpointer)filesel);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(filesel), "user-data", (gpointer)uc);
|
|
|
|
g_signal_connect
|
|
|
|
(G_OBJECT(GTK_FILE_SELECTION(filesel)->ok_button), "clicked",
|
|
|
|
G_CALLBACK(filesel_ok), (gpointer)dp);
|
|
|
|
g_signal_connect_swapped
|
|
|
|
(G_OBJECT(GTK_FILE_SELECTION(filesel)->ok_button), "clicked",
|
|
|
|
G_CALLBACK(gtk_widget_destroy), (gpointer)filesel);
|
|
|
|
g_signal_connect_swapped
|
|
|
|
(G_OBJECT(GTK_FILE_SELECTION(filesel)->cancel_button), "clicked",
|
|
|
|
G_CALLBACK(gtk_widget_destroy), (gpointer)filesel);
|
2003-03-15 15:50:42 +00:00
|
|
|
gtk_widget_show(filesel);
|
2015-08-08 16:34:25 +00:00
|
|
|
#endif
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (uc->ctrl->generic.type == CTRL_FONTSELECT) {
|
2007-01-25 19:33:29 +00:00
|
|
|
const gchar *fontname = gtk_entry_get_text(GTK_ENTRY(uc->entry));
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Use the GTK 1 standard font selector.
|
|
|
|
*/
|
|
|
|
|
|
|
|
gchar *spacings[] = { "c", "m", NULL };
|
|
|
|
GtkWidget *fontsel =
|
|
|
|
gtk_font_selection_dialog_new("Select a font");
|
|
|
|
gtk_window_set_modal(GTK_WINDOW(fontsel), TRUE);
|
|
|
|
gtk_font_selection_dialog_set_filter
|
|
|
|
(GTK_FONT_SELECTION_DIALOG(fontsel),
|
|
|
|
GTK_FONT_FILTER_BASE, GTK_FONT_ALL,
|
|
|
|
NULL, NULL, NULL, NULL, spacings, NULL);
|
|
|
|
if (!gtk_font_selection_dialog_set_font_name
|
|
|
|
(GTK_FONT_SELECTION_DIALOG(fontsel), fontname)) {
|
|
|
|
/*
|
|
|
|
* If the font name wasn't found as it was, try opening
|
|
|
|
* it and extracting its FONT property. This should
|
|
|
|
* have the effect of mapping short aliases into true
|
|
|
|
* XLFDs.
|
|
|
|
*/
|
|
|
|
GdkFont *font = gdk_font_load(fontname);
|
|
|
|
if (font) {
|
|
|
|
XFontStruct *xfs = GDK_FONT_XFONT(font);
|
|
|
|
Display *disp = GDK_FONT_XDISPLAY(font);
|
|
|
|
Atom fontprop = XInternAtom(disp, "FONT", False);
|
|
|
|
unsigned long ret;
|
|
|
|
gdk_font_ref(font);
|
|
|
|
if (XGetFontProperty(xfs, fontprop, &ret)) {
|
|
|
|
char *name = XGetAtomName(disp, (Atom)ret);
|
|
|
|
if (name)
|
|
|
|
gtk_font_selection_dialog_set_font_name
|
|
|
|
(GTK_FONT_SELECTION_DIALOG(fontsel), name);
|
|
|
|
}
|
|
|
|
gdk_font_unref(font);
|
|
|
|
}
|
|
|
|
}
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data
|
|
|
|
(G_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->ok_button),
|
2008-04-04 10:56:26 +00:00
|
|
|
"user-data", (gpointer)fontsel);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(fontsel), "user-data", (gpointer)uc);
|
|
|
|
g_signal_connect
|
|
|
|
(G_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->ok_button),
|
|
|
|
"clicked", G_CALLBACK(fontsel_ok), (gpointer)dp);
|
|
|
|
g_signal_connect_swapped
|
|
|
|
(G_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->ok_button),
|
|
|
|
"clicked", G_CALLBACK(gtk_widget_destroy),
|
2008-04-04 10:56:26 +00:00
|
|
|
(gpointer)fontsel);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect_swapped
|
|
|
|
(G_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->cancel_button),
|
|
|
|
"clicked", G_CALLBACK(gtk_widget_destroy),
|
2008-04-04 10:56:26 +00:00
|
|
|
(gpointer)fontsel);
|
|
|
|
gtk_widget_show(fontsel);
|
|
|
|
|
|
|
|
#else /* !GTK_CHECK_VERSION(2,0,0) */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Use the unifontsel code provided in gtkfont.c.
|
|
|
|
*/
|
|
|
|
|
2008-03-25 21:49:14 +00:00
|
|
|
unifontsel *fontsel = unifontsel_new("Select a font");
|
|
|
|
|
|
|
|
gtk_window_set_modal(fontsel->window, TRUE);
|
|
|
|
unifontsel_set_name(fontsel, fontname);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(fontsel->ok_button),
|
|
|
|
"user-data", (gpointer)fontsel);
|
2008-03-25 21:49:14 +00:00
|
|
|
fontsel->user_data = uc;
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(fontsel->ok_button), "clicked",
|
|
|
|
G_CALLBACK(fontsel_ok), (gpointer)dp);
|
|
|
|
g_signal_connect_swapped(G_OBJECT(fontsel->ok_button), "clicked",
|
|
|
|
G_CALLBACK(unifontsel_destroy),
|
|
|
|
(gpointer)fontsel);
|
|
|
|
g_signal_connect_swapped(G_OBJECT(fontsel->cancel_button),"clicked",
|
|
|
|
G_CALLBACK(unifontsel_destroy),
|
|
|
|
(gpointer)fontsel);
|
2008-03-25 21:49:14 +00:00
|
|
|
|
|
|
|
gtk_widget_show(GTK_WIDGET(fontsel->window));
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
#endif /* !GTK_CHECK_VERSION(2,0,0) */
|
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-23 14:01:07 +00:00
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
2003-03-22 17:00:06 +00:00
|
|
|
static void label_sizealloc(GtkWidget *widget, GtkAllocation *alloc,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
struct uctrl *uc = dlg_find_bywidget(dp, widget);
|
|
|
|
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_widget_set_size_request(uc->text, alloc->width, -1);
|
2003-03-22 17:00:06 +00:00
|
|
|
gtk_label_set_text(GTK_LABEL(uc->text), uc->ctrl->generic.label);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_handler_disconnect(G_OBJECT(uc->text), uc->textsig);
|
2003-03-22 17:00:06 +00:00
|
|
|
}
|
2015-08-23 14:01:07 +00:00
|
|
|
#endif
|
2003-03-22 17:00:06 +00:00
|
|
|
|
2003-03-15 15:50:42 +00:00
|
|
|
/* ----------------------------------------------------------------------
|
2003-03-13 19:52:28 +00:00
|
|
|
* This function does the main layout work: it reads a controlset,
|
|
|
|
* it creates the relevant GTK controls, and returns a GtkWidget
|
|
|
|
* containing the result. (This widget might be a title of some
|
|
|
|
* sort, it might be a Columns containing many controls, or it
|
|
|
|
* might be a GtkFrame containing a Columns; whatever it is, it's
|
|
|
|
* definitely a GtkWidget and should probably be added to a
|
|
|
|
* GtkVbox.)
|
2003-03-16 12:37:48 +00:00
|
|
|
*
|
|
|
|
* `win' is required for setting the default button. If it is
|
|
|
|
* non-NULL, all buttons created will be default-capable (so they
|
|
|
|
* have extra space round them for the default highlight).
|
2003-03-13 19:52:28 +00:00
|
|
|
*/
|
2003-03-16 12:37:48 +00:00
|
|
|
GtkWidget *layout_ctrls(struct dlgparam *dp, struct Shortcuts *scs,
|
2008-04-02 14:48:06 +00:00
|
|
|
struct controlset *s, GtkWindow *win)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
|
|
|
Columns *cols;
|
|
|
|
GtkWidget *ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!s->boxname && s->boxtitle) {
|
|
|
|
/* This controlset is a panel title. */
|
|
|
|
return gtk_label_new(s->boxtitle);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Otherwise, we expect to be laying out actual controls, so
|
|
|
|
* we'll start by creating a Columns for the purpose.
|
|
|
|
*/
|
|
|
|
cols = COLUMNS(columns_new(4));
|
|
|
|
ret = GTK_WIDGET(cols);
|
|
|
|
gtk_widget_show(ret);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create a containing frame if we have a box name.
|
|
|
|
*/
|
|
|
|
if (*s->boxname) {
|
|
|
|
ret = gtk_frame_new(s->boxtitle); /* NULL is valid here */
|
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(cols), 4);
|
|
|
|
gtk_container_add(GTK_CONTAINER(ret), GTK_WIDGET(cols));
|
|
|
|
gtk_widget_show(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now iterate through the controls themselves, create them,
|
|
|
|
* and add them to the Columns.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < s->ncontrols; i++) {
|
|
|
|
union control *ctrl = s->ctrls[i];
|
2003-03-16 12:37:48 +00:00
|
|
|
struct uctrl *uc;
|
|
|
|
int left = FALSE;
|
2003-03-13 19:52:28 +00:00
|
|
|
GtkWidget *w = NULL;
|
|
|
|
|
|
|
|
switch (ctrl->generic.type) {
|
|
|
|
case CTRL_COLUMNS:
|
|
|
|
{
|
|
|
|
static const int simplecols[1] = { 100 };
|
|
|
|
columns_set_cols(cols, ctrl->columns.ncols,
|
|
|
|
(ctrl->columns.percentages ?
|
|
|
|
ctrl->columns.percentages : simplecols));
|
|
|
|
}
|
|
|
|
continue; /* no actual control created */
|
|
|
|
case CTRL_TABDELAY:
|
2003-03-15 15:50:42 +00:00
|
|
|
{
|
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, ctrl->tabdelay.ctrl);
|
|
|
|
if (uc)
|
|
|
|
columns_taborder_last(cols, uc->toplevel);
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
continue; /* no actual control created */
|
2003-03-16 12:37:48 +00:00
|
|
|
}
|
|
|
|
|
2003-03-29 16:14:26 +00:00
|
|
|
uc = snew(struct uctrl);
|
2003-03-16 12:37:48 +00:00
|
|
|
uc->ctrl = ctrl;
|
|
|
|
uc->buttons = NULL;
|
2008-04-04 10:56:26 +00:00
|
|
|
uc->entry = NULL;
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
uc->list = uc->menu = uc->optmenu = NULL;
|
|
|
|
#else
|
2008-05-31 19:23:45 +00:00
|
|
|
uc->combo = NULL;
|
|
|
|
#endif
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
uc->treeview = NULL;
|
2008-04-02 14:48:06 +00:00
|
|
|
uc->listmodel = NULL;
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-04-02 14:48:06 +00:00
|
|
|
uc->button = uc->text = NULL;
|
2006-08-28 14:29:02 +00:00
|
|
|
uc->label = NULL;
|
2007-03-27 18:16:36 +00:00
|
|
|
uc->nclicks = 0;
|
2003-03-16 12:37:48 +00:00
|
|
|
|
|
|
|
switch (ctrl->generic.type) {
|
2003-03-13 19:52:28 +00:00
|
|
|
case CTRL_BUTTON:
|
|
|
|
w = gtk_button_new_with_label(ctrl->generic.label);
|
2003-03-16 12:37:48 +00:00
|
|
|
if (win) {
|
2015-08-08 14:53:43 +00:00
|
|
|
gtk_widget_set_can_default(w, TRUE);
|
2003-03-16 12:37:48 +00:00
|
|
|
if (ctrl->button.isdefault)
|
|
|
|
gtk_window_set_default(win, w);
|
2003-03-18 19:06:51 +00:00
|
|
|
if (ctrl->button.iscancel)
|
|
|
|
dp->cancelbutton = w;
|
2003-03-16 12:37:48 +00:00
|
|
|
}
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "clicked",
|
|
|
|
G_CALLBACK(button_clicked), dp);
|
|
|
|
g_signal_connect(G_OBJECT(w), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2015-08-08 14:53:43 +00:00
|
|
|
shortcut_add(scs, gtk_bin_get_child(GTK_BIN(w)),
|
|
|
|
ctrl->button.shortcut, SHORTCUT_UCTRL, uc);
|
2003-03-13 19:52:28 +00:00
|
|
|
break;
|
|
|
|
case CTRL_CHECKBOX:
|
|
|
|
w = gtk_check_button_new_with_label(ctrl->generic.label);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "toggled",
|
|
|
|
G_CALLBACK(button_toggled), dp);
|
|
|
|
g_signal_connect(G_OBJECT(w), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2015-08-08 14:53:43 +00:00
|
|
|
shortcut_add(scs, gtk_bin_get_child(GTK_BIN(w)),
|
|
|
|
ctrl->checkbox.shortcut, SHORTCUT_UCTRL, uc);
|
2003-03-16 12:37:48 +00:00
|
|
|
left = TRUE;
|
2003-03-13 19:52:28 +00:00
|
|
|
break;
|
|
|
|
case CTRL_RADIO:
|
|
|
|
/*
|
|
|
|
* Radio buttons get to go inside their own Columns, no
|
|
|
|
* matter what.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
gint i, *percentages;
|
|
|
|
GSList *group;
|
|
|
|
|
2003-03-22 17:00:06 +00:00
|
|
|
w = columns_new(0);
|
2003-03-13 19:52:28 +00:00
|
|
|
if (ctrl->generic.label) {
|
|
|
|
GtkWidget *label = gtk_label_new(ctrl->generic.label);
|
|
|
|
columns_add(COLUMNS(w), label, 0, 1);
|
|
|
|
columns_force_left_align(COLUMNS(w), label);
|
|
|
|
gtk_widget_show(label);
|
2003-03-16 12:37:48 +00:00
|
|
|
shortcut_add(scs, label, ctrl->radio.shortcut,
|
|
|
|
SHORTCUT_UCTRL, uc);
|
2006-08-28 14:29:02 +00:00
|
|
|
uc->label = label;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
percentages = g_new(gint, ctrl->radio.ncolumns);
|
|
|
|
for (i = 0; i < ctrl->radio.ncolumns; i++) {
|
|
|
|
percentages[i] =
|
|
|
|
((100 * (i+1) / ctrl->radio.ncolumns) -
|
|
|
|
100 * i / ctrl->radio.ncolumns);
|
|
|
|
}
|
|
|
|
columns_set_cols(COLUMNS(w), ctrl->radio.ncolumns,
|
|
|
|
percentages);
|
|
|
|
g_free(percentages);
|
|
|
|
group = NULL;
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
uc->nbuttons = ctrl->radio.nbuttons;
|
2003-03-29 16:14:26 +00:00
|
|
|
uc->buttons = snewn(uc->nbuttons, GtkWidget *);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-13 19:52:28 +00:00
|
|
|
for (i = 0; i < ctrl->radio.nbuttons; i++) {
|
|
|
|
GtkWidget *b;
|
|
|
|
gint colstart;
|
|
|
|
|
|
|
|
b = (gtk_radio_button_new_with_label
|
|
|
|
(group, ctrl->radio.buttons[i]));
|
2003-03-16 12:37:48 +00:00
|
|
|
uc->buttons[i] = b;
|
2015-08-08 17:02:01 +00:00
|
|
|
group = gtk_radio_button_get_group(GTK_RADIO_BUTTON(b));
|
2003-03-13 19:52:28 +00:00
|
|
|
colstart = i % ctrl->radio.ncolumns;
|
|
|
|
columns_add(COLUMNS(w), b, colstart,
|
|
|
|
(i == ctrl->radio.nbuttons-1 ?
|
|
|
|
ctrl->radio.ncolumns - colstart : 1));
|
2003-03-16 12:37:48 +00:00
|
|
|
columns_force_left_align(COLUMNS(w), b);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_widget_show(b);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(b), "toggled",
|
|
|
|
G_CALLBACK(button_toggled), dp);
|
|
|
|
g_signal_connect(G_OBJECT(b), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2003-03-16 12:37:48 +00:00
|
|
|
if (ctrl->radio.shortcuts) {
|
2015-08-08 14:53:43 +00:00
|
|
|
shortcut_add(scs, gtk_bin_get_child(GTK_BIN(b)),
|
2003-03-16 12:37:48 +00:00
|
|
|
ctrl->radio.shortcuts[i],
|
|
|
|
SHORTCUT_UCTRL, uc);
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CTRL_EDITBOX:
|
2003-03-22 17:00:06 +00:00
|
|
|
{
|
2008-04-04 10:56:26 +00:00
|
|
|
GtkWidget *signalobject;
|
2003-03-22 17:00:06 +00:00
|
|
|
|
|
|
|
if (ctrl->editbox.has_list) {
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
/*
|
|
|
|
* GTK 1 combo box.
|
|
|
|
*/
|
|
|
|
w = gtk_combo_new();
|
|
|
|
gtk_combo_set_value_in_list(GTK_COMBO(w), FALSE, TRUE);
|
|
|
|
uc->entry = GTK_COMBO(w)->entry;
|
|
|
|
uc->list = GTK_COMBO(w)->list;
|
|
|
|
signalobject = uc->entry;
|
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* GTK 2 combo box.
|
|
|
|
*/
|
2008-04-02 14:48:06 +00:00
|
|
|
uc->listmodel = gtk_list_store_new(2, G_TYPE_INT,
|
|
|
|
G_TYPE_STRING);
|
2015-08-08 17:02:01 +00:00
|
|
|
w = gtk_combo_box_new_with_model_and_entry
|
|
|
|
(GTK_TREE_MODEL(uc->listmodel));
|
2015-12-06 07:23:59 +00:00
|
|
|
g_object_set(G_OBJECT(w), "entry-text-column", 1,
|
|
|
|
(const char *)NULL);
|
2008-04-02 14:48:06 +00:00
|
|
|
/* We cannot support password combo boxes. */
|
|
|
|
assert(!ctrl->editbox.password);
|
|
|
|
uc->combo = w;
|
2008-04-04 10:56:26 +00:00
|
|
|
signalobject = uc->combo;
|
|
|
|
#endif
|
2003-03-22 17:00:06 +00:00
|
|
|
} else {
|
|
|
|
w = gtk_entry_new();
|
|
|
|
if (ctrl->editbox.password)
|
|
|
|
gtk_entry_set_visibility(GTK_ENTRY(w), FALSE);
|
|
|
|
uc->entry = w;
|
2008-04-04 10:56:26 +00:00
|
|
|
signalobject = w;
|
2003-03-22 17:00:06 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
uc->entrysig =
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(signalobject), "changed",
|
|
|
|
G_CALLBACK(editbox_changed), dp);
|
|
|
|
g_signal_connect(G_OBJECT(signalobject), "key_press_event",
|
|
|
|
G_CALLBACK(editbox_key), dp);
|
|
|
|
g_signal_connect(G_OBJECT(signalobject), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
|
|
|
g_signal_connect(G_OBJECT(signalobject), "focus_out_event",
|
|
|
|
G_CALLBACK(editbox_lostfocus), dp);
|
|
|
|
g_signal_connect(G_OBJECT(signalobject), "focus_out_event",
|
|
|
|
G_CALLBACK(editbox_lostfocus), dp);
|
2015-08-22 09:37:48 +00:00
|
|
|
|
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
2003-03-22 17:00:06 +00:00
|
|
|
/*
|
|
|
|
* Edit boxes, for some strange reason, have a minimum
|
|
|
|
* width of 150 in GTK 1.2. We don't want this - we'd
|
|
|
|
* rather the edit boxes acquired their natural width
|
|
|
|
* from the column layout of the rest of the box.
|
|
|
|
*/
|
2015-08-24 18:23:38 +00:00
|
|
|
{
|
|
|
|
GtkRequisition req;
|
|
|
|
gtk_widget_size_request(w, &req);
|
|
|
|
gtk_widget_set_size_request(w, 10, req.height);
|
|
|
|
}
|
2015-08-22 09:37:48 +00:00
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* In GTK 3, this is still true, but there's a special
|
|
|
|
* method for GtkEntry in particular to fix it.
|
|
|
|
*/
|
|
|
|
if (GTK_IS_ENTRY(w))
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(w), 1);
|
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2003-03-22 17:00:06 +00:00
|
|
|
if (ctrl->generic.label) {
|
|
|
|
GtkWidget *label, *container;
|
|
|
|
|
|
|
|
label = gtk_label_new(ctrl->generic.label);
|
|
|
|
|
|
|
|
shortcut_add(scs, label, ctrl->editbox.shortcut,
|
|
|
|
SHORTCUT_FOCUS, uc->entry);
|
|
|
|
|
|
|
|
container = columns_new(4);
|
|
|
|
if (ctrl->editbox.percentwidth == 100) {
|
|
|
|
columns_add(COLUMNS(container), label, 0, 1);
|
|
|
|
columns_force_left_align(COLUMNS(container), label);
|
|
|
|
columns_add(COLUMNS(container), w, 0, 1);
|
|
|
|
} else {
|
|
|
|
gint percentages[2];
|
|
|
|
percentages[1] = ctrl->editbox.percentwidth;
|
|
|
|
percentages[0] = 100 - ctrl->editbox.percentwidth;
|
|
|
|
columns_set_cols(COLUMNS(container), 2, percentages);
|
|
|
|
columns_add(COLUMNS(container), label, 0, 1);
|
|
|
|
columns_force_left_align(COLUMNS(container), label);
|
|
|
|
columns_add(COLUMNS(container), w, 1, 1);
|
2015-08-24 18:23:38 +00:00
|
|
|
columns_force_same_height(COLUMNS(container),
|
|
|
|
label, w);
|
2003-03-22 17:00:06 +00:00
|
|
|
}
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_widget_show(w);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2003-03-22 17:00:06 +00:00
|
|
|
w = container;
|
2006-08-28 14:29:02 +00:00
|
|
|
uc->label = label;
|
2003-03-22 17:00:06 +00:00
|
|
|
}
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
break;
|
2003-03-14 18:35:01 +00:00
|
|
|
case CTRL_FILESELECT:
|
|
|
|
case CTRL_FONTSELECT:
|
|
|
|
{
|
|
|
|
GtkWidget *ww;
|
2015-05-15 10:15:42 +00:00
|
|
|
const char *browsebtn =
|
2003-03-14 18:35:01 +00:00
|
|
|
(ctrl->generic.type == CTRL_FILESELECT ?
|
|
|
|
"Browse..." : "Change...");
|
|
|
|
|
|
|
|
gint percentages[] = { 75, 25 };
|
|
|
|
w = columns_new(4);
|
|
|
|
columns_set_cols(COLUMNS(w), 2, percentages);
|
|
|
|
|
|
|
|
if (ctrl->generic.label) {
|
|
|
|
ww = gtk_label_new(ctrl->generic.label);
|
|
|
|
columns_add(COLUMNS(w), ww, 0, 2);
|
|
|
|
columns_force_left_align(COLUMNS(w), ww);
|
|
|
|
gtk_widget_show(ww);
|
2003-03-16 12:37:48 +00:00
|
|
|
shortcut_add(scs, ww,
|
|
|
|
(ctrl->generic.type == CTRL_FILESELECT ?
|
|
|
|
ctrl->fileselect.shortcut :
|
|
|
|
ctrl->fontselect.shortcut),
|
|
|
|
SHORTCUT_UCTRL, uc);
|
2006-08-28 14:29:02 +00:00
|
|
|
uc->label = ww;
|
2003-03-14 18:35:01 +00:00
|
|
|
}
|
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
uc->entry = ww = gtk_entry_new();
|
2015-08-24 18:04:37 +00:00
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
|
|
|
{
|
|
|
|
GtkRequisition req;
|
|
|
|
gtk_widget_size_request(ww, &req);
|
|
|
|
gtk_widget_set_size_request(ww, 10, req.height);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(ww), 1);
|
|
|
|
#endif
|
2003-03-14 18:35:01 +00:00
|
|
|
columns_add(COLUMNS(w), ww, 0, 1);
|
|
|
|
gtk_widget_show(ww);
|
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
uc->button = ww = gtk_button_new_with_label(browsebtn);
|
2003-03-14 18:35:01 +00:00
|
|
|
columns_add(COLUMNS(w), ww, 1, 1);
|
|
|
|
gtk_widget_show(ww);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2015-08-24 18:23:38 +00:00
|
|
|
columns_force_same_height(COLUMNS(w), uc->entry, uc->button);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(uc->entry), "key_press_event",
|
|
|
|
G_CALLBACK(editbox_key), dp);
|
2007-01-25 19:47:36 +00:00
|
|
|
uc->entrysig =
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(uc->entry), "changed",
|
|
|
|
G_CALLBACK(editbox_changed), dp);
|
|
|
|
g_signal_connect(G_OBJECT(uc->entry), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
|
|
|
g_signal_connect(G_OBJECT(uc->button), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
|
|
|
g_signal_connect(G_OBJECT(ww), "clicked",
|
|
|
|
G_CALLBACK(filefont_clicked), dp);
|
2003-03-14 18:35:01 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CTRL_LISTBOX:
|
2008-04-04 10:56:26 +00:00
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
/*
|
2008-05-31 19:23:45 +00:00
|
|
|
* First construct the list data store, with the right
|
|
|
|
* number of columns.
|
2008-04-04 10:56:26 +00:00
|
|
|
*/
|
2008-05-31 19:23:45 +00:00
|
|
|
# if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
/* (For GTK 2.0 to 2.3, we do this for full listboxes only,
|
|
|
|
* because combo boxes are still done the old GTK1 way.) */
|
|
|
|
if (ctrl->listbox.height > 0)
|
|
|
|
# endif
|
|
|
|
{
|
|
|
|
GType *types;
|
|
|
|
int i;
|
|
|
|
int cols;
|
|
|
|
|
|
|
|
cols = ctrl->listbox.ncols;
|
|
|
|
cols = cols ? cols : 1;
|
|
|
|
types = snewn(1 + cols, GType);
|
|
|
|
|
|
|
|
types[0] = G_TYPE_INT;
|
|
|
|
for (i = 0; i < cols; i++)
|
|
|
|
types[i+1] = G_TYPE_STRING;
|
|
|
|
|
|
|
|
uc->listmodel = gtk_list_store_newv(1 + cols, types);
|
|
|
|
|
|
|
|
sfree(types);
|
|
|
|
}
|
|
|
|
#endif
|
2008-04-04 10:56:26 +00:00
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
/*
|
|
|
|
* See if it's a drop-down list (non-editable combo
|
|
|
|
* box).
|
|
|
|
*/
|
|
|
|
if (ctrl->listbox.height == 0) {
|
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
|
|
|
/*
|
|
|
|
* GTK1 and early-GTK2 option-menu style of
|
|
|
|
* drop-down list.
|
|
|
|
*/
|
2008-04-04 10:56:26 +00:00
|
|
|
uc->optmenu = w = gtk_option_menu_new();
|
|
|
|
uc->menu = gtk_menu_new();
|
|
|
|
gtk_option_menu_set_menu(GTK_OPTION_MENU(w), uc->menu);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set_data(G_OBJECT(uc->menu), "user-data",
|
|
|
|
(gpointer)uc->optmenu);
|
|
|
|
g_signal_connect(G_OBJECT(uc->optmenu), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-05-31 19:23:45 +00:00
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* Late-GTK2 style using a GtkComboBox.
|
|
|
|
*/
|
|
|
|
GtkCellRenderer *cr;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create a non-editable GtkComboBox (that is, not
|
|
|
|
* its subclass GtkComboBoxEntry).
|
|
|
|
*/
|
|
|
|
w = gtk_combo_box_new_with_model
|
|
|
|
(GTK_TREE_MODEL(uc->listmodel));
|
|
|
|
uc->combo = w;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Tell it how to render a list item (i.e. which
|
|
|
|
* column to look at in the list model).
|
|
|
|
*/
|
|
|
|
cr = gtk_cell_renderer_text_new();
|
|
|
|
gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(w), cr, TRUE);
|
|
|
|
gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(w), cr,
|
|
|
|
"text", 1, NULL);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* And tell it to notify us when the selection
|
|
|
|
* changes.
|
|
|
|
*/
|
|
|
|
g_signal_connect(G_OBJECT(w), "changed",
|
|
|
|
G_CALLBACK(droplist_selchange), dp);
|
2015-09-26 10:48:21 +00:00
|
|
|
|
|
|
|
g_signal_connect(G_OBJECT(w), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
2008-04-04 10:56:26 +00:00
|
|
|
} else {
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
/*
|
|
|
|
* GTK1-style full list box.
|
|
|
|
*/
|
2008-04-04 10:56:26 +00:00
|
|
|
uc->list = gtk_list_new();
|
|
|
|
if (ctrl->listbox.multisel == 2) {
|
|
|
|
gtk_list_set_selection_mode(GTK_LIST(uc->list),
|
|
|
|
GTK_SELECTION_EXTENDED);
|
|
|
|
} else if (ctrl->listbox.multisel == 1) {
|
|
|
|
gtk_list_set_selection_mode(GTK_LIST(uc->list),
|
|
|
|
GTK_SELECTION_MULTIPLE);
|
|
|
|
} else {
|
|
|
|
gtk_list_set_selection_mode(GTK_LIST(uc->list),
|
|
|
|
GTK_SELECTION_SINGLE);
|
|
|
|
}
|
|
|
|
w = gtk_scrolled_window_new(NULL, NULL);
|
|
|
|
gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(w),
|
|
|
|
uc->list);
|
|
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(w),
|
|
|
|
GTK_POLICY_NEVER,
|
|
|
|
GTK_POLICY_AUTOMATIC);
|
|
|
|
uc->adj = gtk_scrolled_window_get_vadjustment
|
|
|
|
(GTK_SCROLLED_WINDOW(w));
|
|
|
|
|
|
|
|
gtk_widget_show(uc->list);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(uc->list), "selection-changed",
|
|
|
|
G_CALLBACK(list_selchange), dp);
|
|
|
|
g_signal_connect(G_OBJECT(uc->list), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Adjust the height of the scrolled window to the
|
|
|
|
* minimum given by the height parameter.
|
|
|
|
*
|
|
|
|
* This piece of guesswork is a horrid hack based
|
|
|
|
* on looking inside the GTK 1.2 sources
|
|
|
|
* (specifically gtkviewport.c, which appears to be
|
|
|
|
* the widget which provides the border around the
|
|
|
|
* scrolling area). Anyone lets me know how I can
|
|
|
|
* do this in a way which isn't at risk from GTK
|
|
|
|
* upgrades, I'd be grateful.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
int edge;
|
|
|
|
edge = GTK_WIDGET(uc->list)->style->klass->ythickness;
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_widget_set_size_request(w, 10,
|
2008-04-04 10:56:26 +00:00
|
|
|
2*edge + (ctrl->listbox.height *
|
|
|
|
get_listitemheight(w)));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctrl->listbox.draglist) {
|
|
|
|
/*
|
|
|
|
* GTK doesn't appear to make it easy to
|
|
|
|
* implement a proper draggable list; so
|
|
|
|
* instead I'm just going to have to put an Up
|
|
|
|
* and a Down button to the right of the actual
|
|
|
|
* list box. Ah well.
|
|
|
|
*/
|
|
|
|
GtkWidget *cols, *button;
|
|
|
|
static const gint percentages[2] = { 80, 20 };
|
|
|
|
|
|
|
|
cols = columns_new(4);
|
|
|
|
columns_set_cols(COLUMNS(cols), 2, percentages);
|
|
|
|
columns_add(COLUMNS(cols), w, 0, 1);
|
|
|
|
gtk_widget_show(w);
|
|
|
|
button = gtk_button_new_with_label("Up");
|
|
|
|
columns_add(COLUMNS(cols), button, 1, 1);
|
|
|
|
gtk_widget_show(button);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(button), "clicked",
|
|
|
|
G_CALLBACK(draglist_up), dp);
|
|
|
|
g_signal_connect(G_OBJECT(button), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-04-04 10:56:26 +00:00
|
|
|
button = gtk_button_new_with_label("Down");
|
|
|
|
columns_add(COLUMNS(cols), button, 1, 1);
|
|
|
|
gtk_widget_show(button);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(button), "clicked",
|
|
|
|
G_CALLBACK(draglist_down), dp);
|
|
|
|
g_signal_connect(G_OBJECT(button), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-04-04 10:56:26 +00:00
|
|
|
|
|
|
|
w = cols;
|
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
/*
|
2008-05-31 19:23:45 +00:00
|
|
|
* GTK2 treeview-based full list box.
|
2008-04-02 14:48:06 +00:00
|
|
|
*/
|
|
|
|
GtkTreeSelection *sel;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create the list box itself, its columns, and
|
|
|
|
* its containing scrolled window.
|
|
|
|
*/
|
|
|
|
w = gtk_tree_view_new_with_model
|
|
|
|
(GTK_TREE_MODEL(uc->listmodel));
|
|
|
|
g_object_set_data(G_OBJECT(uc->listmodel), "user-data",
|
|
|
|
(gpointer)w);
|
|
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(w), FALSE);
|
|
|
|
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(w));
|
|
|
|
gtk_tree_selection_set_mode
|
|
|
|
(sel, ctrl->listbox.multisel ? GTK_SELECTION_MULTIPLE :
|
|
|
|
GTK_SELECTION_SINGLE);
|
2008-04-04 10:16:24 +00:00
|
|
|
uc->treeview = w;
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "row-activated",
|
|
|
|
G_CALLBACK(listbox_doubleclick), dp);
|
2015-09-26 10:48:21 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "focus_in_event",
|
|
|
|
G_CALLBACK(widget_focus), dp);
|
2008-04-02 14:48:06 +00:00
|
|
|
g_signal_connect(G_OBJECT(sel), "changed",
|
|
|
|
G_CALLBACK(listbox_selchange), dp);
|
|
|
|
|
|
|
|
if (ctrl->listbox.draglist) {
|
|
|
|
gtk_tree_view_set_reorderable(GTK_TREE_VIEW(w), TRUE);
|
|
|
|
g_signal_connect(G_OBJECT(uc->listmodel), "row-inserted",
|
|
|
|
G_CALLBACK(listbox_reorder), dp);
|
|
|
|
}
|
2003-03-14 18:35:01 +00:00
|
|
|
|
|
|
|
{
|
2008-04-02 14:48:06 +00:00
|
|
|
int i;
|
|
|
|
int cols;
|
|
|
|
|
|
|
|
cols = ctrl->listbox.ncols;
|
|
|
|
cols = cols ? cols : 1;
|
|
|
|
for (i = 0; i < cols; i++) {
|
|
|
|
GtkTreeViewColumn *column;
|
2014-09-09 11:46:14 +00:00
|
|
|
GtkCellRenderer *cellrend;
|
2008-04-02 14:48:06 +00:00
|
|
|
/*
|
|
|
|
* It appears that GTK 2 doesn't leave us any
|
|
|
|
* particularly sensible way to honour the
|
|
|
|
* "percentages" specification in the ctrl
|
|
|
|
* structure.
|
|
|
|
*/
|
2014-09-09 11:46:14 +00:00
|
|
|
cellrend = gtk_cell_renderer_text_new();
|
|
|
|
if (!ctrl->listbox.hscroll) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_set(G_OBJECT(cellrend),
|
|
|
|
"ellipsize", PANGO_ELLIPSIZE_END,
|
|
|
|
"ellipsize-set", TRUE,
|
2015-08-31 13:36:12 +00:00
|
|
|
(const char *)NULL);
|
2014-09-09 11:46:14 +00:00
|
|
|
}
|
2008-04-02 14:48:06 +00:00
|
|
|
column = gtk_tree_view_column_new_with_attributes
|
2014-09-09 11:46:14 +00:00
|
|
|
("heading", cellrend, "text", i+1, (char *)NULL);
|
2008-04-02 14:48:06 +00:00
|
|
|
gtk_tree_view_column_set_sizing
|
|
|
|
(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
|
|
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(w), column);
|
|
|
|
}
|
2003-03-14 18:35:01 +00:00
|
|
|
}
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
{
|
|
|
|
GtkWidget *scroll;
|
2003-03-14 18:35:01 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
scroll = gtk_scrolled_window_new(NULL, NULL);
|
|
|
|
gtk_scrolled_window_set_shadow_type
|
|
|
|
(GTK_SCROLLED_WINDOW(scroll), GTK_SHADOW_IN);
|
|
|
|
gtk_widget_show(w);
|
|
|
|
gtk_container_add(GTK_CONTAINER(scroll), w);
|
|
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll),
|
|
|
|
GTK_POLICY_AUTOMATIC,
|
|
|
|
GTK_POLICY_ALWAYS);
|
|
|
|
gtk_widget_set_size_request
|
|
|
|
(scroll, -1,
|
|
|
|
ctrl->listbox.height * get_listitemheight(w));
|
|
|
|
|
|
|
|
w = scroll;
|
|
|
|
}
|
2008-05-31 19:23:45 +00:00
|
|
|
#endif
|
|
|
|
}
|
2003-03-14 18:35:01 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
if (ctrl->generic.label) {
|
|
|
|
GtkWidget *label, *container;
|
|
|
|
|
|
|
|
label = gtk_label_new(ctrl->generic.label);
|
2015-08-25 18:50:23 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
2015-08-23 14:01:07 +00:00
|
|
|
gtk_label_set_width_chars(GTK_LABEL(label), 3);
|
2015-08-25 18:50:23 +00:00
|
|
|
#endif
|
2003-03-14 18:35:01 +00:00
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
shortcut_add(scs, label, ctrl->listbox.shortcut,
|
2015-09-26 12:22:49 +00:00
|
|
|
SHORTCUT_UCTRL, uc);
|
2003-03-14 18:35:01 +00:00
|
|
|
|
|
|
|
container = columns_new(4);
|
2008-04-02 14:48:06 +00:00
|
|
|
if (ctrl->listbox.percentwidth == 100) {
|
|
|
|
columns_add(COLUMNS(container), label, 0, 1);
|
2003-03-14 18:35:01 +00:00
|
|
|
columns_force_left_align(COLUMNS(container), label);
|
2008-04-02 14:48:06 +00:00
|
|
|
columns_add(COLUMNS(container), w, 0, 1);
|
|
|
|
} else {
|
|
|
|
gint percentages[2];
|
|
|
|
percentages[1] = ctrl->listbox.percentwidth;
|
|
|
|
percentages[0] = 100 - ctrl->listbox.percentwidth;
|
|
|
|
columns_set_cols(COLUMNS(container), 2, percentages);
|
|
|
|
columns_add(COLUMNS(container), label, 0, 1);
|
2003-03-14 18:35:01 +00:00
|
|
|
columns_force_left_align(COLUMNS(container), label);
|
2008-04-02 14:48:06 +00:00
|
|
|
columns_add(COLUMNS(container), w, 1, 1);
|
2015-08-24 18:23:38 +00:00
|
|
|
columns_force_same_height(COLUMNS(container),
|
|
|
|
label, w);
|
2008-04-02 14:48:06 +00:00
|
|
|
}
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_widget_show(w);
|
|
|
|
|
|
|
|
w = container;
|
2006-08-28 14:29:02 +00:00
|
|
|
uc->label = label;
|
2008-04-02 14:48:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2003-03-13 19:52:28 +00:00
|
|
|
case CTRL_TEXT:
|
2015-08-23 14:01:07 +00:00
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
2003-03-22 17:00:06 +00:00
|
|
|
/*
|
2015-08-23 14:01:07 +00:00
|
|
|
* Wrapping text widgets don't sit well with the GTK2
|
2003-03-22 17:00:06 +00:00
|
|
|
* layout model, in which widgets state a minimum size
|
|
|
|
* and the whole window then adjusts to the smallest
|
|
|
|
* size it can sensibly take given its contents. A
|
|
|
|
* wrapping text widget _has_ no clear minimum size;
|
|
|
|
* instead it has a range of possibilities. It can be
|
|
|
|
* one line deep but 2000 wide, or two lines deep and
|
|
|
|
* 1000 pixels, or three by 867, or four by 500 and so
|
|
|
|
* on. It can be as short as you like provided you
|
|
|
|
* don't mind it being wide, or as narrow as you like
|
|
|
|
* provided you don't mind it being tall.
|
|
|
|
*
|
|
|
|
* Therefore, it fits very badly into the layout model.
|
|
|
|
* Hence the only thing to do is pick a width and let
|
|
|
|
* it choose its own number of lines. To do this I'm
|
|
|
|
* going to cheat a little. All new wrapping text
|
|
|
|
* widgets will be created with a minimal text content
|
|
|
|
* "X"; then, after the rest of the dialog box is set
|
|
|
|
* up and its size calculated, the text widgets will be
|
|
|
|
* told their width and given their real text, which
|
|
|
|
* will cause the size to be recomputed in the y
|
|
|
|
* direction (because many of them will expand to more
|
|
|
|
* than one line).
|
|
|
|
*/
|
|
|
|
uc->text = w = gtk_label_new("X");
|
|
|
|
uc->textsig =
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "size-allocate",
|
|
|
|
G_CALLBACK(label_sizealloc), dp);
|
2015-08-23 14:01:07 +00:00
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* In GTK3, this is all fixed, because the main aim of the
|
|
|
|
* new 'height-for-width' geometry management is to make
|
|
|
|
* wrapping labels behave sensibly. So now we can just do
|
|
|
|
* the obvious thing.
|
|
|
|
*/
|
|
|
|
uc->text = w = gtk_label_new(uc->ctrl->generic.label);
|
|
|
|
#endif
|
2015-08-31 12:57:34 +00:00
|
|
|
align_label_left(GTK_LABEL(w));
|
2015-08-23 14:01:07 +00:00
|
|
|
gtk_label_set_line_wrap(GTK_LABEL(w), TRUE);
|
2003-03-13 19:52:28 +00:00
|
|
|
break;
|
|
|
|
}
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
assert(w != NULL);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
columns_add(cols, w,
|
|
|
|
COLUMN_START(ctrl->generic.column),
|
|
|
|
COLUMN_SPAN(ctrl->generic.column));
|
|
|
|
if (left)
|
|
|
|
columns_force_left_align(cols, w);
|
|
|
|
gtk_widget_show(w);
|
|
|
|
|
|
|
|
uc->toplevel = w;
|
|
|
|
dlg_add_uctrl(dp, uc);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct selparam {
|
2003-03-16 12:37:48 +00:00
|
|
|
struct dlgparam *dp;
|
2006-12-29 14:35:34 +00:00
|
|
|
GtkNotebook *panels;
|
2007-01-25 19:33:29 +00:00
|
|
|
GtkWidget *panel;
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkWidget *treeitem;
|
2008-04-04 11:37:06 +00:00
|
|
|
#else
|
|
|
|
int depth;
|
|
|
|
GtkTreePath *treepath;
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-16 12:37:48 +00:00
|
|
|
struct Shortcuts shortcuts;
|
2003-03-13 19:52:28 +00:00
|
|
|
};
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
static void treeselection_changed(GtkTreeSelection *treeselection,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct selparam *sps = (struct selparam *)data, *sp;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkTreeIter treeiter;
|
|
|
|
gint spindex;
|
|
|
|
gint page_num;
|
|
|
|
|
|
|
|
if (!gtk_tree_selection_get_selected(treeselection, &treemodel, &treeiter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_tree_model_get(treemodel, &treeiter, TREESTORE_PARAMS, &spindex, -1);
|
|
|
|
sp = &sps[spindex];
|
|
|
|
|
|
|
|
page_num = gtk_notebook_page_num(sp->panels, sp->panel);
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_notebook_set_current_page(sp->panels, page_num);
|
2007-01-25 19:33:29 +00:00
|
|
|
|
|
|
|
dlg_refresh(NULL, sp->dp);
|
|
|
|
|
|
|
|
sp->dp->shortcuts = &sp->shortcuts;
|
|
|
|
}
|
|
|
|
#else
|
2003-03-13 19:52:28 +00:00
|
|
|
static void treeitem_sel(GtkItem *item, gpointer data)
|
|
|
|
{
|
|
|
|
struct selparam *sp = (struct selparam *)data;
|
2006-12-29 14:35:34 +00:00
|
|
|
gint page_num;
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2006-12-29 14:35:34 +00:00
|
|
|
page_num = gtk_notebook_page_num(sp->panels, sp->panel);
|
|
|
|
gtk_notebook_set_page(sp->panels, page_num);
|
2003-03-16 12:37:48 +00:00
|
|
|
|
2006-08-28 14:29:02 +00:00
|
|
|
dlg_refresh(NULL, sp->dp);
|
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
sp->dp->shortcuts = &sp->shortcuts;
|
2003-03-18 19:06:51 +00:00
|
|
|
sp->dp->currtreeitem = sp->treeitem;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2003-03-22 17:00:06 +00:00
|
|
|
static int tree_grab_focus(struct dlgparam *dp)
|
|
|
|
{
|
|
|
|
int i, f;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See if any of the treeitems has the focus.
|
|
|
|
*/
|
|
|
|
f = -1;
|
|
|
|
for (i = 0; i < dp->ntreeitems; i++)
|
|
|
|
if (GTK_WIDGET_HAS_FOCUS(dp->treeitems[i])) {
|
|
|
|
f = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (f >= 0)
|
|
|
|
return FALSE;
|
|
|
|
else {
|
|
|
|
gtk_widget_grab_focus(dp->currtreeitem);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gint tree_focus(GtkContainer *container, GtkDirectionType direction,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(container), "focus");
|
2003-03-22 17:00:06 +00:00
|
|
|
/*
|
|
|
|
* If there's a focused treeitem, we return FALSE to cause the
|
|
|
|
* focus to move on to some totally other control. If not, we
|
|
|
|
* focus the selected one.
|
|
|
|
*/
|
|
|
|
return tree_grab_focus(dp);
|
|
|
|
}
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-22 17:00:06 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
int win_key_press(GtkWidget *widget, GdkEventKey *event, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
|
2015-08-08 15:23:54 +00:00
|
|
|
if (event->keyval == GDK_KEY_Escape && dp->cancelbutton) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name(G_OBJECT(dp->cancelbutton), "clicked");
|
2003-03-16 12:37:48 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((event->state & GDK_MOD1_MASK) &&
|
|
|
|
(unsigned char)event->string[0] > 0 &&
|
|
|
|
(unsigned char)event->string[0] <= 127) {
|
|
|
|
int schr = (unsigned char)event->string[0];
|
|
|
|
struct Shortcut *sc = &dp->shortcuts->sc[schr];
|
|
|
|
|
|
|
|
switch (sc->action) {
|
2003-03-22 17:00:06 +00:00
|
|
|
case SHORTCUT_TREE:
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
gtk_widget_grab_focus(sc->widget);
|
|
|
|
#else
|
2003-03-22 17:00:06 +00:00
|
|
|
tree_grab_focus(dp);
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-22 17:00:06 +00:00
|
|
|
break;
|
2003-03-16 12:37:48 +00:00
|
|
|
case SHORTCUT_FOCUS:
|
|
|
|
gtk_widget_grab_focus(sc->widget);
|
|
|
|
break;
|
|
|
|
case SHORTCUT_UCTRL:
|
|
|
|
/*
|
|
|
|
* We must do something sensible with a uctrl.
|
|
|
|
* Precisely what this is depends on the type of
|
|
|
|
* control.
|
|
|
|
*/
|
|
|
|
switch (sc->uc->ctrl->generic.type) {
|
|
|
|
case CTRL_CHECKBOX:
|
|
|
|
case CTRL_BUTTON:
|
|
|
|
/* Check boxes and buttons get the focus _and_ get toggled. */
|
|
|
|
gtk_widget_grab_focus(sc->uc->toplevel);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name(G_OBJECT(sc->uc->toplevel), "clicked");
|
2003-03-16 12:37:48 +00:00
|
|
|
break;
|
|
|
|
case CTRL_FILESELECT:
|
|
|
|
case CTRL_FONTSELECT:
|
|
|
|
/* File/font selectors have their buttons pressed (ooer),
|
|
|
|
* and focus transferred to the edit box. */
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name(G_OBJECT(sc->uc->button), "clicked");
|
2003-03-16 12:37:48 +00:00
|
|
|
gtk_widget_grab_focus(sc->uc->entry);
|
|
|
|
break;
|
|
|
|
case CTRL_RADIO:
|
|
|
|
/*
|
|
|
|
* Radio buttons are fun, because they have
|
|
|
|
* multiple shortcuts. We must find whether the
|
|
|
|
* activated shortcut is the shortcut for the whole
|
|
|
|
* group, or for a particular button. In the former
|
|
|
|
* case, we find the currently selected button and
|
|
|
|
* focus it; in the latter, we focus-and-click the
|
|
|
|
* button whose shortcut was pressed.
|
|
|
|
*/
|
|
|
|
if (schr == sc->uc->ctrl->radio.shortcut) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < sc->uc->ctrl->radio.nbuttons; i++)
|
|
|
|
if (gtk_toggle_button_get_active
|
|
|
|
(GTK_TOGGLE_BUTTON(sc->uc->buttons[i]))) {
|
|
|
|
gtk_widget_grab_focus(sc->uc->buttons[i]);
|
|
|
|
}
|
|
|
|
} else if (sc->uc->ctrl->radio.shortcuts) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < sc->uc->ctrl->radio.nbuttons; i++)
|
|
|
|
if (schr == sc->uc->ctrl->radio.shortcuts[i]) {
|
|
|
|
gtk_widget_grab_focus(sc->uc->buttons[i]);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name
|
|
|
|
(G_OBJECT(sc->uc->buttons[i]), "clicked");
|
2003-03-16 12:37:48 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
break;
|
|
|
|
case CTRL_LISTBOX:
|
|
|
|
|
2008-05-31 19:23:45 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,4,0)
|
2008-04-04 10:56:26 +00:00
|
|
|
if (sc->uc->optmenu) {
|
|
|
|
GdkEventButton bev;
|
|
|
|
gint returnval;
|
|
|
|
|
|
|
|
gtk_widget_grab_focus(sc->uc->optmenu);
|
|
|
|
/* Option menus don't work using the "clicked" signal.
|
|
|
|
* We need to manufacture a button press event :-/ */
|
|
|
|
bev.type = GDK_BUTTON_PRESS;
|
|
|
|
bev.button = 1;
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name(G_OBJECT(sc->uc->optmenu),
|
|
|
|
"button_press_event",
|
|
|
|
&bev, &returnval);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (sc->uc->combo) {
|
|
|
|
gtk_widget_grab_focus(sc->uc->combo);
|
|
|
|
gtk_combo_box_popup(GTK_COMBO_BOX(sc->uc->combo));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (sc->uc->list) {
|
|
|
|
/*
|
|
|
|
* For GTK-1 style list boxes, we tell it to
|
|
|
|
* focus one of its children, which appears to
|
|
|
|
* do the Right Thing.
|
|
|
|
*/
|
2008-04-04 10:56:26 +00:00
|
|
|
gtk_container_focus(GTK_CONTAINER(sc->uc->list),
|
|
|
|
GTK_DIR_TAB_FORWARD);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
2008-04-04 10:56:26 +00:00
|
|
|
}
|
|
|
|
#else
|
2008-04-04 11:02:26 +00:00
|
|
|
if (sc->uc->treeview) {
|
|
|
|
gtk_widget_grab_focus(sc->uc->treeview);
|
2008-05-31 19:23:45 +00:00
|
|
|
break;
|
2008-04-04 11:02:26 +00:00
|
|
|
}
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2008-05-31 19:23:45 +00:00
|
|
|
assert(!"We shouldn't get here");
|
2003-03-16 12:37:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2003-03-18 19:06:51 +00:00
|
|
|
int tree_key_press(GtkWidget *widget, GdkEventKey *event, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
|
|
|
|
if (event->keyval == GDK_Up || event->keyval == GDK_KP_Up ||
|
|
|
|
event->keyval == GDK_Down || event->keyval == GDK_KP_Down) {
|
2003-03-22 17:00:06 +00:00
|
|
|
int dir, i, j = -1;
|
2003-03-18 19:06:51 +00:00
|
|
|
for (i = 0; i < dp->ntreeitems; i++)
|
2003-03-22 17:00:06 +00:00
|
|
|
if (widget == dp->treeitems[i])
|
|
|
|
break;
|
|
|
|
if (i < dp->ntreeitems) {
|
|
|
|
if (event->keyval == GDK_Up || event->keyval == GDK_KP_Up)
|
|
|
|
dir = -1;
|
|
|
|
else
|
|
|
|
dir = +1;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
i += dir;
|
|
|
|
if (i < 0 || i >= dp->ntreeitems)
|
|
|
|
break; /* nothing in that dir to select */
|
|
|
|
/*
|
|
|
|
* Determine if this tree item is visible.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
GtkWidget *w = dp->treeitems[i];
|
|
|
|
int vis = TRUE;
|
2003-03-29 19:52:50 +00:00
|
|
|
while (w && (GTK_IS_TREE_ITEM(w) || GTK_IS_TREE(w))) {
|
2003-03-22 17:00:06 +00:00
|
|
|
if (!GTK_WIDGET_VISIBLE(w)) {
|
|
|
|
vis = FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
w = w->parent;
|
|
|
|
}
|
|
|
|
if (vis) {
|
|
|
|
j = i; /* got one */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(widget), "key_press_event");
|
2003-03-18 19:06:51 +00:00
|
|
|
if (j >= 0) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_emit_by_name(G_OBJECT(dp->treeitems[j]), "toggle");
|
2003-03-18 19:06:51 +00:00
|
|
|
gtk_widget_grab_focus(dp->treeitems[j]);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2003-03-22 17:00:06 +00:00
|
|
|
* It's nice for Left and Right to expand and collapse tree
|
|
|
|
* branches.
|
2003-03-18 19:06:51 +00:00
|
|
|
*/
|
2003-03-22 17:00:06 +00:00
|
|
|
if (event->keyval == GDK_Left || event->keyval == GDK_KP_Left) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(widget), "key_press_event");
|
2003-03-22 17:00:06 +00:00
|
|
|
gtk_tree_item_collapse(GTK_TREE_ITEM(widget));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
if (event->keyval == GDK_Right || event->keyval == GDK_KP_Right) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_stop_emission_by_name(G_OBJECT(widget), "key_press_event");
|
2003-03-22 17:00:06 +00:00
|
|
|
gtk_tree_item_expand(GTK_TREE_ITEM(widget));
|
|
|
|
return TRUE;
|
2003-03-18 19:06:51 +00:00
|
|
|
}
|
2003-03-22 17:00:06 +00:00
|
|
|
|
|
|
|
return FALSE;
|
2003-03-18 19:06:51 +00:00
|
|
|
}
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-18 19:06:51 +00:00
|
|
|
|
2006-08-28 14:29:02 +00:00
|
|
|
static void shortcut_highlight(GtkWidget *labelw, int chr)
|
2003-03-16 12:37:48 +00:00
|
|
|
{
|
|
|
|
GtkLabel *label = GTK_LABEL(labelw);
|
2015-08-08 17:02:01 +00:00
|
|
|
const gchar *currstr;
|
|
|
|
gchar *pattern;
|
2003-03-16 12:37:48 +00:00
|
|
|
int i;
|
|
|
|
|
2015-08-08 17:02:01 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
{
|
|
|
|
gchar *currstr_nonconst;
|
|
|
|
gtk_label_get(label, &currstr_nonconst);
|
|
|
|
currstr = currstr_nonconst;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
currstr = gtk_label_get_text(label);
|
|
|
|
#endif
|
|
|
|
|
2006-08-28 14:29:02 +00:00
|
|
|
for (i = 0; currstr[i]; i++)
|
|
|
|
if (tolower((unsigned char)currstr[i]) == chr) {
|
|
|
|
pattern = dupprintf("%*s_", i, "");
|
|
|
|
gtk_label_set_pattern(label, pattern);
|
|
|
|
sfree(pattern);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void shortcut_add(struct Shortcuts *scs, GtkWidget *labelw,
|
|
|
|
int chr, int action, void *ptr)
|
|
|
|
{
|
2003-03-16 12:37:48 +00:00
|
|
|
if (chr == NO_SHORTCUT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
chr = tolower((unsigned char)chr);
|
|
|
|
|
|
|
|
assert(scs->sc[chr].action == SHORTCUT_EMPTY);
|
|
|
|
|
|
|
|
scs->sc[chr].action = action;
|
|
|
|
|
2015-09-26 12:12:25 +00:00
|
|
|
if (action == SHORTCUT_FOCUS || action == SHORTCUT_TREE) {
|
2003-03-16 12:37:48 +00:00
|
|
|
scs->sc[chr].uc = NULL;
|
|
|
|
scs->sc[chr].widget = (GtkWidget *)ptr;
|
|
|
|
} else {
|
|
|
|
scs->sc[chr].widget = NULL;
|
|
|
|
scs->sc[chr].uc = (struct uctrl *)ptr;
|
|
|
|
}
|
|
|
|
|
2006-08-28 14:29:02 +00:00
|
|
|
shortcut_highlight(labelw, chr);
|
2003-03-16 12:37:48 +00:00
|
|
|
}
|
|
|
|
|
2008-04-02 14:48:06 +00:00
|
|
|
int get_listitemheight(GtkWidget *w)
|
2003-04-09 18:46:45 +00:00
|
|
|
{
|
2008-04-04 10:56:26 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkWidget *listitem = gtk_list_item_new_with_label("foo");
|
|
|
|
GtkRequisition req;
|
|
|
|
gtk_widget_size_request(listitem, &req);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_ref_sink(G_OBJECT(listitem));
|
2008-04-04 10:56:26 +00:00
|
|
|
return req.height;
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
int height;
|
|
|
|
GtkCellRenderer *cr = gtk_cell_renderer_text_new();
|
2015-08-22 12:53:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
{
|
|
|
|
GtkRequisition req;
|
|
|
|
/*
|
|
|
|
* Since none of my list items wraps in this GUI, no
|
|
|
|
* interesting width-for-height behaviour should be happening,
|
|
|
|
* so I don't think it should matter here whether I ask for
|
|
|
|
* the minimum or natural height.
|
|
|
|
*/
|
|
|
|
gtk_cell_renderer_get_preferred_size(cr, w, &req, NULL);
|
|
|
|
height = req.height;
|
|
|
|
}
|
|
|
|
#else
|
2008-04-02 14:48:06 +00:00
|
|
|
gtk_cell_renderer_get_size(cr, w, NULL, NULL, NULL, NULL, &height);
|
2015-08-22 12:53:29 +00:00
|
|
|
#endif
|
2008-04-02 14:48:06 +00:00
|
|
|
g_object_ref(G_OBJECT(cr));
|
2015-08-08 16:29:02 +00:00
|
|
|
g_object_ref_sink(G_OBJECT(cr));
|
2008-04-02 14:48:06 +00:00
|
|
|
g_object_unref(G_OBJECT(cr));
|
|
|
|
return height;
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
|
2015-08-25 18:44:44 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
void initial_treeview_collapse(struct dlgparam *dp, GtkWidget *tree)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Collapse the deeper branches of the treeview into the state we
|
|
|
|
* like them to start off in. See comment below in do_config_box.
|
|
|
|
*/
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < dp->nselparams; i++)
|
|
|
|
if (dp->selparams[i].depth >= 2)
|
|
|
|
gtk_tree_view_collapse_row(GTK_TREE_VIEW(tree),
|
|
|
|
dp->selparams[i].treepath);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if GTK_CHECK_VERSION(3,0,0)
|
|
|
|
void treeview_map_event(GtkWidget *tree, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
|
|
|
GtkAllocation alloc;
|
|
|
|
gtk_widget_get_allocation(tree, &alloc);
|
|
|
|
gtk_widget_set_size_request(tree, alloc.width, -1);
|
|
|
|
initial_treeview_collapse(dp, tree);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
GtkWidget *create_config_box(const char *title, Conf *conf,
|
|
|
|
int midsession, int protcfginfo,
|
|
|
|
post_dialog_fn_t after, void *afterctx)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
|
|
|
GtkWidget *window, *hbox, *vbox, *cols, *label,
|
|
|
|
*tree, *treescroll, *panels, *panelvbox;
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
int index, level, protocol;
|
2003-03-13 19:52:28 +00:00
|
|
|
char *path;
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkTreeStore *treestore;
|
|
|
|
GtkCellRenderer *treerenderer;
|
|
|
|
GtkTreeViewColumn *treecolumn;
|
|
|
|
GtkTreeSelection *treeselection;
|
|
|
|
GtkTreeIter treeiterlevels[8];
|
|
|
|
#else
|
2003-03-13 19:52:28 +00:00
|
|
|
GtkTreeItem *treeitemlevels[8];
|
|
|
|
GtkTree *treelevels[8];
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
struct dlgparam *dp;
|
2003-03-16 12:37:48 +00:00
|
|
|
struct Shortcuts scs;
|
2003-03-13 19:52:28 +00:00
|
|
|
|
|
|
|
struct selparam *selparams = NULL;
|
|
|
|
int nselparams = 0, selparamsize = 0;
|
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp = snew(struct dlgparam);
|
|
|
|
dp->after = after;
|
|
|
|
dp->afterctx = afterctx;
|
|
|
|
|
|
|
|
dlg_init(dp);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
2003-03-16 12:37:48 +00:00
|
|
|
for (index = 0; index < lenof(scs.sc); index++) {
|
|
|
|
scs.sc[index].action = SHORTCUT_EMPTY;
|
|
|
|
}
|
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
window = our_dialog_new();
|
2003-04-10 08:53:43 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->ctrlbox = ctrl_new_box();
|
Post-release destabilisation! Completely remove the struct type
'Config' in putty.h, which stores all PuTTY's settings and includes an
arbitrary length limit on every single one of those settings which is
stored in string form. In place of it is 'Conf', an opaque data type
everywhere outside the new file conf.c, which stores a list of (key,
value) pairs in which every key contains an integer identifying a
configuration setting, and for some of those integers the key also
contains extra parts (so that, for instance, CONF_environmt is a
string-to-string mapping). Everywhere that a Config was previously
used, a Conf is now; everywhere there was a Config structure copy,
conf_copy() is called; every lookup, adjustment, load and save
operation on a Config has been rewritten; and there's a mechanism for
serialising a Conf into a binary blob and back for use with Duplicate
Session.
User-visible effects of this change _should_ be minimal, though I
don't doubt I've introduced one or two bugs here and there which will
eventually be found. The _intended_ visible effects of this change are
that all arbitrary limits on configuration strings and lists (e.g.
limit on number of port forwardings) should now disappear; that list
boxes in the configuration will now be displayed in a sorted order
rather than the arbitrary order in which they were added to the list
(since the underlying data structure is now a sorted tree234 rather
than an ad-hoc comma-separated string); and one more specific change,
which is that local and dynamic port forwardings on the same port
number are now mutually exclusive in the configuration (putting 'D' in
the key rather than the value was a mistake in the first place).
One other reorganisation as a result of this is that I've moved all
the dialog.c standard handlers (dlg_stdeditbox_handler and friends)
out into config.c, because I can't really justify calling them generic
any more. When they took a pointer to an arbitrary structure type and
the offset of a field within that structure, they were independent of
whether that structure was a Config or something completely different,
but now they really do expect to talk to a Conf, which can _only_ be
used for PuTTY configuration, so I've renamed them all things like
conf_editbox_handler and moved them out of the nominally independent
dialog-box management module into the PuTTY-specific config.c.
[originally from svn r9214]
2011-07-14 18:52:21 +00:00
|
|
|
protocol = conf_get_int(conf, CONF_protocol);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
setup_config_box(dp->ctrlbox, midsession, protocol, protcfginfo);
|
|
|
|
unix_setup_config_box(dp->ctrlbox, midsession, protocol);
|
|
|
|
gtk_setup_config_box(dp->ctrlbox, midsession, window);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2003-03-22 17:00:06 +00:00
|
|
|
gtk_window_set_title(GTK_WINDOW(window), title);
|
2003-03-13 19:52:28 +00:00
|
|
|
hbox = gtk_hbox_new(FALSE, 4);
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_add_to_content_area(GTK_WINDOW(window), hbox, TRUE, TRUE, 0);
|
2003-03-14 18:35:01 +00:00
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(hbox), 10);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_widget_show(hbox);
|
|
|
|
vbox = gtk_vbox_new(FALSE, 4);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), vbox, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show(vbox);
|
|
|
|
cols = columns_new(4);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), cols, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show(cols);
|
|
|
|
label = gtk_label_new("Category:");
|
|
|
|
columns_add(COLUMNS(cols), label, 0, 1);
|
|
|
|
columns_force_left_align(COLUMNS(cols), label);
|
|
|
|
gtk_widget_show(label);
|
|
|
|
treescroll = gtk_scrolled_window_new(NULL, NULL);
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
treestore = gtk_tree_store_new
|
|
|
|
(TREESTORE_NUM, G_TYPE_STRING, G_TYPE_INT);
|
|
|
|
tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(treestore));
|
|
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(tree), FALSE);
|
|
|
|
treerenderer = gtk_cell_renderer_text_new();
|
|
|
|
treecolumn = gtk_tree_view_column_new_with_attributes
|
|
|
|
("Label", treerenderer, "text", 0, NULL);
|
|
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(tree), treecolumn);
|
|
|
|
treeselection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree));
|
|
|
|
gtk_tree_selection_set_mode(treeselection, GTK_SELECTION_BROWSE);
|
|
|
|
gtk_container_add(GTK_CONTAINER(treescroll), tree);
|
|
|
|
#else
|
2003-03-13 19:52:28 +00:00
|
|
|
tree = gtk_tree_new();
|
|
|
|
gtk_tree_set_view_mode(GTK_TREE(tree), GTK_TREE_VIEW_ITEM);
|
2003-03-14 18:35:01 +00:00
|
|
|
gtk_tree_set_selection_mode(GTK_TREE(tree), GTK_SELECTION_BROWSE);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(tree), "focus", G_CALLBACK(tree_focus), dp);
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(tree), "focus_in_event",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(widget_focus), dp);
|
2007-01-25 19:33:29 +00:00
|
|
|
shortcut_add(&scs, label, 'g', SHORTCUT_TREE, tree);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_widget_show(treescroll);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), treescroll, TRUE, TRUE, 0);
|
2006-12-29 14:35:34 +00:00
|
|
|
panels = gtk_notebook_new();
|
|
|
|
gtk_notebook_set_show_tabs(GTK_NOTEBOOK(panels), FALSE);
|
|
|
|
gtk_notebook_set_show_border(GTK_NOTEBOOK(panels), FALSE);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), panels, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show(panels);
|
|
|
|
|
|
|
|
panelvbox = NULL;
|
|
|
|
path = NULL;
|
|
|
|
level = 0;
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
for (index = 0; index < dp->ctrlbox->nctrlsets; index++) {
|
|
|
|
struct controlset *s = dp->ctrlbox->ctrlsets[index];
|
2003-03-16 12:37:48 +00:00
|
|
|
GtkWidget *w;
|
2003-03-13 19:52:28 +00:00
|
|
|
|
|
|
|
if (!*s->pathname) {
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
w = layout_ctrls(dp, &scs, s, GTK_WINDOW(window));
|
2007-01-25 19:45:50 +00:00
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_set_action_area(GTK_WINDOW(window), w);
|
2003-03-13 19:52:28 +00:00
|
|
|
} else {
|
|
|
|
int j = path ? ctrl_path_compare(s->pathname, path) : 0;
|
|
|
|
if (j != INT_MAX) { /* add to treeview, start new panel */
|
|
|
|
char *c;
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
GtkTreeIter treeiter;
|
|
|
|
#else
|
2003-03-13 19:52:28 +00:00
|
|
|
GtkWidget *treeitem;
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
int first;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We expect never to find an implicit path
|
|
|
|
* component. For example, we expect never to see
|
|
|
|
* A/B/C followed by A/D/E, because that would
|
|
|
|
* _implicitly_ create A/D. All our path prefixes
|
|
|
|
* are expected to contain actual controls and be
|
|
|
|
* selectable in the treeview; so we would expect
|
|
|
|
* to see A/D _explicitly_ before encountering
|
|
|
|
* A/D/E.
|
|
|
|
*/
|
|
|
|
assert(j == ctrl_path_elements(s->pathname) - 1);
|
|
|
|
|
|
|
|
c = strrchr(s->pathname, '/');
|
|
|
|
if (!c)
|
|
|
|
c = s->pathname;
|
|
|
|
else
|
|
|
|
c++;
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
path = s->pathname;
|
|
|
|
|
|
|
|
first = (panelvbox == NULL);
|
|
|
|
|
|
|
|
panelvbox = gtk_vbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(panelvbox);
|
|
|
|
gtk_notebook_append_page(GTK_NOTEBOOK(panels), panelvbox,
|
|
|
|
NULL);
|
|
|
|
if (first) {
|
|
|
|
gint page_num;
|
|
|
|
|
|
|
|
page_num = gtk_notebook_page_num(GTK_NOTEBOOK(panels),
|
|
|
|
panelvbox);
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_notebook_set_current_page(GTK_NOTEBOOK(panels),
|
|
|
|
page_num);
|
2007-01-25 19:33:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (nselparams >= selparamsize) {
|
|
|
|
selparamsize += 16;
|
|
|
|
selparams = sresize(selparams, selparamsize,
|
|
|
|
struct selparam);
|
|
|
|
}
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
selparams[nselparams].dp = dp;
|
2007-01-25 19:33:29 +00:00
|
|
|
selparams[nselparams].panels = GTK_NOTEBOOK(panels);
|
|
|
|
selparams[nselparams].panel = panelvbox;
|
|
|
|
selparams[nselparams].shortcuts = scs; /* structure copy */
|
|
|
|
|
2003-03-13 19:52:28 +00:00
|
|
|
assert(j-1 < level);
|
2007-01-25 19:33:29 +00:00
|
|
|
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
if (j > 0)
|
|
|
|
/* treeiterlevels[j-1] will always be valid because we
|
|
|
|
* don't allow implicit path components; see above.
|
|
|
|
*/
|
|
|
|
gtk_tree_store_append(treestore, &treeiter,
|
|
|
|
&treeiterlevels[j-1]);
|
|
|
|
else
|
|
|
|
gtk_tree_store_append(treestore, &treeiter, NULL);
|
|
|
|
gtk_tree_store_set(treestore, &treeiter,
|
|
|
|
TREESTORE_PATH, c,
|
|
|
|
TREESTORE_PARAMS, nselparams,
|
|
|
|
-1);
|
|
|
|
treeiterlevels[j] = treeiter;
|
2007-01-26 08:01:47 +00:00
|
|
|
|
2008-04-05 12:53:32 +00:00
|
|
|
selparams[nselparams].depth = j;
|
2007-01-26 08:01:47 +00:00
|
|
|
if (j > 0) {
|
2008-04-04 11:37:06 +00:00
|
|
|
selparams[nselparams].treepath =
|
|
|
|
gtk_tree_model_get_path(GTK_TREE_MODEL(treestore),
|
|
|
|
&treeiterlevels[j-1]);
|
|
|
|
/*
|
|
|
|
* We are going to collapse all tree branches
|
|
|
|
* at depth greater than 2, but not _yet_; see
|
|
|
|
* the comment at the call to
|
|
|
|
* gtk_tree_view_collapse_row below.
|
|
|
|
*/
|
|
|
|
gtk_tree_view_expand_row(GTK_TREE_VIEW(tree),
|
|
|
|
selparams[nselparams].treepath,
|
|
|
|
FALSE);
|
2008-04-05 12:53:32 +00:00
|
|
|
} else {
|
|
|
|
selparams[nselparams].treepath = NULL;
|
2007-01-26 08:01:47 +00:00
|
|
|
}
|
2007-01-25 19:33:29 +00:00
|
|
|
#else
|
|
|
|
treeitem = gtk_tree_item_new_with_label(c);
|
2003-03-13 19:52:28 +00:00
|
|
|
if (j > 0) {
|
|
|
|
if (!treelevels[j-1]) {
|
|
|
|
treelevels[j-1] = GTK_TREE(gtk_tree_new());
|
|
|
|
gtk_tree_item_set_subtree
|
|
|
|
(treeitemlevels[j-1],
|
|
|
|
GTK_WIDGET(treelevels[j-1]));
|
2007-01-16 18:48:47 +00:00
|
|
|
if (j < 2)
|
|
|
|
gtk_tree_item_expand(treeitemlevels[j-1]);
|
|
|
|
else
|
|
|
|
gtk_tree_item_collapse(treeitemlevels[j-1]);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
gtk_tree_append(treelevels[j-1], treeitem);
|
|
|
|
} else {
|
|
|
|
gtk_tree_append(GTK_TREE(tree), treeitem);
|
|
|
|
}
|
|
|
|
treeitemlevels[j] = GTK_TREE_ITEM(treeitem);
|
|
|
|
treelevels[j] = NULL;
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(treeitem), "key_press_event",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(tree_key_press), dp);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(treeitem), "focus_in_event",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(widget_focus), dp);
|
2003-03-18 19:06:51 +00:00
|
|
|
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_widget_show(treeitem);
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
if (first)
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_tree_select_child(GTK_TREE(tree), treeitem);
|
|
|
|
selparams[nselparams].treeitem = treeitem;
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
level = j+1;
|
|
|
|
nselparams++;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
w = layout_ctrls(dp, &selparams[nselparams-1].shortcuts, s, NULL);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(panelvbox), w, FALSE, FALSE, 0);
|
2003-03-14 18:35:01 +00:00
|
|
|
gtk_widget_show(w);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-04 11:37:06 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
2015-08-25 18:44:44 +00:00
|
|
|
/*
|
|
|
|
* We want our tree view to come up with all branches at depth 2
|
|
|
|
* or more collapsed. However, if we start off with those branches
|
|
|
|
* collapsed, then the tree view's size request will be calculated
|
|
|
|
* based on the width of the collapsed tree, and then when the
|
|
|
|
* collapsed branches are expanded later, the tree view will
|
|
|
|
* jarringly change size.
|
|
|
|
*
|
|
|
|
* So instead we start with everything expanded; then, once the
|
|
|
|
* tree view has computed its resulting width requirement, we
|
|
|
|
* collapse the relevant rows, but force the width to be the value
|
|
|
|
* we just retrieved. This arranges that the tree view is wide
|
|
|
|
* enough to have all branches expanded without further resizing.
|
|
|
|
*/
|
2008-04-04 11:37:06 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->nselparams = nselparams;
|
|
|
|
dp->selparams = selparams;
|
2008-04-04 11:37:06 +00:00
|
|
|
|
2015-08-25 18:44:44 +00:00
|
|
|
#if !GTK_CHECK_VERSION(3,0,0)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* In GTK2, we can just do the job right now.
|
|
|
|
*/
|
|
|
|
GtkRequisition req;
|
2008-04-04 11:37:06 +00:00
|
|
|
gtk_widget_size_request(tree, &req);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
initial_treeview_collapse(dp, tree);
|
2008-04-04 11:37:06 +00:00
|
|
|
gtk_widget_set_size_request(tree, req.width, -1);
|
|
|
|
}
|
2015-08-25 18:44:44 +00:00
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* But in GTK3, we have to wait until the widget is about to be
|
|
|
|
* mapped, because the size computation won't have been done yet.
|
|
|
|
*/
|
|
|
|
g_signal_connect(G_OBJECT(tree), "map",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(treeview_map_event), dp);
|
2015-08-25 18:44:44 +00:00
|
|
|
#endif /* GTK 2 vs 3 */
|
|
|
|
#endif /* GTK 2+ vs 1 */
|
2008-04-04 11:37:06 +00:00
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
g_signal_connect(G_OBJECT(treeselection), "changed",
|
|
|
|
G_CALLBACK(treeselection_changed), selparams);
|
|
|
|
#else
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->ntreeitems = nselparams;
|
|
|
|
dp->treeitems = snewn(dp->ntreeitems, GtkWidget *);
|
2003-03-13 19:52:28 +00:00
|
|
|
for (index = 0; index < nselparams; index++) {
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(selparams[index].treeitem), "select",
|
|
|
|
G_CALLBACK(treeitem_sel),
|
|
|
|
&selparams[index]);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->treeitems[index] = selparams[index].treeitem;
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2003-03-13 19:52:28 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->data = conf;
|
|
|
|
dlg_refresh(NULL, dp);
|
2003-03-15 15:50:42 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->shortcuts = &selparams[0].shortcuts;
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->currtreeitem = dp->treeitems[0];
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dp->lastfocus = NULL;
|
|
|
|
dp->retval = -1;
|
|
|
|
dp->window = window;
|
2003-03-16 12:37:48 +00:00
|
|
|
|
2007-01-06 18:27:00 +00:00
|
|
|
{
|
|
|
|
/* in gtkwin.c */
|
|
|
|
extern void set_window_icon(GtkWidget *window,
|
|
|
|
const char *const *const *icon,
|
|
|
|
int n_icon);
|
|
|
|
extern const char *const *const cfg_icon[];
|
|
|
|
extern const int n_cfg_icon;
|
|
|
|
set_window_icon(window, cfg_icon, n_cfg_icon);
|
|
|
|
}
|
|
|
|
|
2007-01-25 19:33:29 +00:00
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
2007-01-16 18:48:47 +00:00
|
|
|
gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(treescroll),
|
|
|
|
tree);
|
2007-01-25 19:33:29 +00:00
|
|
|
#endif
|
2007-01-16 18:48:47 +00:00
|
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(treescroll),
|
|
|
|
GTK_POLICY_NEVER,
|
|
|
|
GTK_POLICY_AUTOMATIC);
|
|
|
|
gtk_widget_show(tree);
|
|
|
|
|
2003-03-18 19:47:28 +00:00
|
|
|
gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
|
2003-03-13 19:52:28 +00:00
|
|
|
gtk_widget_show(window);
|
|
|
|
|
2003-03-18 19:06:51 +00:00
|
|
|
/*
|
|
|
|
* Set focus into the first available control.
|
|
|
|
*/
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
for (index = 0; index < dp->ctrlbox->nctrlsets; index++) {
|
|
|
|
struct controlset *s = dp->ctrlbox->ctrlsets[index];
|
2003-03-18 19:06:51 +00:00
|
|
|
int done = 0;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
if (*s->pathname) {
|
|
|
|
for (j = 0; j < s->ncontrols; j++)
|
|
|
|
if (s->ctrls[j]->generic.type != CTRL_TABDELAY &&
|
|
|
|
s->ctrls[j]->generic.type != CTRL_COLUMNS &&
|
|
|
|
s->ctrls[j]->generic.type != CTRL_TEXT) {
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dlg_set_focus(s->ctrls[j], dp);
|
|
|
|
dp->lastfocus = s->ctrls[j];
|
2003-03-18 19:06:51 +00:00
|
|
|
done = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (done)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(window), "destroy",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(dlgparam_destroy), dp);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(window), "key_press_event",
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
G_CALLBACK(win_key_press), dp);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
return window;
|
|
|
|
}
|
2003-03-18 19:06:51 +00:00
|
|
|
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
static void dlgparam_destroy(GtkWidget *widget, gpointer data)
|
|
|
|
{
|
|
|
|
struct dlgparam *dp = (struct dlgparam *)data;
|
2017-11-26 14:37:38 +00:00
|
|
|
dp->after(dp->afterctx, dp->retval);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
dlg_cleanup(dp);
|
|
|
|
ctrl_free_box(dp->ctrlbox);
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
2017-11-26 14:29:31 +00:00
|
|
|
if (dp->selparams) {
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < dp->nselparams; i++)
|
|
|
|
if (dp->selparams[i].treepath)
|
|
|
|
gtk_tree_path_free(dp->selparams[i].treepath);
|
2017-11-26 14:29:31 +00:00
|
|
|
sfree(dp->selparams);
|
Make the configuration dialog non-modal.
Now every call to do_config_box is replaced with a call to
create_config_box, which returns immediately having constructed the
new GTK window object, and is passed a callback function which it will
arrange to be called when the dialog terminates (whether by OK or by
Cancel). That callback is now what triggers the construction of a
session window after 'Open' is pressed in the initial config box, or
the actual mid-session reconfiguration action after 'Apply' is pressed
in a Change Settings box.
We were already prepared to ignore the re-selection of 'Change
Settings' from the context menu of a window that already had a Change
Settings box open (and not accidentally create a second config box for
the same window); but now we do slightly better, by finding the
existing config box and un-minimising and raising it, in case the user
had forgotten it was there.
That's a useful featurelet, but not the main purpose of this change.
The mani point, of course, is that now the multi-window GtkApplication
based front ends now don't do anything confusing to the nesting of
gtk_main() when config boxes are involved. Whether you're changing the
settings of one (or more than one) of your already-running sessions,
preparing to start up a new PuTTY connection, or both at once, we stay
in the same top-level instance of gtk_main() and all sessions' top-
level callbacks continue to run sensibly.
2017-11-26 11:58:02 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
sfree(dp);
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
static void messagebox_handler(union control *ctrl, void *dlg,
|
|
|
|
void *data, int event)
|
|
|
|
{
|
|
|
|
if (event == EVENT_ACTION)
|
|
|
|
dlg_end(dlg, ctrl->generic.context.i);
|
|
|
|
}
|
2017-11-26 14:05:02 +00:00
|
|
|
|
|
|
|
const struct message_box_button button_array_yn[] = {
|
|
|
|
{"Yes", 'y', +1, 1},
|
|
|
|
{"No", 'n', -1, 0},
|
|
|
|
};
|
|
|
|
const struct message_box_buttons buttons_yn = {
|
|
|
|
button_array_yn, lenof(button_array_yn),
|
|
|
|
};
|
|
|
|
const struct message_box_button button_array_ok[] = {
|
|
|
|
{"OK", 'o', 1, 1},
|
|
|
|
};
|
|
|
|
const struct message_box_buttons buttons_ok = {
|
|
|
|
button_array_ok, lenof(button_array_ok),
|
|
|
|
};
|
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
GtkWidget *create_message_box(
|
2017-11-26 14:05:02 +00:00
|
|
|
GtkWidget *parentwin, const char *title, const char *msg, int minwid,
|
2017-11-26 14:29:31 +00:00
|
|
|
int selectable, const struct message_box_buttons *buttons,
|
|
|
|
post_dialog_fn_t after, void *afterctx)
|
2003-03-31 11:21:07 +00:00
|
|
|
{
|
|
|
|
GtkWidget *window, *w0, *w1;
|
|
|
|
struct controlset *s0, *s1;
|
2015-12-22 12:32:48 +00:00
|
|
|
union control *c, *textctrl;
|
2017-11-26 14:29:31 +00:00
|
|
|
struct dlgparam *dp;
|
2003-03-31 11:21:07 +00:00
|
|
|
struct Shortcuts scs;
|
2017-11-26 14:05:02 +00:00
|
|
|
int i, index, ncols, min_type;
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
dp = snew(struct dlgparam);
|
|
|
|
dp->after = after;
|
|
|
|
dp->afterctx = afterctx;
|
|
|
|
|
|
|
|
dlg_init(dp);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
for (index = 0; index < lenof(scs.sc); index++) {
|
|
|
|
scs.sc[index].action = SHORTCUT_EMPTY;
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
dp->ctrlbox = ctrl_new_box();
|
2003-03-22 17:00:06 +00:00
|
|
|
|
2015-09-26 12:57:12 +00:00
|
|
|
/*
|
2017-11-26 14:05:02 +00:00
|
|
|
* Count up the number of buttons and find out what kinds there
|
|
|
|
* are.
|
2015-09-26 12:57:12 +00:00
|
|
|
*/
|
2003-03-31 11:21:07 +00:00
|
|
|
ncols = 0;
|
2015-09-26 12:57:12 +00:00
|
|
|
min_type = +1;
|
2017-11-26 14:05:02 +00:00
|
|
|
for (i = 0; i < buttons->nbuttons; i++) {
|
|
|
|
const struct message_box_button *button = &buttons->buttons[i];
|
2015-09-26 12:57:12 +00:00
|
|
|
ncols++;
|
2017-11-26 14:05:02 +00:00
|
|
|
if (min_type > button->type)
|
|
|
|
min_type = button->type;
|
2017-11-26 14:29:31 +00:00
|
|
|
assert(button->value >= 0); /* <0 means no return value available */
|
2003-03-31 11:21:07 +00:00
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
s0 = ctrl_getset(dp->ctrlbox, "", "", "");
|
2003-03-31 11:21:07 +00:00
|
|
|
c = ctrl_columns(s0, 2, 50, 50);
|
|
|
|
c->columns.ncols = s0->ncolumns = ncols;
|
|
|
|
c->columns.percentages = sresize(c->columns.percentages, ncols, int);
|
|
|
|
for (index = 0; index < ncols; index++)
|
|
|
|
c->columns.percentages[index] = (index+1)*100/ncols - index*100/ncols;
|
|
|
|
index = 0;
|
2017-11-26 14:05:02 +00:00
|
|
|
for (i = 0; i < buttons->nbuttons; i++) {
|
|
|
|
const struct message_box_button *button = &buttons->buttons[i];
|
|
|
|
c = ctrl_pushbutton(s0, button->title, button->shortcut,
|
|
|
|
HELPCTX(no_help), messagebox_handler,
|
|
|
|
I(button->value));
|
2003-03-31 11:21:07 +00:00
|
|
|
c->generic.column = index++;
|
2017-11-26 14:05:02 +00:00
|
|
|
if (button->type > 0)
|
2003-03-31 11:21:07 +00:00
|
|
|
c->button.isdefault = TRUE;
|
2015-09-26 12:57:12 +00:00
|
|
|
|
|
|
|
/* We always arrange that _some_ button is labelled as
|
|
|
|
* 'iscancel', so that pressing Escape will always cause
|
|
|
|
* win_key_press to do something. The button we choose is
|
|
|
|
* whichever has the smallest type value: this means that real
|
|
|
|
* cancel buttons (labelled -1) will be picked if one is
|
|
|
|
* there, or in cases where the options are yes/no (1,0) then
|
|
|
|
* no will be picked, and if there's only one option (a box
|
|
|
|
* that really is just showing a _message_ and not even asking
|
|
|
|
* a question) then that will be picked. */
|
2017-11-26 14:05:02 +00:00
|
|
|
if (button->type == min_type)
|
2003-03-31 11:21:07 +00:00
|
|
|
c->button.iscancel = TRUE;
|
|
|
|
}
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
s1 = ctrl_getset(dp->ctrlbox, "x", "", "");
|
2015-12-22 12:32:48 +00:00
|
|
|
textctrl = ctrl_text(s1, msg, HELPCTX(no_help));
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
window = our_dialog_new();
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_window_set_title(GTK_WINDOW(window), title);
|
2017-11-26 14:29:31 +00:00
|
|
|
w0 = layout_ctrls(dp, &scs, s0, GTK_WINDOW(window));
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_set_action_area(GTK_WINDOW(window), w0);
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_widget_show(w0);
|
2017-11-26 14:29:31 +00:00
|
|
|
w1 = layout_ctrls(dp, &scs, s1, GTK_WINDOW(window));
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(w1), 10);
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_widget_set_size_request(w1, minwid+20, -1);
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_add_to_content_area(GTK_WINDOW(window), w1, TRUE, TRUE, 0);
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_widget_show(w1);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
dp->shortcuts = &scs;
|
|
|
|
dp->lastfocus = NULL;
|
|
|
|
dp->retval = 0;
|
|
|
|
dp->window = window;
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2015-12-22 12:32:48 +00:00
|
|
|
if (selectable) {
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
2017-11-26 14:29:31 +00:00
|
|
|
struct uctrl *uc = dlg_find_byctrl(dp, textctrl);
|
2015-12-22 12:32:48 +00:00
|
|
|
gtk_label_set_selectable(GTK_LABEL(uc->text), TRUE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GTK selectable labels have a habit of selecting their
|
|
|
|
* entire contents when they gain focus. It's ugly to have
|
|
|
|
* text in a message box start up all selected, so we suppress
|
|
|
|
* this by manually selecting none of it - but we must do this
|
|
|
|
* when the widget _already has_ focus, otherwise our work
|
|
|
|
* will be undone when it gains it shortly.
|
|
|
|
*/
|
|
|
|
gtk_widget_grab_focus(uc->text);
|
|
|
|
gtk_label_select_region(GTK_LABEL(uc->text), 0, 0);
|
|
|
|
#else
|
|
|
|
(void)textctrl; /* placate warning */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
if (parentwin) {
|
2003-04-10 08:53:43 +00:00
|
|
|
set_transient_window_pos(parentwin, window);
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_window_set_transient_for(GTK_WINDOW(window),
|
|
|
|
GTK_WINDOW(parentwin));
|
|
|
|
} else
|
|
|
|
gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
|
2015-12-22 12:32:48 +00:00
|
|
|
gtk_container_set_focus_child(GTK_CONTAINER(window), NULL);
|
2003-03-31 11:21:07 +00:00
|
|
|
gtk_widget_show(window);
|
2015-12-22 12:32:48 +00:00
|
|
|
gtk_window_set_focus(GTK_WINDOW(window), NULL);
|
2003-03-13 19:52:28 +00:00
|
|
|
|
2017-11-26 14:29:31 +00:00
|
|
|
dp->selparams = NULL;
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(window), "destroy",
|
2017-11-26 14:29:31 +00:00
|
|
|
G_CALLBACK(dlgparam_destroy), dp);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(window), "key_press_event",
|
2017-11-26 14:29:31 +00:00
|
|
|
G_CALLBACK(win_key_press), dp);
|
|
|
|
|
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
2017-11-26 14:29:42 +00:00
|
|
|
struct verify_ssh_host_key_result_ctx {
|
|
|
|
char *host;
|
|
|
|
int port;
|
|
|
|
char *keytype;
|
|
|
|
char *keystr;
|
|
|
|
void (*callback)(void *callback_ctx, int result);
|
|
|
|
void *callback_ctx;
|
|
|
|
void *frontend;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void verify_ssh_host_key_result_callback(void *vctx, int result)
|
|
|
|
{
|
|
|
|
struct verify_ssh_host_key_result_ctx *ctx =
|
|
|
|
(struct verify_ssh_host_key_result_ctx *)vctx;
|
|
|
|
|
|
|
|
if (result >= 0) {
|
|
|
|
int logical_result;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert the dialog-box return value (one of three
|
|
|
|
* possibilities) into the return value we pass back to the SSH
|
|
|
|
* code (one of only two possibilities, because the SSH code
|
|
|
|
* doesn't care whether we saved the host key or not).
|
|
|
|
*/
|
|
|
|
if (result == 2) {
|
|
|
|
store_host_key(ctx->host, ctx->port, ctx->keytype, ctx->keystr);
|
|
|
|
logical_result = 1; /* continue with connection */
|
|
|
|
} else if (result == 1) {
|
|
|
|
logical_result = 1; /* continue with connection */
|
|
|
|
} else {
|
|
|
|
logical_result = 0; /* do not continue with connection */
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx->callback(ctx->callback_ctx, logical_result);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clean up this context structure, whether or not a result was
|
|
|
|
* ever actually delivered from the dialog box.
|
|
|
|
*/
|
2017-11-26 16:51:19 +00:00
|
|
|
unregister_dialog(ctx->frontend, DIALOG_SLOT_NETWORK_PROMPT);
|
2017-11-26 14:29:42 +00:00
|
|
|
|
|
|
|
sfree(ctx->host);
|
|
|
|
sfree(ctx->keytype);
|
|
|
|
sfree(ctx->keystr);
|
|
|
|
sfree(ctx);
|
|
|
|
}
|
|
|
|
|
2015-05-15 09:12:06 +00:00
|
|
|
int verify_ssh_host_key(void *frontend, char *host, int port,
|
|
|
|
const char *keytype, char *keystr, char *fingerprint,
|
2005-02-17 18:34:24 +00:00
|
|
|
void (*callback)(void *ctx, int result), void *ctx)
|
2003-03-31 11:21:07 +00:00
|
|
|
{
|
|
|
|
static const char absenttxt[] =
|
|
|
|
"The server's host key is not cached. You have no guarantee "
|
|
|
|
"that the server is the computer you think it is.\n"
|
2003-06-26 14:19:33 +00:00
|
|
|
"The server's %s key fingerprint is:\n"
|
2003-03-31 11:21:07 +00:00
|
|
|
"%s\n"
|
|
|
|
"If you trust this host, press \"Accept\" to add the key to "
|
|
|
|
"PuTTY's cache and carry on connecting.\n"
|
|
|
|
"If you want to carry on connecting just once, without "
|
|
|
|
"adding the key to the cache, press \"Connect Once\".\n"
|
|
|
|
"If you do not trust this host, press \"Cancel\" to abandon the "
|
|
|
|
"connection.";
|
|
|
|
static const char wrongtxt[] =
|
|
|
|
"WARNING - POTENTIAL SECURITY BREACH!\n"
|
|
|
|
"The server's host key does not match the one PuTTY has "
|
|
|
|
"cached. This means that either the server administrator "
|
|
|
|
"has changed the host key, or you have actually connected "
|
|
|
|
"to another computer pretending to be the server.\n"
|
2003-06-26 14:19:33 +00:00
|
|
|
"The new %s key fingerprint is:\n"
|
2003-03-31 11:21:07 +00:00
|
|
|
"%s\n"
|
|
|
|
"If you were expecting this change and trust the new key, "
|
|
|
|
"press \"Accept\" to update PuTTY's cache and continue connecting.\n"
|
|
|
|
"If you want to carry on connecting but without updating "
|
|
|
|
"the cache, press \"Connect Once\".\n"
|
|
|
|
"If you want to abandon the connection completely, press "
|
|
|
|
"\"Cancel\" to cancel. Pressing \"Cancel\" is the ONLY guaranteed "
|
|
|
|
"safe choice.";
|
2017-11-26 14:05:02 +00:00
|
|
|
static const struct message_box_button button_array_hostkey[] = {
|
|
|
|
{"Accept", 'a', 0, 2},
|
|
|
|
{"Connect Once", 'o', 0, 1},
|
|
|
|
{"Cancel", 'c', -1, 0},
|
|
|
|
};
|
|
|
|
static const struct message_box_buttons buttons_hostkey = {
|
|
|
|
button_array_hostkey, lenof(button_array_hostkey),
|
|
|
|
};
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
char *text;
|
|
|
|
int ret;
|
2017-11-26 14:29:42 +00:00
|
|
|
struct verify_ssh_host_key_result_ctx *result_ctx;
|
|
|
|
GtkWidget *mainwin, *msgbox;
|
2003-03-31 11:21:07 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify the key.
|
|
|
|
*/
|
|
|
|
ret = verify_host_key(host, port, keytype, keystr);
|
|
|
|
|
|
|
|
if (ret == 0) /* success - key matched OK */
|
2005-02-17 18:34:24 +00:00
|
|
|
return 1;
|
2003-03-31 11:21:07 +00:00
|
|
|
|
2003-06-26 14:19:33 +00:00
|
|
|
text = dupprintf((ret == 2 ? wrongtxt : absenttxt), keytype, fingerprint);
|
2003-03-31 11:21:07 +00:00
|
|
|
|
2017-11-26 14:29:42 +00:00
|
|
|
result_ctx = snew(struct verify_ssh_host_key_result_ctx);
|
|
|
|
result_ctx->callback = callback;
|
|
|
|
result_ctx->callback_ctx = ctx;
|
|
|
|
result_ctx->host = dupstr(host);
|
|
|
|
result_ctx->port = port;
|
|
|
|
result_ctx->keytype = dupstr(keytype);
|
|
|
|
result_ctx->keystr = dupstr(keystr);
|
|
|
|
result_ctx->frontend = frontend;
|
|
|
|
|
|
|
|
mainwin = GTK_WIDGET(get_window(frontend));
|
|
|
|
msgbox = create_message_box(
|
|
|
|
mainwin, "PuTTY Security Alert", text, string_width(fingerprint), TRUE,
|
|
|
|
&buttons_hostkey, verify_ssh_host_key_result_callback, result_ctx);
|
2017-11-26 16:51:19 +00:00
|
|
|
register_dialog(frontend, DIALOG_SLOT_NETWORK_PROMPT, msgbox);
|
2003-03-31 11:21:07 +00:00
|
|
|
|
|
|
|
sfree(text);
|
|
|
|
|
2017-11-26 14:29:42 +00:00
|
|
|
return -1; /* dialog still in progress */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
struct simple_prompt_result_ctx {
|
2017-11-26 15:13:08 +00:00
|
|
|
void (*callback)(void *callback_ctx, int result);
|
|
|
|
void *callback_ctx;
|
|
|
|
void *frontend;
|
2017-11-26 16:56:03 +00:00
|
|
|
enum DialogSlot dialog_slot;
|
2017-11-26 15:13:08 +00:00
|
|
|
};
|
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
static void simple_prompt_result_callback(void *vctx, int result)
|
2017-11-26 15:13:08 +00:00
|
|
|
{
|
2017-11-26 16:56:03 +00:00
|
|
|
struct simple_prompt_result_ctx *ctx =
|
|
|
|
(struct simple_prompt_result_ctx *)vctx;
|
2017-11-26 15:13:08 +00:00
|
|
|
|
|
|
|
if (result >= 0)
|
|
|
|
ctx->callback(ctx->callback_ctx, result);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clean up this context structure, whether or not a result was
|
|
|
|
* ever actually delivered from the dialog box.
|
|
|
|
*/
|
2017-11-26 16:56:03 +00:00
|
|
|
unregister_dialog(ctx->frontend, ctx->dialog_slot);
|
2017-11-26 15:13:08 +00:00
|
|
|
sfree(ctx);
|
|
|
|
}
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
/*
|
2004-12-23 02:24:07 +00:00
|
|
|
* Ask whether the selected algorithm is acceptable (since it was
|
2003-03-31 11:21:07 +00:00
|
|
|
* below the configured 'warn' threshold).
|
|
|
|
*/
|
2005-02-17 18:34:24 +00:00
|
|
|
int askalg(void *frontend, const char *algtype, const char *algname,
|
|
|
|
void (*callback)(void *ctx, int result), void *ctx)
|
2003-03-31 11:21:07 +00:00
|
|
|
{
|
|
|
|
static const char msg[] =
|
2004-12-23 02:24:07 +00:00
|
|
|
"The first %s supported by the server is "
|
2003-03-31 11:21:07 +00:00
|
|
|
"%s, which is below the configured warning threshold.\n"
|
|
|
|
"Continue with connection?";
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
char *text;
|
2017-11-26 16:56:03 +00:00
|
|
|
struct simple_prompt_result_ctx *result_ctx;
|
2017-11-26 15:13:08 +00:00
|
|
|
GtkWidget *mainwin, *msgbox;
|
2003-03-31 11:21:07 +00:00
|
|
|
|
2004-12-23 02:24:07 +00:00
|
|
|
text = dupprintf(msg, algtype, algname);
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
result_ctx = snew(struct simple_prompt_result_ctx);
|
2017-11-26 15:13:08 +00:00
|
|
|
result_ctx->callback = callback;
|
|
|
|
result_ctx->callback_ctx = ctx;
|
|
|
|
result_ctx->frontend = frontend;
|
2017-11-26 16:56:03 +00:00
|
|
|
result_ctx->dialog_slot = DIALOG_SLOT_NETWORK_PROMPT;
|
2017-11-26 15:13:08 +00:00
|
|
|
|
|
|
|
mainwin = GTK_WIDGET(get_window(frontend));
|
|
|
|
msgbox = create_message_box(
|
|
|
|
mainwin, "PuTTY Security Alert", text,
|
|
|
|
string_width("Reasonably long line of text as a width template"),
|
2017-11-26 16:56:03 +00:00
|
|
|
FALSE, &buttons_yn, simple_prompt_result_callback, result_ctx);
|
|
|
|
register_dialog(frontend, result_ctx->dialog_slot, msgbox);
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2016-03-27 17:08:49 +00:00
|
|
|
sfree(text);
|
|
|
|
|
2017-11-26 15:13:08 +00:00
|
|
|
return -1; /* dialog still in progress */
|
2016-03-27 17:08:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int askhk(void *frontend, const char *algname, const char *betteralgs,
|
|
|
|
void (*callback)(void *ctx, int result), void *ctx)
|
|
|
|
{
|
|
|
|
static const char msg[] =
|
|
|
|
"The first host key type we have stored for this server\n"
|
|
|
|
"is %s, which is below the configured warning threshold.\n"
|
|
|
|
"The server also provides the following types of host key\n"
|
|
|
|
"above the threshold, which we do not have stored:\n"
|
|
|
|
"%s\n"
|
|
|
|
"Continue with connection?";
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2016-03-27 17:08:49 +00:00
|
|
|
char *text;
|
2017-11-26 16:56:03 +00:00
|
|
|
struct simple_prompt_result_ctx *result_ctx;
|
2017-11-26 15:13:08 +00:00
|
|
|
GtkWidget *mainwin, *msgbox;
|
2016-03-27 17:08:49 +00:00
|
|
|
|
|
|
|
text = dupprintf(msg, algname, betteralgs);
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
result_ctx = snew(struct simple_prompt_result_ctx);
|
2017-11-26 15:13:08 +00:00
|
|
|
result_ctx->callback = callback;
|
|
|
|
result_ctx->callback_ctx = ctx;
|
|
|
|
result_ctx->frontend = frontend;
|
2017-11-26 16:56:03 +00:00
|
|
|
result_ctx->dialog_slot = DIALOG_SLOT_NETWORK_PROMPT;
|
2017-11-26 15:13:08 +00:00
|
|
|
|
|
|
|
mainwin = GTK_WIDGET(get_window(frontend));
|
|
|
|
msgbox = create_message_box(
|
|
|
|
mainwin, "PuTTY Security Alert", text,
|
|
|
|
string_width("is ecdsa-nistp521, which is below the configured"
|
|
|
|
" warning threshold."),
|
2017-11-26 16:56:03 +00:00
|
|
|
FALSE, &buttons_yn, simple_prompt_result_callback, result_ctx);
|
|
|
|
register_dialog(frontend, result_ctx->dialog_slot, msgbox);
|
2017-11-26 15:13:08 +00:00
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
sfree(text);
|
|
|
|
|
2017-11-26 15:13:08 +00:00
|
|
|
return -1; /* dialog still in progress */
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2003-03-31 11:21:07 +00:00
|
|
|
void old_keyfile_warning(void)
|
2003-03-13 19:52:28 +00:00
|
|
|
{
|
2003-03-31 11:21:07 +00:00
|
|
|
/*
|
|
|
|
* This should never happen on Unix. We hope.
|
|
|
|
*/
|
2003-03-13 19:52:28 +00:00
|
|
|
}
|
|
|
|
|
2015-05-15 10:15:42 +00:00
|
|
|
void nonfatal_message_box(void *window, const char *msg)
|
2013-07-19 17:44:28 +00:00
|
|
|
{
|
2017-11-26 17:55:00 +00:00
|
|
|
char *title = dupcat(appname, " Error", NULL);
|
|
|
|
create_message_box(
|
|
|
|
window, title, msg,
|
|
|
|
string_width("REASONABLY LONG LINE OF TEXT FOR BASIC SANITY"),
|
|
|
|
FALSE, &buttons_ok, trivial_post_dialog_fn, NULL);
|
|
|
|
sfree(title);
|
2013-07-19 17:44:28 +00:00
|
|
|
}
|
|
|
|
|
2015-05-15 10:15:42 +00:00
|
|
|
void nonfatal(const char *p, ...)
|
2013-07-19 17:44:28 +00:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
char *msg;
|
|
|
|
va_start(ap, p);
|
|
|
|
msg = dupvprintf(p, ap);
|
|
|
|
va_end(ap);
|
2013-09-23 14:35:08 +00:00
|
|
|
nonfatal_message_box(NULL, msg);
|
2013-07-19 17:44:28 +00:00
|
|
|
sfree(msg);
|
|
|
|
}
|
|
|
|
|
2003-04-05 16:05:00 +00:00
|
|
|
static GtkWidget *aboutbox = NULL;
|
|
|
|
|
|
|
|
static void about_close_clicked(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy(aboutbox);
|
|
|
|
aboutbox = NULL;
|
|
|
|
}
|
|
|
|
|
2015-09-26 13:09:27 +00:00
|
|
|
static void about_key_press(GtkWidget *widget, GdkEventKey *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
if (event->keyval == GDK_KEY_Escape && aboutbox) {
|
|
|
|
gtk_widget_destroy(aboutbox);
|
|
|
|
aboutbox = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-04-05 16:05:00 +00:00
|
|
|
static void licence_clicked(GtkButton *button, gpointer data)
|
|
|
|
{
|
|
|
|
char *title;
|
|
|
|
|
|
|
|
title = dupcat(appname, " Licence", NULL);
|
|
|
|
assert(aboutbox != NULL);
|
2017-11-26 17:32:01 +00:00
|
|
|
create_message_box(aboutbox, title, LICENCE_TEXT("\n\n"),
|
|
|
|
string_width("LONGISH LINE OF TEXT SO THE LICENCE"
|
|
|
|
" BOX ISN'T EXCESSIVELY TALL AND THIN"),
|
|
|
|
TRUE, &buttons_ok, trivial_post_dialog_fn, NULL);
|
2003-04-05 16:05:00 +00:00
|
|
|
sfree(title);
|
|
|
|
}
|
|
|
|
|
2003-04-10 08:53:43 +00:00
|
|
|
void about_box(void *window)
|
2003-04-05 16:05:00 +00:00
|
|
|
{
|
|
|
|
GtkWidget *w;
|
2015-08-31 14:45:18 +00:00
|
|
|
GtkBox *action_area;
|
2003-04-05 16:05:00 +00:00
|
|
|
char *title;
|
|
|
|
|
|
|
|
if (aboutbox) {
|
|
|
|
gtk_widget_grab_focus(aboutbox);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
aboutbox = our_dialog_new();
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(aboutbox), 10);
|
|
|
|
title = dupcat("About ", appname, NULL);
|
|
|
|
gtk_window_set_title(GTK_WINDOW(aboutbox), title);
|
|
|
|
sfree(title);
|
|
|
|
|
|
|
|
w = gtk_button_new_with_label("Close");
|
2015-08-08 14:53:43 +00:00
|
|
|
gtk_widget_set_can_default(w, TRUE);
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_window_set_default(GTK_WINDOW(aboutbox), w);
|
2015-08-31 14:45:18 +00:00
|
|
|
action_area = our_dialog_make_action_hbox(GTK_WINDOW(aboutbox));
|
|
|
|
gtk_box_pack_end(action_area, w, FALSE, FALSE, 0);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "clicked",
|
|
|
|
G_CALLBACK(about_close_clicked), NULL);
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_widget_show(w);
|
|
|
|
|
|
|
|
w = gtk_button_new_with_label("View Licence");
|
2015-08-08 14:53:43 +00:00
|
|
|
gtk_widget_set_can_default(w, TRUE);
|
2015-08-31 14:45:18 +00:00
|
|
|
gtk_box_pack_end(action_area, w, FALSE, FALSE, 0);
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(w), "clicked",
|
|
|
|
G_CALLBACK(licence_clicked), NULL);
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_widget_show(w);
|
|
|
|
|
2015-12-22 12:32:48 +00:00
|
|
|
{
|
2017-01-21 14:55:53 +00:00
|
|
|
char *buildinfo_text = buildinfo("\n");
|
2015-12-22 12:32:48 +00:00
|
|
|
char *label_text = dupprintf
|
2017-01-21 14:55:53 +00:00
|
|
|
("%s\n\n%s\n\n%s\n\n%s",
|
|
|
|
appname, ver, buildinfo_text,
|
2015-12-22 12:43:31 +00:00
|
|
|
"Copyright " SHORT_COPYRIGHT_DETAILS ". All rights reserved");
|
2015-12-22 12:32:48 +00:00
|
|
|
w = gtk_label_new(label_text);
|
|
|
|
gtk_label_set_justify(GTK_LABEL(w), GTK_JUSTIFY_CENTER);
|
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
gtk_label_set_selectable(GTK_LABEL(w), TRUE);
|
|
|
|
#endif
|
|
|
|
sfree(label_text);
|
|
|
|
}
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_add_to_content_area(GTK_WINDOW(aboutbox), w, FALSE, FALSE, 0);
|
2015-12-22 12:32:48 +00:00
|
|
|
#if GTK_CHECK_VERSION(2,0,0)
|
|
|
|
/*
|
2017-11-26 20:01:16 +00:00
|
|
|
* Same precautions against initial select-all as in
|
|
|
|
* create_message_box().
|
2015-12-22 12:32:48 +00:00
|
|
|
*/
|
|
|
|
gtk_widget_grab_focus(w);
|
|
|
|
gtk_label_select_region(GTK_LABEL(w), 0, 0);
|
|
|
|
#endif
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_widget_show(w);
|
|
|
|
|
2015-09-26 13:09:27 +00:00
|
|
|
g_signal_connect(G_OBJECT(aboutbox), "key_press_event",
|
|
|
|
G_CALLBACK(about_key_press), NULL);
|
|
|
|
|
2003-04-10 08:53:43 +00:00
|
|
|
set_transient_window_pos(GTK_WIDGET(window), aboutbox);
|
2017-12-18 11:46:48 +00:00
|
|
|
if (window)
|
|
|
|
gtk_window_set_transient_for(GTK_WINDOW(aboutbox),
|
|
|
|
GTK_WINDOW(window));
|
2015-12-22 12:32:48 +00:00
|
|
|
gtk_container_set_focus_child(GTK_CONTAINER(aboutbox), NULL);
|
2003-04-05 16:05:00 +00:00
|
|
|
gtk_widget_show(aboutbox);
|
2015-12-22 12:32:48 +00:00
|
|
|
gtk_window_set_focus(GTK_WINDOW(aboutbox), NULL);
|
2003-04-05 16:05:00 +00:00
|
|
|
}
|
2003-04-09 18:46:45 +00:00
|
|
|
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
#define LOGEVENT_INITIAL_MAX 128
|
|
|
|
#define LOGEVENT_CIRCULAR_MAX 128
|
|
|
|
|
2003-04-09 18:46:45 +00:00
|
|
|
struct eventlog_stuff {
|
|
|
|
GtkWidget *parentwin, *window;
|
|
|
|
struct controlbox *eventbox;
|
|
|
|
struct Shortcuts scs;
|
|
|
|
struct dlgparam dp;
|
|
|
|
union control *listctrl;
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
char **events_initial;
|
|
|
|
char **events_circular;
|
|
|
|
int ninitial, ncircular, circular_first;
|
2003-04-11 17:40:52 +00:00
|
|
|
char *seldata;
|
|
|
|
int sellen;
|
|
|
|
int ignore_selchange;
|
2003-04-09 18:46:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void eventlog_destroy(GtkWidget *widget, gpointer data)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)data;
|
|
|
|
|
|
|
|
es->window = NULL;
|
2003-04-11 17:40:52 +00:00
|
|
|
sfree(es->seldata);
|
2014-11-08 22:22:34 +00:00
|
|
|
es->seldata = NULL;
|
2003-04-09 18:46:45 +00:00
|
|
|
dlg_cleanup(&es->dp);
|
|
|
|
ctrl_free_box(es->eventbox);
|
|
|
|
}
|
|
|
|
static void eventlog_ok_handler(union control *ctrl, void *dlg,
|
|
|
|
void *data, int event)
|
|
|
|
{
|
|
|
|
if (event == EVENT_ACTION)
|
|
|
|
dlg_end(dlg, 0);
|
|
|
|
}
|
|
|
|
static void eventlog_list_handler(union control *ctrl, void *dlg,
|
|
|
|
void *data, int event)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)data;
|
|
|
|
|
|
|
|
if (event == EVENT_REFRESH) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
dlg_update_start(ctrl, dlg);
|
|
|
|
dlg_listbox_clear(ctrl, dlg);
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
for (i = 0; i < es->ninitial; i++) {
|
|
|
|
dlg_listbox_add(ctrl, dlg, es->events_initial[i]);
|
|
|
|
}
|
|
|
|
for (i = 0; i < es->ncircular; i++) {
|
|
|
|
dlg_listbox_add(ctrl, dlg, es->events_circular[(es->circular_first + i) % LOGEVENT_CIRCULAR_MAX]);
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
dlg_update_done(ctrl, dlg);
|
2003-04-11 17:40:52 +00:00
|
|
|
} else if (event == EVENT_SELCHANGE) {
|
|
|
|
int i;
|
|
|
|
int selsize = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If this SELCHANGE event is happening as a result of
|
|
|
|
* deliberate deselection because someone else has grabbed
|
|
|
|
* the selection, the last thing we want to do is pre-empt
|
|
|
|
* them.
|
|
|
|
*/
|
|
|
|
if (es->ignore_selchange)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Construct the data to use as the selection.
|
|
|
|
*/
|
|
|
|
sfree(es->seldata);
|
|
|
|
es->seldata = NULL;
|
|
|
|
es->sellen = 0;
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
for (i = 0; i < es->ninitial; i++) {
|
2003-04-11 17:40:52 +00:00
|
|
|
if (dlg_listbox_issel(ctrl, dlg, i)) {
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
int extralen = strlen(es->events_initial[i]);
|
|
|
|
|
|
|
|
if (es->sellen + extralen + 2 > selsize) {
|
|
|
|
selsize = es->sellen + extralen + 512;
|
|
|
|
es->seldata = sresize(es->seldata, selsize, char);
|
|
|
|
}
|
|
|
|
|
|
|
|
strcpy(es->seldata + es->sellen, es->events_initial[i]);
|
|
|
|
es->sellen += extralen;
|
|
|
|
es->seldata[es->sellen++] = '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; i < es->ncircular; i++) {
|
|
|
|
if (dlg_listbox_issel(ctrl, dlg, es->ninitial + i)) {
|
|
|
|
int j = (es->circular_first + i) % LOGEVENT_CIRCULAR_MAX;
|
|
|
|
int extralen = strlen(es->events_circular[j]);
|
2003-04-11 17:40:52 +00:00
|
|
|
|
|
|
|
if (es->sellen + extralen + 2 > selsize) {
|
|
|
|
selsize = es->sellen + extralen + 512;
|
|
|
|
es->seldata = sresize(es->seldata, selsize, char);
|
|
|
|
}
|
|
|
|
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
strcpy(es->seldata + es->sellen, es->events_circular[j]);
|
2003-04-11 17:40:52 +00:00
|
|
|
es->sellen += extralen;
|
|
|
|
es->seldata[es->sellen++] = '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gtk_selection_owner_set(es->window, GDK_SELECTION_PRIMARY,
|
|
|
|
GDK_CURRENT_TIME)) {
|
|
|
|
extern GdkAtom compound_text_atom;
|
|
|
|
|
|
|
|
gtk_selection_add_target(es->window, GDK_SELECTION_PRIMARY,
|
|
|
|
GDK_SELECTION_TYPE_STRING, 1);
|
|
|
|
gtk_selection_add_target(es->window, GDK_SELECTION_PRIMARY,
|
|
|
|
compound_text_atom, 1);
|
|
|
|
}
|
|
|
|
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
}
|
2003-04-11 17:40:52 +00:00
|
|
|
|
|
|
|
void eventlog_selection_get(GtkWidget *widget, GtkSelectionData *seldata,
|
|
|
|
guint info, guint time_stamp, gpointer data)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)data;
|
|
|
|
|
2015-08-08 14:53:43 +00:00
|
|
|
gtk_selection_data_set(seldata, gtk_selection_data_get_target(seldata), 8,
|
2006-12-30 23:00:14 +00:00
|
|
|
(unsigned char *)es->seldata, es->sellen);
|
2003-04-11 17:40:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gint eventlog_selection_clear(GtkWidget *widget, GdkEventSelection *seldata,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)data;
|
|
|
|
struct uctrl *uc;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Deselect everything in the list box.
|
|
|
|
*/
|
|
|
|
uc = dlg_find_byctrl(&es->dp, es->listctrl);
|
2008-04-04 10:56:26 +00:00
|
|
|
es->ignore_selchange = 1;
|
|
|
|
#if !GTK_CHECK_VERSION(2,0,0)
|
|
|
|
assert(uc->list);
|
|
|
|
gtk_list_unselect_all(GTK_LIST(uc->list));
|
|
|
|
#else
|
2008-04-04 10:16:24 +00:00
|
|
|
assert(uc->treeview);
|
2008-04-02 14:48:06 +00:00
|
|
|
gtk_tree_selection_unselect_all
|
2008-04-04 10:16:24 +00:00
|
|
|
(gtk_tree_view_get_selection(GTK_TREE_VIEW(uc->treeview)));
|
2008-04-04 10:56:26 +00:00
|
|
|
#endif
|
|
|
|
es->ignore_selchange = 0;
|
2003-04-11 17:40:52 +00:00
|
|
|
|
|
|
|
sfree(es->seldata);
|
|
|
|
es->sellen = 0;
|
|
|
|
es->seldata = NULL;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2003-04-09 18:46:45 +00:00
|
|
|
void showeventlog(void *estuff, void *parentwin)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)estuff;
|
|
|
|
GtkWidget *window, *w0, *w1;
|
|
|
|
GtkWidget *parent = GTK_WIDGET(parentwin);
|
|
|
|
struct controlset *s0, *s1;
|
|
|
|
union control *c;
|
2008-04-02 14:48:06 +00:00
|
|
|
int index;
|
2003-04-09 18:46:45 +00:00
|
|
|
char *title;
|
|
|
|
|
|
|
|
if (es->window) {
|
|
|
|
gtk_widget_grab_focus(es->window);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dlg_init(&es->dp);
|
|
|
|
|
|
|
|
for (index = 0; index < lenof(es->scs.sc); index++) {
|
|
|
|
es->scs.sc[index].action = SHORTCUT_EMPTY;
|
|
|
|
}
|
|
|
|
|
|
|
|
es->eventbox = ctrl_new_box();
|
|
|
|
|
|
|
|
s0 = ctrl_getset(es->eventbox, "", "", "");
|
|
|
|
ctrl_columns(s0, 3, 33, 34, 33);
|
|
|
|
c = ctrl_pushbutton(s0, "Close", 'c', HELPCTX(no_help),
|
|
|
|
eventlog_ok_handler, P(NULL));
|
|
|
|
c->button.column = 1;
|
|
|
|
c->button.isdefault = TRUE;
|
|
|
|
|
|
|
|
s1 = ctrl_getset(es->eventbox, "x", "", "");
|
|
|
|
es->listctrl = c = ctrl_listbox(s1, NULL, NO_SHORTCUT, HELPCTX(no_help),
|
|
|
|
eventlog_list_handler, P(es));
|
|
|
|
c->listbox.height = 10;
|
|
|
|
c->listbox.multisel = 2;
|
|
|
|
c->listbox.ncols = 3;
|
|
|
|
c->listbox.percentages = snewn(3, int);
|
|
|
|
c->listbox.percentages[0] = 25;
|
|
|
|
c->listbox.percentages[1] = 10;
|
|
|
|
c->listbox.percentages[2] = 65;
|
|
|
|
|
2015-08-31 14:45:18 +00:00
|
|
|
es->window = window = our_dialog_new();
|
2003-04-09 18:46:45 +00:00
|
|
|
title = dupcat(appname, " Event Log", NULL);
|
|
|
|
gtk_window_set_title(GTK_WINDOW(window), title);
|
|
|
|
sfree(title);
|
2008-04-02 14:48:06 +00:00
|
|
|
w0 = layout_ctrls(&es->dp, &es->scs, s0, GTK_WINDOW(window));
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_set_action_area(GTK_WINDOW(window), w0);
|
2003-04-09 18:46:45 +00:00
|
|
|
gtk_widget_show(w0);
|
2008-04-02 14:48:06 +00:00
|
|
|
w1 = layout_ctrls(&es->dp, &es->scs, s1, GTK_WINDOW(window));
|
2003-04-09 18:46:45 +00:00
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(w1), 10);
|
2015-08-08 17:02:01 +00:00
|
|
|
gtk_widget_set_size_request(w1, 20 + string_width
|
|
|
|
("LINE OF TEXT GIVING WIDTH OF EVENT LOG IS "
|
|
|
|
"QUITE LONG 'COS SSH LOG ENTRIES ARE WIDE"),
|
|
|
|
-1);
|
2015-08-31 14:45:18 +00:00
|
|
|
our_dialog_add_to_content_area(GTK_WINDOW(window), w1, TRUE, TRUE, 0);
|
2003-04-09 18:46:45 +00:00
|
|
|
gtk_widget_show(w1);
|
|
|
|
|
|
|
|
es->dp.data = es;
|
|
|
|
es->dp.shortcuts = &es->scs;
|
|
|
|
es->dp.lastfocus = NULL;
|
|
|
|
es->dp.retval = 0;
|
|
|
|
es->dp.window = window;
|
|
|
|
|
|
|
|
dlg_refresh(NULL, &es->dp);
|
|
|
|
|
|
|
|
if (parent) {
|
2003-04-10 08:53:43 +00:00
|
|
|
set_transient_window_pos(parent, window);
|
2003-04-09 18:46:45 +00:00
|
|
|
gtk_window_set_transient_for(GTK_WINDOW(window),
|
|
|
|
GTK_WINDOW(parent));
|
|
|
|
} else
|
|
|
|
gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
|
|
|
|
gtk_widget_show(window);
|
|
|
|
|
2015-08-08 16:29:02 +00:00
|
|
|
g_signal_connect(G_OBJECT(window), "destroy",
|
|
|
|
G_CALLBACK(eventlog_destroy), es);
|
|
|
|
g_signal_connect(G_OBJECT(window), "key_press_event",
|
|
|
|
G_CALLBACK(win_key_press), &es->dp);
|
|
|
|
g_signal_connect(G_OBJECT(window), "selection_get",
|
|
|
|
G_CALLBACK(eventlog_selection_get), es);
|
|
|
|
g_signal_connect(G_OBJECT(window), "selection_clear_event",
|
|
|
|
G_CALLBACK(eventlog_selection_clear), es);
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void *eventlogstuff_new(void)
|
|
|
|
{
|
|
|
|
struct eventlog_stuff *es;
|
|
|
|
es = snew(struct eventlog_stuff);
|
|
|
|
memset(es, 0, sizeof(*es));
|
|
|
|
return es;
|
|
|
|
}
|
|
|
|
|
2003-05-04 14:18:18 +00:00
|
|
|
void logevent_dlg(void *estuff, const char *string)
|
2003-04-09 18:46:45 +00:00
|
|
|
{
|
|
|
|
struct eventlog_stuff *es = (struct eventlog_stuff *)estuff;
|
|
|
|
char timebuf[40];
|
2005-01-09 14:27:48 +00:00
|
|
|
struct tm tm;
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
char **location;
|
|
|
|
size_t i;
|
2003-04-09 18:46:45 +00:00
|
|
|
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
if (es->ninitial == 0) {
|
|
|
|
es->events_initial = sresize(es->events_initial, LOGEVENT_INITIAL_MAX, char *);
|
|
|
|
for (i = 0; i < LOGEVENT_INITIAL_MAX; i++)
|
|
|
|
es->events_initial[i] = NULL;
|
|
|
|
es->events_circular = sresize(es->events_circular, LOGEVENT_CIRCULAR_MAX, char *);
|
|
|
|
for (i = 0; i < LOGEVENT_CIRCULAR_MAX; i++)
|
|
|
|
es->events_circular[i] = NULL;
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
if (es->ninitial < LOGEVENT_INITIAL_MAX)
|
|
|
|
location = &es->events_initial[es->ninitial];
|
|
|
|
else
|
|
|
|
location = &es->events_circular[(es->circular_first + es->ncircular) % LOGEVENT_CIRCULAR_MAX];
|
|
|
|
|
2005-01-09 14:27:48 +00:00
|
|
|
tm=ltime();
|
|
|
|
strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S\t", &tm);
|
2003-04-09 18:46:45 +00:00
|
|
|
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
sfree(*location);
|
|
|
|
*location = dupcat(timebuf, string, NULL);
|
2003-04-09 18:46:45 +00:00
|
|
|
if (es->window) {
|
Don't grow logevent buf indefinitely
The PuTTY GUIs (Unix and Windows) maintain an in-memory event log
for display to users as they request. This uses ints for tracking
eventlog size, which is subject to memory exhaustion and (given
enough heap space) overflow attacks by servers (via, e.g., constant
rekeying).
Also a bounded log is more user-friendly. It is rare to want more
than the initial logging and the logging from a few recent rekey
events.
The Windows fix has been tested using Dr. Memory as a valgrind
substitute. No errors corresponding to the affected code showed up.
The Dr. Memory results.txt was split into a file per-error and then
grep Error $(grep -l windlg *)|cut -d: -f3-|sort |uniq -c
was used to compare. Differences arose from different usage of the GUI,
but no error could be traced to the code modified in this commit.
The Unix fix has been tested using valgrind. We don't destroy the
eventlog_stuff eventlog arrays, so we can't be entirely sure that we
don't leak more than we did before, but from code inspection it looks
like we don't (and anyways, if we leaked as much as before, just without
the integer overflow, well, that's still an improvement).
2013-07-22 23:09:04 +00:00
|
|
|
dlg_listbox_add(es->listctrl, &es->dp, *location);
|
|
|
|
}
|
|
|
|
if (es->ninitial < LOGEVENT_INITIAL_MAX) {
|
|
|
|
es->ninitial++;
|
|
|
|
} else if (es->ncircular < LOGEVENT_CIRCULAR_MAX) {
|
|
|
|
es->ncircular++;
|
|
|
|
} else if (es->ncircular == LOGEVENT_CIRCULAR_MAX) {
|
|
|
|
es->circular_first = (es->circular_first + 1) % LOGEVENT_CIRCULAR_MAX;
|
|
|
|
sfree(es->events_circular[es->circular_first]);
|
|
|
|
es->events_circular[es->circular_first] = dupstr("..");
|
2003-04-09 18:46:45 +00:00
|
|
|
}
|
|
|
|
}
|
2003-04-11 18:44:05 +00:00
|
|
|
|
2011-10-02 11:01:57 +00:00
|
|
|
int askappend(void *frontend, Filename *filename,
|
2005-02-18 18:33:31 +00:00
|
|
|
void (*callback)(void *ctx, int result), void *ctx)
|
2003-04-11 18:44:05 +00:00
|
|
|
{
|
|
|
|
static const char msgtemplate[] =
|
|
|
|
"The session log file \"%.*s\" already exists. "
|
|
|
|
"You can overwrite it with a new session log, "
|
|
|
|
"append your session log to the end of it, "
|
|
|
|
"or disable session logging for this session.";
|
2017-11-26 14:05:02 +00:00
|
|
|
static const struct message_box_button button_array_append[] = {
|
|
|
|
{"Overwrite", 'o', 1, 2},
|
|
|
|
{"Append", 'a', 0, 1},
|
|
|
|
{"Disable", 'd', -1, 0},
|
|
|
|
};
|
|
|
|
static const struct message_box_buttons buttons_append = {
|
|
|
|
button_array_append, lenof(button_array_append),
|
|
|
|
};
|
|
|
|
|
2003-04-11 18:44:05 +00:00
|
|
|
char *message;
|
|
|
|
char *mbtitle;
|
2017-11-26 16:56:03 +00:00
|
|
|
struct simple_prompt_result_ctx *result_ctx;
|
|
|
|
GtkWidget *mainwin, *msgbox;
|
2003-04-11 18:44:05 +00:00
|
|
|
|
2011-10-02 11:01:57 +00:00
|
|
|
message = dupprintf(msgtemplate, FILENAME_MAX, filename->path);
|
2003-04-11 18:44:05 +00:00
|
|
|
mbtitle = dupprintf("%s Log to File", appname);
|
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
result_ctx = snew(struct simple_prompt_result_ctx);
|
|
|
|
result_ctx->callback = callback;
|
|
|
|
result_ctx->callback_ctx = ctx;
|
|
|
|
result_ctx->frontend = frontend;
|
|
|
|
result_ctx->dialog_slot = DIALOG_SLOT_LOGFILE_PROMPT;
|
|
|
|
|
|
|
|
mainwin = GTK_WIDGET(get_window(frontend));
|
|
|
|
msgbox = create_message_box(
|
|
|
|
mainwin, mbtitle, message,
|
|
|
|
string_width("LINE OF TEXT SUITABLE FOR THE ASKAPPEND WIDTH"),
|
|
|
|
FALSE, &buttons_append, simple_prompt_result_callback, result_ctx);
|
|
|
|
register_dialog(frontend, result_ctx->dialog_slot, msgbox);
|
2003-04-11 18:44:05 +00:00
|
|
|
|
|
|
|
sfree(message);
|
|
|
|
sfree(mbtitle);
|
|
|
|
|
2017-11-26 16:56:03 +00:00
|
|
|
return -1; /* dialog still in progress */
|
2003-04-11 18:44:05 +00:00
|
|
|
}
|