]> jfr.im git - irc/hexchat/hexchat.git/commitdiff
Change preferences sub-dialogs to be modal
authorPatrick Griffis <redacted>
Sat, 16 Apr 2022 23:41:34 +0000 (18:41 -0500)
committerPatrick Griffis <redacted>
Sat, 16 Apr 2022 23:41:34 +0000 (18:41 -0500)
This solves the issue where the parent dialog is closed and then
the child dialog is used.

This is however only a partial fix:

- Many other dialogs throughout the codebase do not currently have
  parent windows and need to be refactored.

- Not all window managers respect modal so users can still trigger
  bugs. We can be more defensive against this but it requires more
  refactoring.

Closes #2686

12 files changed:
src/common/fe.h
src/fe-gtk/chanlist.c
src/fe-gtk/dccgui.c
src/fe-gtk/fe-gtk.c
src/fe-gtk/gtkutil.c
src/fe-gtk/gtkutil.h
src/fe-gtk/menu.c
src/fe-gtk/plugingui.c
src/fe-gtk/rawlog.c
src/fe-gtk/setup.c
src/fe-gtk/textgui.c
src/fe-gtk/urlgrab.c

index 9da4e2309fda0c5f0cfa404a165a6653c5f18419..b8a6279e055969ae81011215d6513b9b079f02e2 100644 (file)
@@ -141,6 +141,7 @@ void fe_get_int (char *prompt, int def, void *callback, void *ud);
 #define FRF_NOASKOVERWRITE 32  /* don't ask to overwrite existing files */
 #define FRF_EXTENSIONS 64              /* specify file extensions to be displayed */
 #define FRF_MIMETYPES 128              /* specify file mimetypes to be displayed */
+#define FRF_MODAL 256           /* dialog should be modal to parent */
 void fe_get_file (const char *title, char *initial,
                                 void (*callback) (void *userdata, char *file), void *userdata,
                                 int flags);
index aeddc41734dbd04226d7aa44217c43c44842b8b5..abf62843a0e981bff54a66564cd00ef369c88204 100644 (file)
@@ -512,7 +512,7 @@ chanlist_save (GtkWidget * wid, server *serv)
        GtkTreeModel *model = GET_MODEL (serv);
 
        if (gtk_tree_model_get_iter_first (model, &iter))
-               gtkutil_file_req (_("Select an output filename"), chanlist_filereq_done,
+               gtkutil_file_req (NULL, _("Select an output filename"), chanlist_filereq_done,
                                                                serv, NULL, NULL, FRF_WRITE);
 }
 
index 5b8dbac9a44ddc9a792ee65fd81b741f5aadd9a2..728698e3a3594ac8f04cf47a1a07966012a4b5c3 100644 (file)
@@ -146,7 +146,7 @@ fe_dcc_send_filereq (struct session *sess, char *nick, int maxcps, int passive)
        mdc->maxcps = maxcps;
        mdc->passive = passive;
 
-       gtkutil_file_req (tbuf, dcc_send_filereq_file, mdc, prefs.hex_dcc_dir, NULL, FRF_MULTIPLE|FRF_FILTERISINITIAL);
+       gtkutil_file_req (NULL, tbuf, dcc_send_filereq_file, mdc, prefs.hex_dcc_dir, NULL, FRF_MULTIPLE|FRF_FILTERISINITIAL);
 
        g_free (tbuf);
 }
