2010-11-24 17:45:42 +00:00
|
|
|
/*
|
2010-11-29 11:19:18 +00:00
|
|
|
* gtkappchooserbutton.h: an app-chooser combobox
|
2010-11-24 17:45:42 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 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: Cosimo Cecchi <ccecchi@redhat.com>
|
|
|
|
*/
|
|
|
|
|
2010-11-24 23:32:05 +00:00
|
|
|
#include "config.h"
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
#include "gtkappchooserbutton.h"
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
#include "gtkappchooser.h"
|
2010-11-25 16:17:45 +00:00
|
|
|
#include "gtkappchooserdialog.h"
|
2010-11-24 17:45:42 +00:00
|
|
|
#include "gtkappchooserprivate.h"
|
|
|
|
#include "gtkcelllayout.h"
|
|
|
|
#include "gtkcellrendererpixbuf.h"
|
|
|
|
#include "gtkcellrenderertext.h"
|
|
|
|
#include "gtkcombobox.h"
|
2010-11-25 16:17:45 +00:00
|
|
|
#include "gtkdialog.h"
|
|
|
|
#include "gtkintl.h"
|
2010-11-29 15:04:59 +00:00
|
|
|
#include "gtkmarshalers.h"
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
PROP_CONTENT_TYPE = 1,
|
2010-11-25 16:17:45 +00:00
|
|
|
PROP_SHOW_DIALOG_ITEM,
|
2010-11-24 17:45:42 +00:00
|
|
|
};
|
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
enum {
|
|
|
|
SIGNAL_CUSTOM_ITEM_ACTIVATED,
|
|
|
|
NUM_SIGNALS
|
|
|
|
};
|
|
|
|
|
2010-11-24 17:45:42 +00:00
|
|
|
enum {
|
|
|
|
COLUMN_APP_INFO,
|
|
|
|
COLUMN_NAME,
|
2010-11-29 15:04:59 +00:00
|
|
|
COLUMN_LABEL,
|
2010-11-24 17:45:42 +00:00
|
|
|
COLUMN_ICON,
|
|
|
|
COLUMN_CUSTOM,
|
|
|
|
COLUMN_SEPARATOR,
|
|
|
|
NUM_COLUMNS,
|
|
|
|
};
|
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
#define CUSTOM_ITEM_OTHER_APP "gtk-internal-item-other-app"
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
static void app_chooser_iface_init (GtkAppChooserIface *iface);
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
static void real_insert_custom_item (GtkAppChooserButton *self,
|
2010-11-29 15:04:59 +00:00
|
|
|
const gchar *name,
|
2010-11-29 15:10:58 +00:00
|
|
|
const gchar *label,
|
|
|
|
GIcon *icon,
|
|
|
|
gboolean custom,
|
|
|
|
GtkTreeIter *iter);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
static void real_insert_separator (GtkAppChooserButton *self,
|
2010-11-29 15:10:58 +00:00
|
|
|
gboolean custom,
|
|
|
|
GtkTreeIter *iter);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
static guint signals[NUM_SIGNALS] = { 0, };
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkAppChooserButton, gtk_app_chooser_button, GTK_TYPE_COMBO_BOX,
|
2010-11-24 23:32:05 +00:00
|
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_APP_CHOOSER,
|
|
|
|
app_chooser_iface_init));
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
struct _GtkAppChooserButtonPrivate {
|
2010-11-24 17:45:42 +00:00
|
|
|
GtkListStore *store;
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
gchar *content_type;
|
|
|
|
gboolean show_dialog_item;
|
2010-11-29 15:04:59 +00:00
|
|
|
|
|
|
|
GHashTable *custom_item_names;
|
2010-11-24 17:45:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
row_separator_func (GtkTreeModel *model,
|
2010-11-24 23:32:05 +00:00
|
|
|
GtkTreeIter *iter,
|
|
|
|
gpointer user_data)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
gboolean separator;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, iter,
|
2010-11-24 23:32:05 +00:00
|
|
|
COLUMN_SEPARATOR, &separator,
|
|
|
|
-1);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
return separator;
|
|
|
|
}
|
|
|
|
|
2010-11-24 18:08:17 +00:00
|
|
|
static void
|
|
|
|
get_first_iter (GtkListStore *store,
|
2010-11-24 23:32:05 +00:00
|
|
|
GtkTreeIter *iter)
|
2010-11-24 18:08:17 +00:00
|
|
|
{
|
|
|
|
GtkTreeIter iter2;
|
|
|
|
|
|
|
|
if (!gtk_tree_model_get_iter_first (GTK_TREE_MODEL (store), iter))
|
|
|
|
{
|
|
|
|
/* the model is empty, append */
|
|
|
|
gtk_list_store_append (store, iter);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_list_store_insert_before (store, &iter2, iter);
|
|
|
|
*iter = iter2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-25 16:17:45 +00:00
|
|
|
typedef struct {
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self;
|
2010-11-25 16:17:45 +00:00
|
|
|
GAppInfo *info;
|
|
|
|
gint active_index;
|
|
|
|
} SelectAppData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
select_app_data_free (SelectAppData *data)
|
|
|
|
{
|
|
|
|
g_clear_object (&data->self);
|
|
|
|
g_clear_object (&data->info);
|
|
|
|
|
|
|
|
g_slice_free (SelectAppData, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
select_application_func_cb (GtkTreeModel *model,
|
2010-11-29 15:10:58 +00:00
|
|
|
GtkTreePath *path,
|
|
|
|
GtkTreeIter *iter,
|
|
|
|
gpointer user_data)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
|
|
|
SelectAppData *data = user_data;
|
|
|
|
GAppInfo *app_to_match = data->info, *app = NULL;
|
|
|
|
gboolean custom;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, iter,
|
2010-11-29 15:10:58 +00:00
|
|
|
COLUMN_APP_INFO, &app,
|
|
|
|
COLUMN_CUSTOM, &custom,
|
|
|
|
-1);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
/* cutsom items are always after GAppInfos, so iterating further here
|
|
|
|
* is just useless.
|
|
|
|
*/
|
|
|
|
if (custom)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (g_app_info_equal (app, app_to_match))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (data->self), iter);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_select_application (GtkAppChooserButton *self,
|
|
|
|
GAppInfo *info)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
|
|
|
SelectAppData *data;
|
|
|
|
|
|
|
|
data = g_slice_new0 (SelectAppData);
|
|
|
|
data->self = g_object_ref (self);
|
|
|
|
data->info = g_object_ref (info);
|
|
|
|
|
|
|
|
gtk_tree_model_foreach (GTK_TREE_MODEL (self->priv->store),
|
2010-11-29 15:10:58 +00:00
|
|
|
select_application_func_cb, data);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
select_app_data_free (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
other_application_dialog_response_cb (GtkDialog *dialog,
|
2010-11-29 15:10:58 +00:00
|
|
|
gint response_id,
|
|
|
|
gpointer user_data)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = user_data;
|
2010-11-25 16:17:45 +00:00
|
|
|
GAppInfo *info;
|
|
|
|
|
|
|
|
if (response_id != GTK_RESPONSE_OK)
|
|
|
|
{
|
|
|
|
/* reset the active item, otherwise we are stuck on
|
|
|
|
* 'Other application...'
|
|
|
|
*/
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (self), 0);
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
info = gtk_app_chooser_get_app_info (GTK_APP_CHOOSER (dialog));
|
|
|
|
|
|
|
|
/* refresh the combobox to get the new application */
|
|
|
|
gtk_app_chooser_refresh (GTK_APP_CHOOSER (self));
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_select_application (self, info);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
g_object_unref (info);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 15:04:59 +00:00
|
|
|
other_application_item_activated_cb (GtkAppChooserButton *self)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
|
|
|
GtkWidget *dialog, *widget;
|
|
|
|
GtkWindow *toplevel;
|
|
|
|
|
|
|
|
toplevel = GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (self)));
|
|
|
|
dialog = gtk_app_chooser_dialog_new_for_content_type (toplevel, GTK_DIALOG_DESTROY_WITH_PARENT,
|
2010-11-29 15:10:58 +00:00
|
|
|
self->priv->content_type);
|
2010-11-25 16:17:45 +00:00
|
|
|
widget = gtk_app_chooser_dialog_get_widget (GTK_APP_CHOOSER_DIALOG (dialog));
|
|
|
|
g_object_set (widget,
|
2010-11-29 15:10:58 +00:00
|
|
|
"show-fallback", TRUE,
|
|
|
|
"show-other", TRUE,
|
|
|
|
NULL);
|
2010-11-25 16:17:45 +00:00
|
|
|
gtk_widget_show (dialog);
|
|
|
|
|
|
|
|
g_signal_connect (dialog, "response",
|
2010-11-29 15:10:58 +00:00
|
|
|
G_CALLBACK (other_application_dialog_response_cb), self);
|
2010-11-25 16:17:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_ensure_dialog_item (GtkAppChooserButton *self,
|
|
|
|
GtkTreeIter *prev_iter)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
|
|
|
GIcon *icon;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
if (!self->priv->show_dialog_item)
|
|
|
|
return;
|
|
|
|
|
|
|
|
icon = g_themed_icon_new ("application-x-executable");
|
|
|
|
|
|
|
|
gtk_list_store_insert_after (self->priv->store, &iter, prev_iter);
|
|
|
|
real_insert_separator (self, FALSE, &iter);
|
|
|
|
*prev_iter = iter;
|
|
|
|
|
|
|
|
gtk_list_store_insert_after (self->priv->store, &iter, prev_iter);
|
2010-11-29 15:04:59 +00:00
|
|
|
real_insert_custom_item (self, CUSTOM_ITEM_OTHER_APP,
|
2010-11-29 15:10:58 +00:00
|
|
|
_("Other application..."), icon,
|
2010-11-29 15:04:59 +00:00
|
|
|
FALSE, &iter);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
g_object_unref (icon);
|
|
|
|
}
|
|
|
|
|
2010-11-24 17:45:42 +00:00
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_populate (GtkAppChooserButton *self)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GList *recommended_apps = NULL, *l;
|
|
|
|
GAppInfo *app;
|
2010-11-24 18:08:17 +00:00
|
|
|
GtkTreeIter iter, iter2;
|
2010-11-24 17:45:42 +00:00
|
|
|
GIcon *icon;
|
2010-11-24 18:08:17 +00:00
|
|
|
gboolean first;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
recommended_apps = g_app_info_get_recommended_for_type (self->priv->content_type);
|
2010-11-24 18:08:17 +00:00
|
|
|
first = TRUE;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-12-01 15:29:57 +00:00
|
|
|
get_first_iter (self->priv->store, &iter);
|
|
|
|
|
2010-11-24 17:45:42 +00:00
|
|
|
for (l = recommended_apps; l != NULL; l = l->next)
|
|
|
|
{
|
|
|
|
app = l->data;
|
|
|
|
|
|
|
|
icon = g_app_info_get_icon (app);
|
|
|
|
|
|
|
|
if (icon == NULL)
|
2010-11-24 23:32:05 +00:00
|
|
|
icon = g_themed_icon_new ("application-x-executable");
|
2010-11-24 17:45:42 +00:00
|
|
|
else
|
2010-11-24 23:32:05 +00:00
|
|
|
g_object_ref (icon);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-24 18:08:17 +00:00
|
|
|
if (first)
|
2010-11-24 23:32:05 +00:00
|
|
|
{
|
|
|
|
first = FALSE;
|
|
|
|
}
|
2010-11-24 18:08:17 +00:00
|
|
|
else
|
2010-11-24 23:32:05 +00:00
|
|
|
{
|
|
|
|
gtk_list_store_insert_after (self->priv->store, &iter2, &iter);
|
|
|
|
iter = iter2;
|
|
|
|
}
|
2010-11-24 18:08:17 +00:00
|
|
|
|
2010-11-24 17:45:42 +00:00
|
|
|
gtk_list_store_set (self->priv->store, &iter,
|
2010-11-24 23:32:05 +00:00
|
|
|
COLUMN_APP_INFO, app,
|
2010-11-29 15:04:59 +00:00
|
|
|
COLUMN_LABEL, g_app_info_get_display_name (app),
|
2010-11-24 23:32:05 +00:00
|
|
|
COLUMN_ICON, icon,
|
|
|
|
COLUMN_CUSTOM, FALSE,
|
|
|
|
-1);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
g_object_unref (icon);
|
|
|
|
}
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_ensure_dialog_item (self, &iter);
|
2010-11-24 17:45:42 +00:00
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (self), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_build_ui (GtkAppChooserButton *self)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GtkCellRenderer *cell;
|
|
|
|
|
|
|
|
self->priv->store = gtk_list_store_new (NUM_COLUMNS,
|
2010-11-24 23:32:05 +00:00
|
|
|
G_TYPE_APP_INFO,
|
2010-11-29 15:04:59 +00:00
|
|
|
G_TYPE_STRING, /* name */
|
|
|
|
G_TYPE_STRING, /* label */
|
2010-11-24 23:32:05 +00:00
|
|
|
G_TYPE_ICON,
|
2010-11-29 15:04:59 +00:00
|
|
|
G_TYPE_BOOLEAN, /* separator */
|
|
|
|
G_TYPE_BOOLEAN); /* custom */
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
gtk_combo_box_set_model (GTK_COMBO_BOX (self),
|
2010-11-24 23:32:05 +00:00
|
|
|
GTK_TREE_MODEL (self->priv->store));
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
gtk_combo_box_set_row_separator_func (GTK_COMBO_BOX (self),
|
2010-11-24 23:32:05 +00:00
|
|
|
row_separator_func, NULL, NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
cell = gtk_cell_renderer_pixbuf_new ();
|
|
|
|
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (self), cell, FALSE);
|
|
|
|
gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (self), cell,
|
2010-11-24 23:32:05 +00:00
|
|
|
"gicon", COLUMN_ICON,
|
|
|
|
NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
cell = gtk_cell_renderer_text_new ();
|
|
|
|
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (self), cell, TRUE);
|
|
|
|
gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (self), cell,
|
2010-11-29 15:04:59 +00:00
|
|
|
"text", COLUMN_LABEL,
|
2010-11-24 23:32:05 +00:00
|
|
|
NULL);
|
2010-11-29 15:04:59 +00:00
|
|
|
g_object_set (cell,
|
|
|
|
"xpad", 6,
|
|
|
|
NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_populate (self);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_remove_non_custom (GtkAppChooserButton *self)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gboolean custom, res;
|
|
|
|
|
2010-11-24 23:32:05 +00:00
|
|
|
model = GTK_TREE_MODEL (self->priv->store);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
if (!gtk_tree_model_get_iter_first (model, &iter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
do {
|
|
|
|
gtk_tree_model_get (model, &iter,
|
2010-11-24 23:32:05 +00:00
|
|
|
COLUMN_CUSTOM, &custom,
|
|
|
|
-1);
|
2010-11-24 17:45:42 +00:00
|
|
|
if (custom)
|
|
|
|
res = gtk_tree_model_iter_next (model, &iter);
|
2010-11-24 18:08:17 +00:00
|
|
|
else
|
|
|
|
res = gtk_list_store_remove (GTK_LIST_STORE (model), &iter);
|
2010-11-24 17:45:42 +00:00
|
|
|
} while (res);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_changed (GtkComboBox *object)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (object);
|
2010-11-24 17:45:42 +00:00
|
|
|
GtkTreeIter iter;
|
2010-11-29 15:04:59 +00:00
|
|
|
gchar *name = NULL;
|
|
|
|
gboolean custom;
|
|
|
|
GQuark name_quark;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-24 18:08:17 +00:00
|
|
|
if (!gtk_combo_box_get_active_iter (object, &iter))
|
|
|
|
return;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
gtk_tree_model_get (GTK_TREE_MODEL (self->priv->store), &iter,
|
2010-11-29 15:04:59 +00:00
|
|
|
COLUMN_NAME, &name,
|
|
|
|
COLUMN_CUSTOM, &custom,
|
2010-11-24 23:32:05 +00:00
|
|
|
-1);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
if (name != NULL)
|
|
|
|
{
|
|
|
|
if (custom)
|
|
|
|
{
|
|
|
|
name_quark = g_quark_from_string (name);
|
|
|
|
g_signal_emit (self, signals[SIGNAL_CUSTOM_ITEM_ACTIVATED], name_quark, name);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* trigger the dialog internally */
|
|
|
|
other_application_item_activated_cb (self);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (name);
|
|
|
|
}
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_refresh (GtkAppChooser *object)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (object);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_remove_non_custom (self);
|
|
|
|
gtk_app_chooser_button_populate (self);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GAppInfo *
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_get_app_info (GtkAppChooser *object)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (object);
|
2010-11-24 17:45:42 +00:00
|
|
|
GtkTreeIter iter;
|
|
|
|
GAppInfo *info;
|
|
|
|
|
|
|
|
if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (self), &iter))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
gtk_tree_model_get (GTK_TREE_MODEL (self->priv->store), &iter,
|
2010-11-24 23:32:05 +00:00
|
|
|
COLUMN_APP_INFO, &info,
|
|
|
|
-1);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_constructed (GObject *obj)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
if (G_OBJECT_CLASS (gtk_app_chooser_button_parent_class)->constructed != NULL)
|
|
|
|
G_OBJECT_CLASS (gtk_app_chooser_button_parent_class)->constructed (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
g_assert (self->priv->content_type != NULL);
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_build_ui (self);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_set_property (GObject *obj,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_CONTENT_TYPE:
|
|
|
|
self->priv->content_type = g_value_dup_string (value);
|
|
|
|
break;
|
2010-11-25 16:17:45 +00:00
|
|
|
case PROP_SHOW_DIALOG_ITEM:
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_set_show_dialog_item (self, g_value_get_boolean (value));
|
2010-11-25 16:17:45 +00:00
|
|
|
break;
|
2010-11-24 17:45:42 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_get_property (GObject *obj,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_CONTENT_TYPE:
|
|
|
|
g_value_set_string (value, self->priv->content_type);
|
|
|
|
break;
|
2010-11-25 16:17:45 +00:00
|
|
|
case PROP_SHOW_DIALOG_ITEM:
|
|
|
|
g_value_set_boolean (value, self->priv->show_dialog_item);
|
|
|
|
break;
|
2010-11-24 17:45:42 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_finalize (GObject *obj)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
GtkAppChooserButton *self = GTK_APP_CHOOSER_BUTTON (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
g_hash_table_destroy (self->priv->custom_item_names);
|
2010-11-24 17:45:42 +00:00
|
|
|
g_free (self->priv->content_type);
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
G_OBJECT_CLASS (gtk_app_chooser_button_parent_class)->finalize (obj);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
app_chooser_iface_init (GtkAppChooserIface *iface)
|
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
iface->get_app_info = gtk_app_chooser_button_get_app_info;
|
|
|
|
iface->refresh = gtk_app_chooser_button_refresh;
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_class_init (GtkAppChooserButtonClass *klass)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GObjectClass *oclass = G_OBJECT_CLASS (klass);
|
|
|
|
GtkComboBoxClass *combo_class = GTK_COMBO_BOX_CLASS (klass);
|
2010-11-25 16:17:45 +00:00
|
|
|
GParamSpec *pspec;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
oclass->set_property = gtk_app_chooser_button_set_property;
|
|
|
|
oclass->get_property = gtk_app_chooser_button_get_property;
|
|
|
|
oclass->finalize = gtk_app_chooser_button_finalize;
|
|
|
|
oclass->constructed = gtk_app_chooser_button_constructed;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
combo_class->changed = gtk_app_chooser_button_changed;
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
g_object_class_override_property (oclass, PROP_CONTENT_TYPE, "content-type");
|
|
|
|
|
2010-11-29 11:25:52 +00:00
|
|
|
/**
|
|
|
|
* GtkAppChooserButton:show-dialog-item:
|
|
|
|
*
|
2010-11-30 16:27:53 +00:00
|
|
|
* The #GtkAppChooserButton:show-dialog-item property determines whether the dropdown menu
|
2010-11-29 11:25:52 +00:00
|
|
|
* should show an item that triggers a #GtkAppChooserDialog when clicked.
|
|
|
|
*/
|
2010-11-29 15:10:58 +00:00
|
|
|
pspec =
|
|
|
|
g_param_spec_boolean ("show-dialog-item",
|
|
|
|
P_("Include an 'Other...' item"),
|
|
|
|
P_("Whether the combobox should include an item that triggers a GtkAppChooserDialog"),
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
|
2010-11-25 16:17:45 +00:00
|
|
|
g_object_class_install_property (oclass, PROP_SHOW_DIALOG_ITEM, pspec);
|
|
|
|
|
2010-11-30 14:03:33 +00:00
|
|
|
/**
|
|
|
|
* GtkAppChooserButton::custom-item-activated:
|
|
|
|
* @self: the object which received the signal
|
|
|
|
* @item_name: the name of the activated item
|
|
|
|
*
|
|
|
|
* Emitted when a custom item, previously added with
|
|
|
|
* gtk_app_chooser_button_append_custom_item(), is activated from the
|
|
|
|
* dropdown menu.
|
|
|
|
*/
|
2010-11-29 15:04:59 +00:00
|
|
|
signals[SIGNAL_CUSTOM_ITEM_ACTIVATED] =
|
|
|
|
g_signal_new ("custom-item-activated",
|
|
|
|
GTK_TYPE_APP_CHOOSER_BUTTON,
|
|
|
|
G_SIGNAL_RUN_FIRST | G_SIGNAL_DETAILED,
|
|
|
|
G_STRUCT_OFFSET (GtkAppChooserButtonClass, custom_item_activated),
|
|
|
|
NULL, NULL,
|
|
|
|
_gtk_marshal_VOID__STRING,
|
|
|
|
G_TYPE_NONE,
|
|
|
|
1, G_TYPE_STRING);
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
g_type_class_add_private (klass, sizeof (GtkAppChooserButtonPrivate));
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_init (GtkAppChooserButton *self)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, GTK_TYPE_APP_CHOOSER_BUTTON,
|
|
|
|
GtkAppChooserButtonPrivate);
|
2010-11-29 15:04:59 +00:00
|
|
|
self->priv->custom_item_names =
|
|
|
|
g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
|
|
g_free, NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
2010-12-01 11:12:03 +00:00
|
|
|
static gboolean
|
|
|
|
app_chooser_button_iter_from_custom_name (GtkAppChooserButton *self,
|
|
|
|
const gchar *name,
|
|
|
|
GtkTreeIter *set_me)
|
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gchar *custom_name = NULL;
|
|
|
|
|
|
|
|
if (!gtk_tree_model_get_iter_first
|
|
|
|
(GTK_TREE_MODEL (self->priv->store), &iter))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
do {
|
|
|
|
gtk_tree_model_get (GTK_TREE_MODEL (self->priv->store), &iter,
|
|
|
|
COLUMN_NAME, &custom_name,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
if (g_strcmp0 (custom_name, name) == 0)
|
|
|
|
{
|
|
|
|
g_free (custom_name);
|
|
|
|
*set_me = iter;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (custom_name);
|
|
|
|
} while (gtk_tree_model_iter_next (GTK_TREE_MODEL (self->priv->store), &iter));
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-11-25 16:17:45 +00:00
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
real_insert_custom_item (GtkAppChooserButton *self,
|
2010-11-29 15:04:59 +00:00
|
|
|
const gchar *name,
|
2010-11-25 16:17:45 +00:00
|
|
|
const gchar *label,
|
|
|
|
GIcon *icon,
|
|
|
|
gboolean custom,
|
|
|
|
GtkTreeIter *iter)
|
|
|
|
{
|
2010-11-29 15:04:59 +00:00
|
|
|
if (custom)
|
|
|
|
{
|
|
|
|
if (g_hash_table_lookup (self->priv->custom_item_names,
|
|
|
|
name) != NULL)
|
|
|
|
{
|
|
|
|
g_warning ("Attempting to add custom item %s to GtkAppChooserButton, "
|
|
|
|
"when there's already an item with the same name", name);
|
|
|
|
return;
|
|
|
|
}
|
2010-11-25 16:17:45 +00:00
|
|
|
|
2010-11-29 15:04:59 +00:00
|
|
|
g_hash_table_insert (self->priv->custom_item_names,
|
|
|
|
g_strdup (name), GINT_TO_POINTER (1));
|
|
|
|
}
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
gtk_list_store_set (self->priv->store, iter,
|
2010-11-29 15:10:58 +00:00
|
|
|
COLUMN_NAME, name,
|
2010-11-29 15:04:59 +00:00
|
|
|
COLUMN_LABEL, label,
|
2010-11-29 15:10:58 +00:00
|
|
|
COLUMN_ICON, icon,
|
|
|
|
COLUMN_CUSTOM, custom,
|
|
|
|
COLUMN_SEPARATOR, FALSE,
|
|
|
|
-1);
|
2010-11-25 16:17:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-11-29 11:19:18 +00:00
|
|
|
real_insert_separator (GtkAppChooserButton *self,
|
2010-11-25 16:17:45 +00:00
|
|
|
gboolean custom,
|
|
|
|
GtkTreeIter *iter)
|
|
|
|
{
|
|
|
|
gtk_list_store_set (self->priv->store, iter,
|
|
|
|
COLUMN_CUSTOM, custom,
|
|
|
|
COLUMN_SEPARATOR, TRUE,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
|
2010-11-24 23:32:05 +00:00
|
|
|
/**
|
2010-11-29 11:19:18 +00:00
|
|
|
* gtk_app_chooser_button_new:
|
2010-11-24 23:32:05 +00:00
|
|
|
* @content_type: the content type to show applications for
|
|
|
|
*
|
2010-11-29 11:19:18 +00:00
|
|
|
* Creates a new #GtkAppChooserButton for applications
|
2010-11-24 23:32:05 +00:00
|
|
|
* that can handle content of the given type.
|
|
|
|
*
|
2010-11-29 11:19:18 +00:00
|
|
|
* Returns: a newly created #GtkAppChooserButton
|
2010-11-24 23:32:05 +00:00
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
2010-11-24 17:45:42 +00:00
|
|
|
GtkWidget *
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_new (const gchar *content_type)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (content_type != NULL, NULL);
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
return g_object_new (GTK_TYPE_APP_CHOOSER_BUTTON,
|
2010-11-24 23:32:05 +00:00
|
|
|
"content-type", content_type,
|
|
|
|
NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
2010-11-24 23:32:05 +00:00
|
|
|
/**
|
2010-11-29 11:19:18 +00:00
|
|
|
* gtk_app_chooser_button_append_separator:
|
|
|
|
* @self: a #GtkAppChooserButton
|
2010-11-24 23:32:05 +00:00
|
|
|
*
|
|
|
|
* Appends a separator to the list of applications that is shown
|
|
|
|
* in the popup.
|
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
2010-11-24 17:45:42 +00:00
|
|
|
void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_append_separator (GtkAppChooserButton *self)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
g_return_if_fail (GTK_IS_APP_CHOOSER_BUTTON (self));
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
gtk_list_store_append (self->priv->store, &iter);
|
2010-11-25 16:17:45 +00:00
|
|
|
real_insert_separator (self, TRUE, &iter);
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|
|
|
|
|
2010-11-24 23:32:05 +00:00
|
|
|
/**
|
2010-11-29 11:19:18 +00:00
|
|
|
* gtk_app_chooser_button_append_custom_item:
|
|
|
|
* @self: a #GtkAppChooserButton
|
2010-11-29 15:04:59 +00:00
|
|
|
* @name: the name of the custom item
|
2010-11-24 23:32:05 +00:00
|
|
|
* @label: the label for the custom item
|
|
|
|
* @icon: the icon for the custom item
|
|
|
|
*
|
|
|
|
* Appends a custom item to the list of applications that is shown
|
2010-11-29 15:04:59 +00:00
|
|
|
* in the popup; the item name must be unique per-widget.
|
|
|
|
* Clients can use the provided name as a detail for the ::custom-item-activated
|
|
|
|
* signal, to add a callback for the activation of a particular
|
|
|
|
* custom item in the list.
|
|
|
|
* See also gtk_app_chooser_button_append_separator().
|
2010-11-24 23:32:05 +00:00
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
2010-11-24 17:45:42 +00:00
|
|
|
void
|
2010-11-29 15:04:59 +00:00
|
|
|
gtk_app_chooser_button_append_custom_item (GtkAppChooserButton *self,
|
|
|
|
const gchar *name,
|
|
|
|
const gchar *label,
|
|
|
|
GIcon *icon)
|
2010-11-24 17:45:42 +00:00
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
2010-11-29 11:19:18 +00:00
|
|
|
g_return_if_fail (GTK_IS_APP_CHOOSER_BUTTON (self));
|
2010-11-29 15:04:59 +00:00
|
|
|
g_return_if_fail (name != NULL);
|
2010-11-24 17:45:42 +00:00
|
|
|
|
|
|
|
gtk_list_store_append (self->priv->store, &iter);
|
2010-11-29 15:04:59 +00:00
|
|
|
real_insert_custom_item (self, name, label, icon, TRUE, &iter);
|
2010-11-25 16:17:45 +00:00
|
|
|
}
|
|
|
|
|
2010-12-01 11:12:03 +00:00
|
|
|
/**
|
|
|
|
* gtk_app_chooser_button_select_custom_item:
|
|
|
|
* @self: a #GtkAppChooserButton
|
|
|
|
* @name: the name of the custom item
|
|
|
|
*
|
|
|
|
* Selects a custom item previously added with
|
|
|
|
* gtk_app_chooser_button_append_custom_item().
|
2010-12-01 15:33:34 +00:00
|
|
|
* Use gtk_app_chooser_refresh() to bring the selection to its initial
|
|
|
|
* state.
|
2010-12-01 11:12:03 +00:00
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_app_chooser_button_set_active_custom_item (GtkAppChooserButton *self,
|
|
|
|
const gchar *name)
|
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_APP_CHOOSER_BUTTON (self));
|
|
|
|
g_return_if_fail (name != NULL);
|
|
|
|
|
|
|
|
if (g_hash_table_lookup (self->priv->custom_item_names, name) == NULL ||
|
|
|
|
!app_chooser_button_iter_from_custom_name (self, name, &iter))
|
|
|
|
{
|
|
|
|
g_warning ("Can't find the item named %s in the app chooser.",
|
|
|
|
name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (self), &iter);
|
|
|
|
}
|
|
|
|
|
2010-11-29 11:25:52 +00:00
|
|
|
/**
|
|
|
|
* gtk_app_chooser_button_get_show_dialog_item:
|
|
|
|
* @self: a #GtkAppChooserButton
|
|
|
|
*
|
|
|
|
* Returns the current value of the #GtkAppChooserButton:show-dialog-item
|
|
|
|
* property.
|
|
|
|
*
|
|
|
|
* Returns: the value of #GtkAppChooserButton:show-dialog-item
|
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
2010-11-25 16:17:45 +00:00
|
|
|
gboolean
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_get_show_dialog_item (GtkAppChooserButton *self)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
2010-11-29 11:19:18 +00:00
|
|
|
g_return_val_if_fail (GTK_IS_APP_CHOOSER_BUTTON (self), FALSE);
|
2010-11-25 16:17:45 +00:00
|
|
|
|
|
|
|
return self->priv->show_dialog_item;
|
|
|
|
}
|
|
|
|
|
2010-11-29 11:25:52 +00:00
|
|
|
/**
|
2010-11-30 16:27:53 +00:00
|
|
|
* gtk_app_chooser_button_set_show_dialog_item:
|
2010-11-29 11:25:52 +00:00
|
|
|
* @self: a #GtkAppChooserButton
|
|
|
|
* @setting: the new value for #GtkAppChooserButton:show-dialog-item
|
|
|
|
*
|
|
|
|
* Sets whether the dropdown menu of this button should show an
|
|
|
|
* entry to trigger a #GtkAppChooserDialog.
|
|
|
|
*
|
|
|
|
* Since: 3.0
|
|
|
|
*/
|
2010-11-25 16:17:45 +00:00
|
|
|
void
|
2010-11-29 11:19:18 +00:00
|
|
|
gtk_app_chooser_button_set_show_dialog_item (GtkAppChooserButton *self,
|
|
|
|
gboolean setting)
|
2010-11-25 16:17:45 +00:00
|
|
|
{
|
|
|
|
if (self->priv->show_dialog_item != setting)
|
|
|
|
{
|
|
|
|
self->priv->show_dialog_item = setting;
|
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (self), "show-dialog-item");
|
|
|
|
|
|
|
|
gtk_app_chooser_refresh (GTK_APP_CHOOSER (self));
|
|
|
|
}
|
2010-11-24 17:45:42 +00:00
|
|
|
}
|