rawTherapee/rtgui/rtappchooserdialog.cc
Lawrence Lee d3e524a491 Fix crash when adding an external editor
On some platforms, the app chooser dialog causes a crash after selecting
an application. The application information returned by the dialog may
also trigger crashes when accessed. See
https://gitlab.gnome.org/GNOME/glib/-/issues/1104 and
https://gitlab.gnome.org/GNOME/glibmm/-/issues/94. This commit overrides
gtkmm's app chooser dialog to work around these bugs.
2021-08-14 16:05:11 -07:00

78 lines
2.5 KiB
C++

/*
* This file is part of RawTherapee.
*
* Copyright (c) 2021 Lawrence Lee <billee@ucdavis.edu>
*
* RawTherapee is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* RawTherapee is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with RawTherapee. If not, see <https://www.gnu.org/licenses/>.
*/
#include "rtappchooserdialog.h"
#if !(defined WIN32 || defined __APPLE__)
#define GTKMM_APPCHOOSERDIALOG
#endif
RTAppChooserDialog::~RTAppChooserDialog() {}
#ifdef GTKMM_APPCHOOSERDIALOG // Use Gtk::AppChooserDialog directly.
RTAppChooserDialog::RTAppChooserDialog(const Glib::ustring &content_type) :
Gtk::AppChooserDialog(content_type)
{
}
Glib::RefPtr<Gio::AppInfo> RTAppChooserDialog::get_app_info()
{
return Gtk::AppChooserDialog::get_app_info();
}
Glib::RefPtr<const Gio::AppInfo> RTAppChooserDialog::get_app_info() const
{
return Gtk::AppChooserDialog::get_app_info();
}
#else // Work around bugs with GLib and glibmm.
RTAppChooserDialog::RTAppChooserDialog(const Glib::ustring &content_type) :
Gtk::AppChooserDialog(content_type)
{
// GTK calls a faulty GLib function to update the most recently selected
// application after an application is selected. This removes all signal
// handlers to prevent the function call.
auto signal_id = g_signal_lookup("response", GTK_TYPE_APP_CHOOSER_DIALOG);
while (true) {
auto handler_id = g_signal_handler_find(gobj(), G_SIGNAL_MATCH_ID, signal_id, GQuark(), nullptr, nullptr, nullptr);
if (!handler_id) {
break;
}
g_signal_handler_disconnect(gobj(), handler_id);
}
}
Glib::RefPtr<Gio::AppInfo> RTAppChooserDialog::get_app_info()
{
// glibmm wrapping of GAppInfo does not work on some platforms. Manually
// wrap it here.
GAppInfo *gAppInfo = gtk_app_chooser_get_app_info(GTK_APP_CHOOSER(gobj()));
return Glib::wrap(gAppInfo, true);
}
Glib::RefPtr<const Gio::AppInfo> RTAppChooserDialog::get_app_info() const
{
GAppInfo *gAppInfo = gtk_app_chooser_get_app_info(GTK_APP_CHOOSER(
const_cast<GtkAppChooserDialog *>(gobj())));
return Glib::wrap(gAppInfo, true);
}
#endif