diff options
author | Cosimo Cecchi <cosimoc@gnome.org> | 2010-11-23 17:23:37 +0100 |
---|---|---|
committer | Cosimo Cecchi <cosimoc@gnome.org> | 2010-11-23 17:50:15 +0100 |
commit | 05bb715c51d8963e47a8b34f94b14f0748f6b8d8 (patch) | |
tree | 51c7877d323de62ff889973b082a3c016f72695d /gtk/gtkappchooserdialog.c | |
parent | 6f0a60589125842294579536e62eb099903a0632 (diff) | |
download | gtk+-05bb715c51d8963e47a8b34f94b14f0748f6b8d8.tar.gz |
app-chooser: rename GtkOpenWith to GtkAppChooser
Diffstat (limited to 'gtk/gtkappchooserdialog.c')
-rw-r--r-- | gtk/gtkappchooserdialog.c | 641 |
1 files changed, 641 insertions, 0 deletions
diff --git a/gtk/gtkappchooserdialog.c b/gtk/gtkappchooserdialog.c new file mode 100644 index 0000000000..c6f2fcb429 --- /dev/null +++ b/gtk/gtkappchooserdialog.c @@ -0,0 +1,641 @@ +/* + * gtkappchooserdialog.c: an app-chooser dialog + * + * Copyright (C) 2004 Novell, Inc. + * Copyright (C) 2007, 2010 Red Hat, Inc. + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with the Gnome Library; see the file COPYING.LIB. If not, + * write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + * + * Authors: Dave Camp <dave@novell.com> + * Alexander Larsson <alexl@redhat.com> + * Cosimo Cecchi <ccecchi@redhat.com> + */ + +#include <config.h> + +#include "gtkappchooserdialog.h" + +#include "gtkintl.h" +#include "gtkappchooser.h" +#include "gtkappchooseronline.h" +#include "gtkappchooserprivate.h" + +#include <string.h> +#include <glib/gi18n-lib.h> +#include <gtk/gtk.h> +#include <gio/gio.h> + +#define sure_string(s) ((const char *) ((s) != NULL ? (s) : "")) + +struct _GtkAppChooserDialogPrivate { + char *content_type; + GFile *gfile; + + GtkWidget *label; + GtkWidget *button; + GtkWidget *online_button; + + GtkWidget *open_label; + + GtkWidget *app_chooser_widget; + GtkWidget *show_more_button; + + gboolean show_more_clicked; +}; + +enum { + PROP_GFILE = 1, + PROP_CONTENT_TYPE, + N_PROPERTIES +}; + +static void gtk_app_chooser_dialog_iface_init (GtkAppChooserIface *iface); +G_DEFINE_TYPE_WITH_CODE (GtkAppChooserDialog, gtk_app_chooser_dialog, GTK_TYPE_DIALOG, + G_IMPLEMENT_INTERFACE (GTK_TYPE_APP_CHOOSER, + gtk_app_chooser_dialog_iface_init)); + +static void +show_error_dialog (const gchar *primary, + const gchar *secondary, + GtkWindow *parent) +{ + GtkWidget *message_dialog; + + message_dialog = gtk_message_dialog_new (parent, 0, + GTK_MESSAGE_ERROR, + GTK_BUTTONS_OK, + NULL); + g_object_set (message_dialog, + "text", primary, + "secondary-text", secondary, + NULL); + gtk_dialog_set_default_response (GTK_DIALOG (message_dialog), GTK_RESPONSE_OK); + + gtk_widget_show (message_dialog); + + g_signal_connect (message_dialog, "response", + G_CALLBACK (gtk_widget_destroy), NULL); +} + +static void +search_for_mimetype_ready_cb (GObject *source, + GAsyncResult *res, + gpointer user_data) +{ + GtkAppChooserOnline *online = GTK_APP_CHOOSER_ONLINE (source); + GtkAppChooserDialog *self = user_data; + GError *error = NULL; + + gtk_app_chooser_online_search_for_mimetype_finish (online, res, &error); + + if (error != NULL) + { + show_error_dialog (_("Failed to look for applications online"), + error->message, GTK_WINDOW (self)); + g_error_free (error); + } + else + { + gtk_app_chooser_refresh (GTK_APP_CHOOSER (self->priv->app_chooser_widget)); + } + + g_object_unref (online); +} + +static void +online_button_clicked_cb (GtkButton *b, + gpointer user_data) +{ + GtkAppChooserOnline *online; + GtkAppChooserDialog *self = user_data; + + online = gtk_app_chooser_online_get_default (); + + gtk_app_chooser_online_search_for_mimetype_async (online, + self->priv->content_type, + GTK_WINDOW (self), + search_for_mimetype_ready_cb, + self); +} + +/* An application is valid if: + * + * 1) The file exists + * 2) The user has permissions to run the file + */ +static gboolean +check_application (GtkAppChooserDialog *self, + GAppInfo **app_out) +{ + const char *command; + char *path = NULL; + char **argv = NULL; + int argc; + GError *error = NULL; + gint retval = TRUE; + GAppInfo *info; + + command = NULL; + + info = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (self->priv->app_chooser_widget)); + command = g_app_info_get_executable (info); + + g_shell_parse_argv (command, &argc, &argv, &error); + + if (error) + { + show_error_dialog (_("Could not run application"), + error->message, + GTK_WINDOW (self)); + g_error_free (error); + retval = FALSE; + goto cleanup; + } + + path = g_find_program_in_path (argv[0]); + if (!path) + { + char *error_message; + + error_message = g_strdup_printf (_("Could not find '%s'"), + argv[0]); + + show_error_dialog (_("Could not find application"), + error_message, + GTK_WINDOW (self)); + g_free (error_message); + retval = FALSE; + goto cleanup; + } + + *app_out = info; + + cleanup: + g_strfreev (argv); + g_free (path); + + return retval; +} + +static void +widget_application_selected_cb (GtkAppChooserWidget *widget, + GAppInfo *app_info, + gpointer user_data) +{ + GtkAppChooserDialog *self = user_data; + + gtk_widget_set_sensitive (self->priv->button, TRUE); +} + +static void +widget_application_activated_cb (GtkAppChooserWidget *widget, + GAppInfo *app_info, + gpointer user_data) +{ + GtkAppChooserDialog *self = user_data; + + gtk_dialog_response (GTK_DIALOG (self), GTK_RESPONSE_OK); +} + +static char * +get_extension (const char *basename) +{ + char *p; + + p = strrchr (basename, '.'); + + if (p && *(p + 1) != '\0') + return g_strdup (p + 1); + + return NULL; +} + +static void +set_dialog_properties (GtkAppChooserDialog *self) +{ + char *label, *name, *extension, *description, *default_text, *string; + PangoFontDescription *font_desc; + + name = NULL; + extension = NULL; + label = NULL; + description = NULL; + + if (self->priv->gfile != NULL) + { + name = g_file_get_basename (self->priv->gfile); + extension = get_extension (name); + } + + description = g_content_type_get_description (self->priv->content_type); + gtk_window_set_title (GTK_WINDOW (self), ""); + + if (name != NULL) + { + /* Translators: %s is a filename */ + label = g_strdup_printf (_("Select an application to open \"%s\""), name); + string = g_strdup_printf (_("No applications available to open \"%s\""), + name); + } + else + { + /* Translators: %s is a file type description */ + label = g_strdup_printf (_("Select an application for \"%s\" files"), + g_content_type_is_unknown (self->priv->content_type) ? + self->priv->content_type : description); + string = g_strdup_printf (_("No applications available to open \"%s\" files"), + g_content_type_is_unknown (self->priv->content_type) ? + self->priv->content_type : description); + } + + font_desc = pango_font_description_new (); + pango_font_description_set_weight (font_desc, PANGO_WEIGHT_BOLD); + gtk_widget_modify_font (self->priv->label, font_desc); + pango_font_description_free (font_desc); + + gtk_label_set_markup (GTK_LABEL (self->priv->label), label); + + default_text = g_strdup_printf ("<big><b>%s</b></big>\n%s", + string, + _("Click \"Show other applications\", for more options, or " + "\"Find applications online\" to install a new application")); + + gtk_app_chooser_widget_set_default_text (GTK_APP_CHOOSER_WIDGET (self->priv->app_chooser_widget), + default_text); + + g_free (label); + g_free (name); + g_free (extension); + g_free (description); + g_free (string); + g_free (default_text); +} + +static void +show_more_button_clicked_cb (GtkButton *button, + gpointer user_data) +{ + GtkAppChooserDialog *self = user_data; + + g_object_set (self->priv->app_chooser_widget, + "show-recommended", TRUE, + "show-fallback", TRUE, + "show-other", TRUE, + NULL); + + gtk_widget_hide (self->priv->show_more_button); + self->priv->show_more_clicked = TRUE; +} + +static void +widget_notify_for_button_cb (GObject *source, + GParamSpec *pspec, + gpointer user_data) +{ + GtkAppChooserDialog *self = user_data; + GtkAppChooserWidget *widget = GTK_APP_CHOOSER_WIDGET (source); + gboolean should_hide; + + should_hide = gtk_app_chooser_widget_get_show_all (widget) || + self->priv->show_more_clicked; + + if (should_hide) + gtk_widget_hide (self->priv->show_more_button); +} + +static void +build_dialog_ui (GtkAppChooserDialog *self) +{ + GtkWidget *vbox; + GtkWidget *vbox2; + GtkWidget *label; + GtkWidget *action_area, *button, *w; + + gtk_container_set_border_width (GTK_CONTAINER (self), 5); + + vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12); + gtk_container_set_border_width (GTK_CONTAINER (vbox), 5); + gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (self))), vbox, TRUE, TRUE, 0); + gtk_widget_show (vbox); + + vbox2 = gtk_box_new (GTK_ORIENTATION_VERTICAL, 6); + gtk_box_pack_start (GTK_BOX (vbox), vbox2, TRUE, TRUE, 0); + gtk_widget_show (vbox2); + + self->priv->label = gtk_label_new (""); + gtk_misc_set_alignment (GTK_MISC (self->priv->label), 0, 0.5); + gtk_label_set_line_wrap (GTK_LABEL (self->priv->label), TRUE); + gtk_box_pack_start (GTK_BOX (vbox2), self->priv->label, + FALSE, FALSE, 0); + gtk_widget_show (self->priv->label); + + self->priv->app_chooser_widget = + gtk_app_chooser_widget_new (self->priv->content_type); + gtk_box_pack_start (GTK_BOX (vbox2), self->priv->app_chooser_widget, TRUE, TRUE, 0); + gtk_widget_show (self->priv->app_chooser_widget); + + g_signal_connect (self->priv->app_chooser_widget, "application-selected", + G_CALLBACK (widget_application_selected_cb), self); + g_signal_connect (self->priv->app_chooser_widget, "application-activated", + G_CALLBACK (widget_application_activated_cb), self); + g_signal_connect (self->priv->app_chooser_widget, "notify::show-all", + G_CALLBACK (widget_notify_for_button_cb), self); + + button = gtk_button_new_with_label (_("Show other applications")); + self->priv->show_more_button = button; + w = gtk_image_new_from_stock (GTK_STOCK_ADD, + GTK_ICON_SIZE_BUTTON); + gtk_button_set_image (GTK_BUTTON (button), w); + gtk_box_pack_start (GTK_BOX (self->priv->app_chooser_widget), button, FALSE, FALSE, 6); + gtk_widget_show_all (button); + + g_signal_connect (button, "clicked", + G_CALLBACK (show_more_button_clicked_cb), self); + + gtk_dialog_add_button (GTK_DIALOG (self), + GTK_STOCK_CANCEL, + GTK_RESPONSE_CANCEL); + + /* Create a custom stock icon */ + self->priv->button = gtk_button_new (); + + label = gtk_label_new_with_mnemonic (_("_Open")); + gtk_label_set_mnemonic_widget (GTK_LABEL (label), GTK_WIDGET (self->priv->button)); + gtk_widget_set_halign (label, GTK_ALIGN_CENTER); + gtk_widget_show (label); + self->priv->open_label = label; + + gtk_container_add (GTK_CONTAINER (self->priv->button), + self->priv->open_label); + + gtk_widget_show (self->priv->button); + gtk_widget_set_can_default (self->priv->button, TRUE); + + gtk_dialog_add_action_widget (GTK_DIALOG (self), + self->priv->button, GTK_RESPONSE_OK); + + action_area = gtk_dialog_get_action_area (GTK_DIALOG (self)); + self->priv->online_button = gtk_button_new_with_label (_("Find applications online")); + gtk_box_pack_start (GTK_BOX (action_area), self->priv->online_button, + FALSE, FALSE, 0); + gtk_button_box_set_child_secondary (GTK_BUTTON_BOX (action_area), self->priv->online_button, + TRUE); + gtk_widget_show (self->priv->online_button); + g_signal_connect (self->priv->online_button, "clicked", + G_CALLBACK (online_button_clicked_cb), self); + + gtk_dialog_set_default_response (GTK_DIALOG (self), + GTK_RESPONSE_OK); +} + +static void +set_gfile_and_content_type (GtkAppChooserDialog *self, + GFile *file) +{ + GFileInfo *info; + + if (file == NULL) + return; + + self->priv->gfile = g_object_ref (file); + + info = g_file_query_info (self->priv->gfile, + G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, + 0, NULL, NULL); + self->priv->content_type = g_strdup (g_file_info_get_content_type (info)); + + g_object_unref (info); +} + +static GAppInfo * +gtk_app_chooser_dialog_get_app_info (GtkAppChooser *object) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + GAppInfo *app = NULL; + + if (!check_application (self, &app)) + return NULL; + + return app; +} + +static void +gtk_app_chooser_dialog_refresh (GtkAppChooser *object) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + + gtk_app_chooser_refresh (GTK_APP_CHOOSER (self->priv->app_chooser_widget)); +} + +static void +gtk_app_chooser_dialog_constructed (GObject *object) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + + g_assert (self->priv->content_type != NULL || + self->priv->gfile != NULL); + + if (G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->constructed != NULL) + G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->constructed (object); + + build_dialog_ui (self); + set_dialog_properties (self); +} + +static void +gtk_app_chooser_dialog_finalize (GObject *object) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + + if (self->priv->gfile) + g_object_unref (self->priv->gfile); + + g_free (self->priv->content_type); + + G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->finalize (object); +} + +static void +gtk_app_chooser_dialog_set_property (GObject *object, + guint property_id, + const GValue *value, + GParamSpec *pspec) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + + switch (property_id) + { + case PROP_GFILE: + set_gfile_and_content_type (self, g_value_get_object (value)); + break; + case PROP_CONTENT_TYPE: + /* don't try to override a value previously set with the GFile */ + if (self->priv->content_type == NULL) + self->priv->content_type = g_value_dup_string (value); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); + break; + } +} + +static void +gtk_app_chooser_dialog_get_property (GObject *object, + guint property_id, + GValue *value, + GParamSpec *pspec) +{ + GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object); + + switch (property_id) + { + case PROP_GFILE: + if (self->priv->gfile != NULL) + g_value_set_object (value, self->priv->gfile); + break; + case PROP_CONTENT_TYPE: + g_value_set_string (value, self->priv->content_type); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); + break; + } +} + +static void +gtk_app_chooser_dialog_iface_init (GtkAppChooserIface *iface) +{ + iface->get_app_info = gtk_app_chooser_dialog_get_app_info; + iface->refresh = gtk_app_chooser_dialog_refresh; +} + +static void +gtk_app_chooser_dialog_class_init (GtkAppChooserDialogClass *klass) +{ + GObjectClass *gobject_class; + GParamSpec *pspec; + + gobject_class = G_OBJECT_CLASS (klass); + gobject_class->finalize = gtk_app_chooser_dialog_finalize; + gobject_class->set_property = gtk_app_chooser_dialog_set_property; + gobject_class->get_property = gtk_app_chooser_dialog_get_property; + gobject_class->constructed = gtk_app_chooser_dialog_constructed; + + g_object_class_override_property (gobject_class, PROP_CONTENT_TYPE, "content-type"); + + pspec = g_param_spec_object ("gfile", + P_("GFile"), + P_("The GFile used by the open with dialog"), + G_TYPE_FILE, + G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | + G_PARAM_STATIC_STRINGS); + g_object_class_install_property (gobject_class, PROP_GFILE, pspec); + + g_type_class_add_private (klass, sizeof (GtkAppChooserDialogPrivate)); +} + +static void +gtk_app_chooser_dialog_init (GtkAppChooserDialog *self) +{ + self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, GTK_TYPE_APP_CHOOSER_DIALOG, + GtkAppChooserDialogPrivate); +} + +static void +set_parent_and_flags (GtkWidget *dialog, + GtkWindow *parent, + GtkDialogFlags flags) +{ + if (parent != NULL) + gtk_window_set_transient_for (GTK_WINDOW (dialog), + parent); + + if (flags & GTK_DIALOG_MODAL) + gtk_window_set_modal (GTK_WINDOW (dialog), TRUE); + + if (flags & GTK_DIALOG_DESTROY_WITH_PARENT) + gtk_window_set_destroy_with_parent (GTK_WINDOW (dialog), TRUE); +} + +/** + * gtk_app_chooser_dialog_new: + * @parent: (allow-none): a #GtkWindow, or %NULL + * @flags: flags for this dialog + * @file: a #GFile + * + * Creates a new #GtkAppChooserDialog for the provided #GFile, to allow + * the user to select an application for it. + * + * Returns: a newly created #GtkAppChooserDialog + * + * Since: 3.0 + **/ +GtkWidget * +gtk_app_chooser_dialog_new (GtkWindow *parent, + GtkDialogFlags flags, + GFile *file) +{ + GtkWidget *retval; + + g_return_val_if_fail (G_IS_FILE (file), NULL); + + retval = g_object_new (GTK_TYPE_APP_CHOOSER_DIALOG, + "gfile", file, + NULL); + + set_parent_and_flags (retval, parent, flags); + + return retval; +} + +/** + * gtk_app_chooser_dialog_new_for_content_type: + * @parent: (allow-none): a #GtkWindow, or %NULL + * @flags: flags for this dialog + * @content_type: a content type string + * + * Creates a new #GtkAppChooserDialog for the provided content type, to allow + * the user to select an application for it. + * + * Returns: a newly created #GtkAppChooserDialog + * + * Since: 3.0 + **/ +GtkWidget * +gtk_app_chooser_dialog_new_for_content_type (GtkWindow *parent, + GtkDialogFlags flags, + const gchar *content_type) +{ + GtkWidget *retval; + + g_return_val_if_fail (content_type != NULL, NULL); + + retval = g_object_new (GTK_TYPE_APP_CHOOSER_DIALOG, + "content-type", content_type, + NULL); + + set_parent_and_flags (retval, parent, flags); + + return retval; +} + +GtkWidget * +gtk_app_chooser_dialog_get_widget (GtkAppChooserDialog *self) +{ + g_return_val_if_fail (GTK_IS_APP_CHOOSER_DIALOG (self), NULL); + + return self->priv->app_chooser_widget; +} |