2 * gtkappchooserdialog.c: an app-chooser dialog
4 * Copyright (C) 2004 Novell, Inc.
5 * Copyright (C) 2007, 2010 Red Hat, Inc.
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Library General Public License as
9 * published by the Free Software Foundation; either version 2 of the
10 * License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Library General Public License for more details.
17 * You should have received a copy of the GNU Library General Public
18 * License along with this library. If not, see <http://www.gnu.org/licenses/>.
20 * Authors: Dave Camp <dave@novell.com>
21 * Alexander Larsson <alexl@redhat.com>
22 * Cosimo Cecchi <ccecchi@redhat.com>
26 * SECTION:gtkappchooserdialog
27 * @Title: GtkAppChooserDialog
28 * @Short_description: An application chooser dialog
30 * #GtkAppChooserDialog shows a #GtkAppChooserWidget inside a #GtkDialog.
32 * Note that #GtkAppChooserDialog does not have any interesting methods
33 * of its own. Instead, you should get the embedded #GtkAppChooserWidget
34 * using gtk_app_chooser_dialog_get_widget() and call its methods if
35 * the generic #GtkAppChooser interface is not sufficient for your needs.
37 * To set the heading that is shown above the #GtkAppChooserWidget,
38 * use gtk_app_chooser_dialog_set_heading().
42 #include "gtkappchooserdialog.h"
45 #include "gtkappchooser.h"
46 #include "gtkappchooseronline.h"
47 #include "gtkappchooserprivate.h"
48 #include "gtkappchooserprivate.h"
50 #include "gtkmessagedialog.h"
53 #include "gtkbutton.h"
54 #include "gtkmenuitem.h"
58 #include <glib/gi18n-lib.h>
61 #define sure_string(s) ((const char *) ((s) != NULL ? (s) : ""))
63 struct _GtkAppChooserDialogPrivate {
70 GtkWidget *online_button;
72 GtkWidget *open_label;
74 GtkWidget *app_chooser_widget;
75 GtkWidget *show_more_button;
77 GtkAppChooserOnline *online;
78 GCancellable *online_cancellable;
80 gboolean show_more_clicked;
90 static void gtk_app_chooser_dialog_iface_init (GtkAppChooserIface *iface);
91 G_DEFINE_TYPE_WITH_CODE (GtkAppChooserDialog, gtk_app_chooser_dialog, GTK_TYPE_DIALOG,
92 G_IMPLEMENT_INTERFACE (GTK_TYPE_APP_CHOOSER,
93 gtk_app_chooser_dialog_iface_init));
96 show_error_dialog (const gchar *primary,
97 const gchar *secondary,
100 GtkWidget *message_dialog;
102 message_dialog = gtk_message_dialog_new (parent, 0,
106 g_object_set (message_dialog,
108 "secondary-text", secondary,
110 gtk_dialog_set_default_response (GTK_DIALOG (message_dialog), GTK_RESPONSE_OK);
112 gtk_widget_show (message_dialog);
114 g_signal_connect (message_dialog, "response",
115 G_CALLBACK (gtk_widget_destroy), NULL);
119 search_for_mimetype_ready_cb (GObject *source,
123 GtkAppChooserOnline *online = GTK_APP_CHOOSER_ONLINE (source);
124 GtkAppChooserDialog *self = user_data;
125 GError *error = NULL;
127 gdk_threads_enter ();
129 _gtk_app_chooser_online_search_for_mimetype_finish (online, res, &error);
131 if (self->priv->dismissed)
135 !g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
137 show_error_dialog (_("Failed to look for applications online"),
138 error->message, GTK_WINDOW (self));
142 gtk_widget_set_sensitive (self->priv->online_button, TRUE);
143 gtk_app_chooser_refresh (GTK_APP_CHOOSER (self->priv->app_chooser_widget));
147 g_clear_object (&self->priv->online_cancellable);
148 g_clear_error (&error);
149 g_object_unref (self);
151 gdk_threads_leave ();
155 online_button_clicked_cb (GtkButton *b,
158 GtkAppChooserDialog *self = user_data;
160 self->priv->online_cancellable = g_cancellable_new ();
161 gtk_widget_set_sensitive (self->priv->online_button, FALSE);
163 _gtk_app_chooser_online_search_for_mimetype_async (self->priv->online,
164 self->priv->content_type,
166 self->priv->online_cancellable,
167 search_for_mimetype_ready_cb,
168 g_object_ref (self));
172 app_chooser_online_get_default_ready_cb (GObject *source,
176 GtkAppChooserDialog *self = user_data;
178 gdk_threads_enter ();
180 self->priv->online = _gtk_app_chooser_online_get_default_finish (source, res);
182 if (self->priv->online != NULL &&
183 !self->priv->dismissed)
185 GtkWidget *action_area;
187 action_area = gtk_dialog_get_action_area (GTK_DIALOG (self));
188 self->priv->online_button = gtk_button_new_with_mnemonic (_("_Find applications online"));
189 gtk_box_pack_start (GTK_BOX (action_area), self->priv->online_button,
191 gtk_button_box_set_child_secondary (GTK_BUTTON_BOX (action_area), self->priv->online_button,
193 g_signal_connect (self->priv->online_button, "clicked",
194 G_CALLBACK (online_button_clicked_cb), self);
197 if (!self->priv->content_type)
198 gtk_widget_set_sensitive (self->priv->online_button, FALSE);
200 gtk_widget_show (self->priv->online_button);
203 g_object_unref (self);
205 gdk_threads_leave ();
209 ensure_online_button (GtkAppChooserDialog *self)
211 _gtk_app_chooser_online_get_default_async (app_chooser_online_get_default_ready_cb,
212 g_object_ref (self));
215 /* An application is valid if:
218 * 2) The user has permissions to run the file
221 check_application (GtkAppChooserDialog *self,
228 GError *error = NULL;
234 info = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (self->priv->app_chooser_widget));
241 command = g_app_info_get_executable (info);
243 g_shell_parse_argv (command, &argc, &argv, &error);
247 show_error_dialog (_("Could not run application"),
250 g_error_free (error);
255 path = g_find_program_in_path (argv[0]);
260 error_message = g_strdup_printf (_("Could not find '%s'"),
263 show_error_dialog (_("Could not find application"),
266 g_free (error_message);
281 add_or_find_application (GtkAppChooserDialog *self)
285 app = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (self));
289 /* we don't care about reporting errors here */
290 if (self->priv->content_type)
291 g_app_info_set_as_last_used_for_type (app,
292 self->priv->content_type,
294 g_object_unref (app);
299 cancel_and_clear_cancellable (GtkAppChooserDialog *self)
301 if (self->priv->online_cancellable != NULL)
303 g_cancellable_cancel (self->priv->online_cancellable);
304 g_clear_object (&self->priv->online_cancellable);
309 gtk_app_chooser_dialog_response (GtkDialog *dialog,
313 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (dialog);
317 case GTK_RESPONSE_OK:
318 add_or_find_application (self);
320 case GTK_RESPONSE_CANCEL:
321 case GTK_RESPONSE_DELETE_EVENT:
322 cancel_and_clear_cancellable (self);
323 self->priv->dismissed = TRUE;
330 widget_application_selected_cb (GtkAppChooserWidget *widget,
334 GtkAppChooserDialog *self = user_data;
336 gtk_widget_set_sensitive (self->priv->button, TRUE);
340 widget_application_activated_cb (GtkAppChooserWidget *widget,
344 GtkAppChooserDialog *self = user_data;
346 gtk_dialog_response (GTK_DIALOG (self), GTK_RESPONSE_OK);
350 get_extension (const char *basename)
354 p = strrchr (basename, '.');
356 if (p && *(p + 1) != '\0')
357 return g_strdup (p + 1);
363 set_dialog_properties (GtkAppChooserDialog *self)
372 PangoFontDescription *font_desc;
380 if (self->priv->gfile != NULL)
382 name = g_file_get_basename (self->priv->gfile);
383 extension = get_extension (name);
386 if (self->priv->content_type)
388 description = g_content_type_get_description (self->priv->content_type);
389 unknown = g_content_type_is_unknown (self->priv->content_type);
392 gtk_window_set_title (GTK_WINDOW (self), "");
396 /* Translators: %s is a filename */
397 label = g_strdup_printf (_("Select an application to open \"%s\""), name);
398 string = g_strdup_printf (_("No applications available to open \"%s\""),
403 /* Translators: %s is a file type description */
404 label = g_strdup_printf (_("Select an application for \"%s\" files"),
405 unknown ? self->priv->content_type : description);
406 string = g_strdup_printf (_("No applications available to open \"%s\" files"),
407 unknown ? self->priv->content_type : description);
410 font_desc = pango_font_description_new ();
411 pango_font_description_set_weight (font_desc, PANGO_WEIGHT_BOLD);
412 gtk_widget_override_font (self->priv->label, font_desc);
413 pango_font_description_free (font_desc);
415 if (self->priv->heading != NULL)
416 gtk_label_set_markup (GTK_LABEL (self->priv->label), self->priv->heading);
418 gtk_label_set_markup (GTK_LABEL (self->priv->label), label);
420 default_text = g_strdup_printf ("<big><b>%s</b></big>\n%s",
422 _("Click \"Show other applications\", for more options, or "
423 "\"Find applications online\" to install a new application"));
425 gtk_app_chooser_widget_set_default_text (GTK_APP_CHOOSER_WIDGET (self->priv->app_chooser_widget),
431 g_free (description);
433 g_free (default_text);
437 show_more_button_clicked_cb (GtkButton *button,
440 GtkAppChooserDialog *self = user_data;
442 g_object_set (self->priv->app_chooser_widget,
443 "show-recommended", TRUE,
444 "show-fallback", TRUE,
448 gtk_widget_hide (self->priv->show_more_button);
449 self->priv->show_more_clicked = TRUE;
453 widget_notify_for_button_cb (GObject *source,
457 GtkAppChooserDialog *self = user_data;
458 GtkAppChooserWidget *widget = GTK_APP_CHOOSER_WIDGET (source);
459 gboolean should_hide;
461 should_hide = gtk_app_chooser_widget_get_show_other (widget) ||
462 self->priv->show_more_clicked;
465 gtk_widget_hide (self->priv->show_more_button);
469 forget_menu_item_activate_cb (GtkMenuItem *item,
472 GtkAppChooserDialog *self = user_data;
475 info = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (self));
479 g_app_info_remove_supports_type (info, self->priv->content_type, NULL);
481 gtk_app_chooser_refresh (GTK_APP_CHOOSER (self));
483 g_object_unref (info);
488 build_forget_menu_item (GtkAppChooserDialog *self)
492 retval = gtk_menu_item_new_with_label (_("Forget association"));
493 gtk_widget_show (retval);
495 g_signal_connect (retval, "activate",
496 G_CALLBACK (forget_menu_item_activate_cb), self);
502 widget_populate_popup_cb (GtkAppChooserWidget *widget,
507 GtkAppChooserDialog *self = user_data;
508 GtkWidget *menu_item;
510 if (g_app_info_can_remove_supports_type (info))
512 menu_item = build_forget_menu_item (self);
513 gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
518 build_dialog_ui (GtkAppChooserDialog *self)
522 GtkWidget *button, *w;
525 gtk_container_set_border_width (GTK_CONTAINER (self), 5);
527 vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
528 gtk_container_set_border_width (GTK_CONTAINER (vbox), 5);
529 gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (self))), vbox, TRUE, TRUE, 0);
530 gtk_widget_show (vbox);
532 vbox2 = gtk_box_new (GTK_ORIENTATION_VERTICAL, 6);
533 gtk_box_pack_start (GTK_BOX (vbox), vbox2, TRUE, TRUE, 0);
534 gtk_widget_show (vbox2);
536 self->priv->label = gtk_label_new ("");
537 gtk_widget_set_halign (self->priv->label, GTK_ALIGN_START);
538 gtk_widget_set_valign (self->priv->label, GTK_ALIGN_CENTER);
539 gtk_label_set_line_wrap (GTK_LABEL (self->priv->label), TRUE);
540 gtk_box_pack_start (GTK_BOX (vbox2), self->priv->label,
542 gtk_widget_show (self->priv->label);
544 self->priv->app_chooser_widget =
545 gtk_app_chooser_widget_new (self->priv->content_type);
546 gtk_box_pack_start (GTK_BOX (vbox2), self->priv->app_chooser_widget, TRUE, TRUE, 0);
547 gtk_widget_show (self->priv->app_chooser_widget);
549 g_signal_connect (self->priv->app_chooser_widget, "application-selected",
550 G_CALLBACK (widget_application_selected_cb), self);
551 g_signal_connect (self->priv->app_chooser_widget, "application-activated",
552 G_CALLBACK (widget_application_activated_cb), self);
553 g_signal_connect (self->priv->app_chooser_widget, "notify::show-other",
554 G_CALLBACK (widget_notify_for_button_cb), self);
555 g_signal_connect (self->priv->app_chooser_widget, "populate-popup",
556 G_CALLBACK (widget_populate_popup_cb), self);
558 button = gtk_button_new_with_label (_("Show other applications"));
559 self->priv->show_more_button = button;
560 w = gtk_image_new_from_stock (GTK_STOCK_ADD,
561 GTK_ICON_SIZE_BUTTON);
562 gtk_button_set_image (GTK_BUTTON (button), w);
563 gtk_box_pack_start (GTK_BOX (self->priv->app_chooser_widget), button, FALSE, FALSE, 6);
564 gtk_widget_show_all (button);
566 g_signal_connect (button, "clicked",
567 G_CALLBACK (show_more_button_clicked_cb), self);
569 gtk_dialog_add_button (GTK_DIALOG (self),
571 GTK_RESPONSE_CANCEL);
573 self->priv->button = gtk_dialog_add_button (GTK_DIALOG (self),
577 info = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (self->priv->app_chooser_widget));
578 gtk_widget_set_sensitive (self->priv->button, info != NULL);
580 g_object_unref (info);
582 gtk_dialog_set_default_response (GTK_DIALOG (self),
587 set_gfile_and_content_type (GtkAppChooserDialog *self,
595 self->priv->gfile = g_object_ref (file);
597 info = g_file_query_info (self->priv->gfile,
598 G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE,
600 self->priv->content_type = g_strdup (g_file_info_get_content_type (info));
602 g_object_unref (info);
606 gtk_app_chooser_dialog_get_app_info (GtkAppChooser *object)
608 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
609 GAppInfo *app = NULL;
611 if (!check_application (self, &app))
618 gtk_app_chooser_dialog_refresh (GtkAppChooser *object)
620 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
622 gtk_app_chooser_refresh (GTK_APP_CHOOSER (self->priv->app_chooser_widget));
626 gtk_app_chooser_dialog_constructed (GObject *object)
628 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
630 if (G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->constructed != NULL)
631 G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->constructed (object);
633 build_dialog_ui (self);
634 set_dialog_properties (self);
635 ensure_online_button (self);
639 gtk_app_chooser_dialog_dispose (GObject *object)
641 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
643 g_clear_object (&self->priv->gfile);
644 cancel_and_clear_cancellable (self);
645 g_clear_object (&self->priv->online);
647 G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->dispose (object);
651 gtk_app_chooser_dialog_finalize (GObject *object)
653 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
655 g_free (self->priv->content_type);
657 G_OBJECT_CLASS (gtk_app_chooser_dialog_parent_class)->finalize (object);
661 gtk_app_chooser_dialog_set_property (GObject *object,
666 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
671 set_gfile_and_content_type (self, g_value_get_object (value));
673 case PROP_CONTENT_TYPE:
674 /* don't try to override a value previously set with the GFile */
675 if (self->priv->content_type == NULL)
676 self->priv->content_type = g_value_dup_string (value);
679 gtk_app_chooser_dialog_set_heading (self, g_value_get_string (value));
682 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
688 gtk_app_chooser_dialog_get_property (GObject *object,
693 GtkAppChooserDialog *self = GTK_APP_CHOOSER_DIALOG (object);
698 if (self->priv->gfile != NULL)
699 g_value_set_object (value, self->priv->gfile);
701 case PROP_CONTENT_TYPE:
702 g_value_set_string (value, self->priv->content_type);
705 g_value_set_string (value, self->priv->heading);
708 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
714 gtk_app_chooser_dialog_iface_init (GtkAppChooserIface *iface)
716 iface->get_app_info = gtk_app_chooser_dialog_get_app_info;
717 iface->refresh = gtk_app_chooser_dialog_refresh;
721 gtk_app_chooser_dialog_class_init (GtkAppChooserDialogClass *klass)
723 GObjectClass *gobject_class;
726 gobject_class = G_OBJECT_CLASS (klass);
727 gobject_class->dispose = gtk_app_chooser_dialog_dispose;
728 gobject_class->finalize = gtk_app_chooser_dialog_finalize;
729 gobject_class->set_property = gtk_app_chooser_dialog_set_property;
730 gobject_class->get_property = gtk_app_chooser_dialog_get_property;
731 gobject_class->constructed = gtk_app_chooser_dialog_constructed;
733 g_object_class_override_property (gobject_class, PROP_CONTENT_TYPE, "content-type");
736 * GtkAppChooserDialog:gfile:
738 * The GFile used by the #GtkAppChooserDialog.
739 * The dialog's #GtkAppChooserWidget content type will be guessed from the
742 pspec = g_param_spec_object ("gfile",
744 P_("The GFile used by the app chooser dialog"),
746 G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE |
747 G_PARAM_STATIC_STRINGS);
748 g_object_class_install_property (gobject_class, PROP_GFILE, pspec);
751 * GtkAppChooserDialog:heading:
753 * The text to show at the top of the dialog.
754 * The string may contain Pango markup.
756 pspec = g_param_spec_string ("heading",
758 P_("The text to show at the top of the dialog"),
760 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
761 g_object_class_install_property (gobject_class, PROP_HEADING, pspec);
764 g_type_class_add_private (klass, sizeof (GtkAppChooserDialogPrivate));
768 gtk_app_chooser_dialog_init (GtkAppChooserDialog *self)
770 self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, GTK_TYPE_APP_CHOOSER_DIALOG,
771 GtkAppChooserDialogPrivate);
773 /* we can't override the class signal handler here, as it's a RUN_LAST;
774 * we want our signal handler instead to be executed before any user code.
776 g_signal_connect (self, "response",
777 G_CALLBACK (gtk_app_chooser_dialog_response), NULL);
781 set_parent_and_flags (GtkWidget *dialog,
783 GtkDialogFlags flags)
786 gtk_window_set_transient_for (GTK_WINDOW (dialog), parent);
788 if (flags & GTK_DIALOG_MODAL)
789 gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
791 if (flags & GTK_DIALOG_DESTROY_WITH_PARENT)
792 gtk_window_set_destroy_with_parent (GTK_WINDOW (dialog), TRUE);
796 * gtk_app_chooser_dialog_new:
797 * @parent: (allow-none): a #GtkWindow, or %NULL
798 * @flags: flags for this dialog
801 * Creates a new #GtkAppChooserDialog for the provided #GFile,
802 * to allow the user to select an application for it.
804 * Returns: a newly created #GtkAppChooserDialog
809 gtk_app_chooser_dialog_new (GtkWindow *parent,
810 GtkDialogFlags flags,
815 g_return_val_if_fail (G_IS_FILE (file), NULL);
817 retval = g_object_new (GTK_TYPE_APP_CHOOSER_DIALOG,
821 set_parent_and_flags (retval, parent, flags);
827 * gtk_app_chooser_dialog_new_for_content_type:
828 * @parent: (allow-none): a #GtkWindow, or %NULL
829 * @flags: flags for this dialog
830 * @content_type: a content type string
832 * Creates a new #GtkAppChooserDialog for the provided content type,
833 * to allow the user to select an application for it.
835 * Returns: a newly created #GtkAppChooserDialog
840 gtk_app_chooser_dialog_new_for_content_type (GtkWindow *parent,
841 GtkDialogFlags flags,
842 const gchar *content_type)
846 g_return_val_if_fail (content_type != NULL, NULL);
848 retval = g_object_new (GTK_TYPE_APP_CHOOSER_DIALOG,
849 "content-type", content_type,
852 set_parent_and_flags (retval, parent, flags);
858 * gtk_app_chooser_dialog_get_widget:
859 * @self: a #GtkAppChooserDialog
861 * Returns the #GtkAppChooserWidget of this dialog.
863 * Returns: (transfer none): the #GtkAppChooserWidget of @self
868 gtk_app_chooser_dialog_get_widget (GtkAppChooserDialog *self)
870 g_return_val_if_fail (GTK_IS_APP_CHOOSER_DIALOG (self), NULL);
872 return self->priv->app_chooser_widget;
876 * gtk_app_chooser_dialog_set_heading:
877 * @self: a #GtkAppChooserDialog
878 * @heading: a string containing Pango markup
880 * Sets the text to display at the top of the dialog.
881 * If the heading is not set, the dialog displays a default text.
884 gtk_app_chooser_dialog_set_heading (GtkAppChooserDialog *self,
885 const gchar *heading)
887 g_return_if_fail (GTK_IS_APP_CHOOSER_DIALOG (self));
889 g_free (self->priv->heading);
890 self->priv->heading = g_strdup (heading);
892 if (self->priv->label && self->priv->heading)
893 gtk_label_set_markup (GTK_LABEL (self->priv->label), self->priv->heading);
895 g_object_notify (G_OBJECT (self), "heading");
899 * gtk_app_chooser_dialog_get_heading:
900 * @self: a #GtkAppChooserDialog
902 * Returns the text to display at the top of the dialog.
904 * Returns: the text to display at the top of the dialog, or %NULL, in which
905 * case a default text is displayed
908 gtk_app_chooser_dialog_get_heading (GtkAppChooserDialog *self)
910 g_return_val_if_fail (GTK_IS_APP_CHOOSER_DIALOG (self), NULL);
912 return self->priv->heading;