index 7eca0710410cba3d19c1177ff9a5f067affd27fd..38e6172d33bf20184507e8f0289e5f2c048132d7 100644 (file)
@@ -903,7 +903,7 @@ fe_confirm (const char *message, void (*yesproc)(void *), void (*noproc)(void *)
        if (dcc->file)
        {
                char *filepath = g_build_filename (prefs.hex_dcc_dir, dcc->file, NULL);
-               gtkutil_file_req (message, dcc_saveas_cb, ud, filepath, NULL,
+               gtkutil_file_req (NULL, message, dcc_saveas_cb, ud, filepath, NULL,
                                                                FRF_WRITE|FRF_NOASKOVERWRITE|FRF_FILTERISINITIAL);
                g_free (filepath);
        }
@@ -1216,7 +1216,7 @@ fe_get_file (const char *title, char *initial,
 {
        /* OK: Call callback once per file, then once more with file=NULL. */
        /* CANCEL: Call callback once with file=NULL. */
-       gtkutil_file_req (title, callback, userdata, initial, NULL, flags | FRF_FILTERISINITIAL);
+       gtkutil_file_req (NULL, title, callback, userdata, initial, NULL, flags | FRF_FILTERISINITIAL);
 }
 
 void
index 674ad4fcb2aec5b4afe18c784546465ec6d2fdb7..98a971f92fccac3ea76092e7e9adca9f62924ce7 100644 (file)
@@ -190,7 +190,7 @@ gtkutil_file_req_response (GtkWidget *dialog, gint res, struct file_req *freq)
 }
 
 void
-gtkutil_file_req (const char *title, void *callback, void *userdata, char *filter, char *extensions,
+gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *userdata, char *filter, char *extensions,
                                                int flags)
 {
        struct file_req *freq;
@@ -269,6 +269,16 @@ gtkutil_file_req (const char *title, void *callback, void *userdata, char *filte
                                                        G_CALLBACK (gtkutil_file_req_response), freq);
        g_signal_connect (G_OBJECT (dialog), "destroy",
                                                   G_CALLBACK (gtkutil_file_req_destroy), (gpointer) freq);
+
+       if (parent)
+               gtk_window_set_transient_for (GTK_WINDOW (dialog), parent);
+
+       if (flags & FRF_MODAL)
+       {
+               g_assert (parent);
+               gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
+       }
+
        gtk_widget_show (dialog);
 }
 
index 0aa364397feff508978c1c7b6785800a47527950..c6e380e9283252480a7b0788e9a0c02444757e8c 100644 (file)
@@ -25,7 +25,7 @@
 
 typedef void (*filereqcallback) (void *, char *file);
 
-void gtkutil_file_req (const char *title, void *callback, void *userdata, char *filter, char *extensions, int flags);
+void gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *userdata, char *filter, char *extensions, int flags);
 void gtkutil_destroy (GtkWidget * igad, GtkWidget * dgad);
 void gtkutil_destroy_on_esc (GtkWidget *win);
 GtkWidget *gtkutil_button (GtkWidget *box, char *stock, char *tip, void *callback,
index 233715e54fe541c3393257c85d4e9ebc986adfad..76bc3906c7afce4ad059ec81abd2977b8189b61b 100644 (file)
@@ -1362,7 +1362,7 @@ savebuffer_req_done (session *sess, char *file)
 static void
 menu_savebuffer (GtkWidget * wid, gpointer none)
 {
-       gtkutil_file_req (_("Select an output filename"), savebuffer_req_done,
+       gtkutil_file_req (NULL, _("Select an output filename"), savebuffer_req_done,
                                                        current_sess, NULL, NULL, FRF_WRITE);
 }
 
index 83bb745fa1087f2411b0ffadcb22c3bcf2f61b40..c40ac304ab53e74962c06012aca9d7bea5fdd6e4 100644 (file)
@@ -161,7 +161,7 @@ plugingui_load (void)
 {
        char *sub_dir = g_build_filename (get_xdir(), "addons", NULL);
 
-       gtkutil_file_req (_("Select a Plugin or Script to load"), plugingui_load_cb, current_sess,
+       gtkutil_file_req (NULL, _("Select a Plugin or Script to load"), plugingui_load_cb, current_sess,
                                                        sub_dir, "*."PLUGIN_SUFFIX";*.lua;*.pl;*.py;*.tcl;*.js", FRF_FILTERISINITIAL|FRF_EXTENSIONS);
 
        g_free (sub_dir);
index 52a7726713bdc0c6fa12d51ba4fa6489bd78807b..666059c69671879456ab6623fa51758525ce9975 100644 (file)
@@ -77,7 +77,7 @@ rawlog_clearbutton (GtkWidget * wid, server *serv)
 static int
 rawlog_savebutton (GtkWidget * wid, server *serv)
 {
-       gtkutil_file_req (_("Save As..."), rawlog_save, serv, NULL, NULL, FRF_WRITE);
+       gtkutil_file_req (NULL, _("Save As..."), rawlog_save, serv, NULL, NULL, FRF_WRITE);
        return FALSE;
 }
 
index a7e3a15cac52f25b672a37d7c378221a972762f4..2f0589bd4047979c18da350eb0e29f711bbccb2b 100644 (file)
@@ -48,6 +48,7 @@ GtkStyle *create_input_style (GtkStyle *);
 
 #define LABEL_INDENT 12
 
+static GtkWidget *setup_window = NULL;
 static int last_selected_page = 0;
 static int last_selected_row = 0; /* sound row */
 static gboolean color_change;
@@ -1105,8 +1106,8 @@ setup_browsefile_cb (GtkWidget *button, GtkWidget *entry)
        filter = "image/*";
        filter_type = FRF_MIMETYPES;
 #endif
-       gtkutil_file_req (_("Select an Image File"), setup_filereq_cb,
-                                       entry, NULL, filter, filter_type|FRF_RECENTLYUSED);
+       gtkutil_file_req (GTK_WINDOW (setup_window), _("Select an Image File"), setup_filereq_cb,
+                                       entry, NULL, filter, filter_type|FRF_RECENTLYUSED|FRF_MODAL);
 }
 
 static void
@@ -1141,7 +1142,7 @@ setup_fontsel_cancel (GtkWidget *button, GtkFontSelectionDialog *dialog)
 static void
 setup_browsefolder_cb (GtkWidget *button, GtkEntry *entry)
 {
-       gtkutil_file_req (_("Select Download Folder"), setup_filereq_cb, entry, (char*)gtk_entry_get_text (entry), NULL, FRF_CHOOSEFOLDER);
+       gtkutil_file_req (GTK_WINDOW (setup_window), _("Select Download Folder"), setup_filereq_cb, entry, (char*)gtk_entry_get_text (entry), NULL, FRF_CHOOSEFOLDER|FRF_MODAL);
 }
 
 static void
@@ -1154,6 +1155,9 @@ setup_browsefont_cb (GtkWidget *button, GtkWidget *entry)
        dialog = (GtkFontSelectionDialog *) gtk_font_selection_dialog_new (_("Select font"));
        font_dialog = (GtkWidget *)dialog;      /* global var */
 
+       gtk_window_set_transient_for (GTK_WINDOW (font_dialog), GTK_WINDOW (setup_window));
+       gtk_window_set_modal (GTK_WINDOW (font_dialog), TRUE);
+
        sel = (GtkFontSelection *) gtk_font_selection_dialog_get_font_selection (dialog);
 
        if (gtk_entry_get_text (GTK_ENTRY (entry))[0])
@@ -1457,6 +1461,8 @@ setup_color_cb (GtkWidget *button, gpointer userdata)
        g_object_set_data (G_OBJECT (ok_button), "b", button);
        gtk_widget_set_sensitive (help_button, FALSE);
        gtk_color_selection_set_current_color (GTK_COLOR_SELECTION (gtk_color_selection_dialog_get_color_selection (cdialog)), color);
+       gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (setup_window));
+       gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
        gtk_widget_show (dialog);
 
        g_object_unref (cancel_button);
@@ -1711,8 +1717,8 @@ setup_snd_browse_cb (GtkWidget *button, GtkEntry *entry)
        filter_type = FRF_MIMETYPES;
 #endif
 
-       gtkutil_file_req (_("Select a sound file"), setup_snd_filereq_cb, entry,
-                                               sounds_dir, filter, FRF_FILTERISINITIAL|filter_type);
+       gtkutil_file_req (GTK_WINDOW (setup_window), _("Select a sound file"), setup_snd_filereq_cb, entry,
+                                               sounds_dir, filter, FRF_MODAL|FRF_FILTERISINITIAL|filter_type);
        g_free (sounds_dir);
 }
 
@@ -2336,8 +2342,6 @@ setup_close_cb (GtkWidget *win, GtkWidget **swin)
 void
 setup_open (void)
 {
-       static GtkWidget *setup_window = NULL;
-
        if (setup_window)
        {
                gtk_window_present (GTK_WINDOW (setup_window));
index b0f2f392f5633313896636b93dc5b7c5dc94d412..b5eaf8938f5aa9092f1c28b78740e95133201c76 100644 (file)
@@ -282,7 +282,7 @@ pevent_save_cb (GtkWidget * wid, void *data)
 {
        if (data)
        {
-               gtkutil_file_req (_("Print Texts File"), pevent_save_req_cb, NULL,
+               gtkutil_file_req (NULL, _("Print Texts File"), pevent_save_req_cb, NULL,
                                                                NULL, NULL, FRF_WRITE);
                return;
        }
@@ -304,7 +304,7 @@ pevent_load_req_cb (void *arg1, char *file)
 static void
 pevent_load_cb (GtkWidget * wid, void *data)
 {
-       gtkutil_file_req (_("Print Texts File"), pevent_load_req_cb, NULL, NULL, NULL, 0);
+       gtkutil_file_req (NULL, _("Print Texts File"), pevent_load_req_cb, NULL, NULL, NULL, 0);
 }
 
 static void
index fd8d8d91c74d9e7bbb9f360b0165f56612e324da..fc2f4b5a0f179a375f8f8a3d32be835e38792e76 100644 (file)
@@ -145,7 +145,7 @@ url_save_callback (void *arg1, char *file)
 static void
 url_button_save (void)
 {
-       gtkutil_file_req (_("Select an output filename"),
+       gtkutil_file_req (NULL, _("Select an output filename"),
                                                        url_save_callback, NULL, NULL, NULL, FRF_WRITE);
 }