/* * Shortcut.c: implements shortcuts and shortcut group models * * Author: * Miguel de Icaza (miguel@kernel.org) * * (C) 2000 Helix Code, Inc. * */ #include <config.h> #include <gtk/gtksignal.h> #include <gtk/gtkmisc.h> #include <libgnome/libgnome.h> #include "e-util/e-util.h" #include "e-shortcut.h" #include "shortcut-bar/e-shortcut-bar.h" #include "shortcut-bar/e-clipped-label.h" #define SHORTCUT_PARENT_TYPE gtk_object_get_type () #define SHORTCUT_BAR_MODEL_PARENT_TYPE gtk_object_get_type () #define SHORTCUT_GROUP_PARENT_TYPE gtk_object_get_type () static GtkObjectClass *shortcut_parent_class; static GtkObjectClass *shortcut_group_parent_class; static GtkObjectClass *shortcut_bar_model_parent_class; enum { STRUCTURE_CHANGED, LAST_SIGNAL }; static void es_destroy (GtkObject *object) { EShortcut *ef = E_SHORTCUT (object); if (ef->efolder != NULL) gtk_object_unref (GTK_OBJECT (ef->efolder)); shortcut_parent_class->destroy (object); } static void e_shortcut_class_init (GtkObjectClass *object_class) { object_class->destroy = es_destroy; shortcut_parent_class = gtk_type_class (SHORTCUT_PARENT_TYPE); } static void esg_destroy (GtkObject *object) { EShortcutGroup *efg = E_SHORTCUT_GROUP (object); const int shortcut_count = efg->shortcuts->len; int i; g_free (efg->title); for (i = 0; i < shortcut_count; i++){ EShortcut *es = g_array_index (efg->shortcuts, EShortcut *, i); if (es != NULL) gtk_object_unref (GTK_OBJECT (es)); } g_array_free (efg->shortcuts, TRUE); efg->model = NULL; shortcut_group_parent_class->destroy (object); } static void e_shortcut_group_class_init (GtkObjectClass *object_class) { object_class->destroy = esg_destroy; shortcut_parent_class = gtk_type_class (SHORTCUT_GROUP_PARENT_TYPE); } static void e_shortcut_group_init (GtkObject *object) { EShortcutGroup *esg = E_SHORTCUT_GROUP (object); GTK_OBJECT_UNSET_FLAGS (GTK_OBJECT (object), GTK_FLOATING); esg->shortcuts = g_array_new (FALSE, FALSE, sizeof (EShortcut *)); } EShortcut * e_shortcut_new (EFolder *efolder) { EShortcut *shortcut = gtk_type_new (e_shortcut_get_type ()); shortcut->efolder = efolder; gtk_object_ref (GTK_OBJECT (efolder)); return shortcut; } EShortcutGroup * e_shortcut_group_new (const char *title, EIconBarViewType type) { EShortcutGroup *shortcut_group = gtk_type_new (e_shortcut_group_get_type ()); shortcut_group->title = g_strdup (title); shortcut_group->type = type; return shortcut_group; } void e_shortcut_group_append (EShortcutGroup *sg, EShortcut *shortcut) { g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); g_return_if_fail (shortcut != NULL); g_return_if_fail (E_IS_SHORTCUT (shortcut)); gtk_object_ref (GTK_OBJECT (shortcut)); gtk_object_sink (GTK_OBJECT (shortcut)); g_array_append_val (sg->shortcuts, shortcut); /* FIXME: Broadcast change */ } void e_shortcut_group_remove (EShortcutGroup *sg, EShortcut *shortcut) { g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); g_return_if_fail (shortcut != NULL); g_return_if_fail (E_IS_SHORTCUT (sg)); { const int len = sg->shortcuts->len; int i; for (i = 0; i < len; i++){ EShortcut *es = g_array_index (sg->shortcuts, EShortcut *, i); if (es == shortcut){ g_array_remove_index (sg->shortcuts, i); /* FIXME: Broadcast change */ return; } } } } void e_shortcut_group_move (EShortcutGroup *sg, int from, int to) { EShortcut *t; g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); g_return_if_fail (from < sg->shortcuts->len); g_return_if_fail (to < sg->shortcuts->len); g_return_if_fail (from >= 0); g_return_if_fail (to >= 0); if (from == to) return; t = g_array_index (sg->shortcuts, EShortcut *, from); g_array_index (sg->shortcuts, EShortcut *, from) = g_array_index (sg->shortcuts, EShortcut *, to); g_array_index (sg->shortcuts, EShortcut *, to) = t; /* FIXME: Broadcast change */ } void e_shortcut_group_rename (EShortcutGroup *sg, const char *text) { GSList *l; int id; g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); id = e_group_num_from_group_ptr (sg->model, sg); for (l = sg->model->views; l; l = l->next){ EShortcutBar *shortcut_bar = l->data; GtkWidget *label; label = e_clipped_label_new (text); gtk_misc_set_alignment (GTK_MISC (label), 0.5, 0.5); gtk_widget_show (label); e_group_bar_set_group_button_label ( E_GROUP_BAR (shortcut_bar), id, label); } } static void esb_destroy (GtkObject *object) { EShortcutBarModel *esb = E_SHORTCUT_BAR_MODEL (object); const int count = esb->groups->len; int i; for (i = 0; i < count; i++){ EShortcutGroup *esg = g_array_index (esb->groups, EShortcutGroup *, i); gtk_object_destroy (GTK_OBJECT (esg)); } g_array_free (esb->groups, TRUE); shortcut_bar_model_parent_class->destroy (object); } static void e_shortcut_bar_model_class_init (GtkObjectClass *object_class) { object_class->destroy = esb_destroy; shortcut_bar_model_parent_class = gtk_type_class (SHORTCUT_BAR_MODEL_PARENT_TYPE); } static void e_shortcut_bar_model_init (GtkObject *object) { EShortcutBarModel *esb = E_SHORTCUT_BAR_MODEL (object); /* The shortcut bar model is self owned */ GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING); esb->groups = g_array_new (FALSE, FALSE, sizeof (EShortcutGroup *)); } EShortcutBarModel * e_shortcut_bar_model_new (void) { EShortcutBarModel *bm; bm = gtk_type_new (e_shortcut_bar_model_get_type ()); return bm; } void e_shortcut_bar_model_append (EShortcutBarModel *bm, EShortcutGroup *sg) { g_return_if_fail (bm != NULL); g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_BAR_MODEL (bm)); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); gtk_object_ref (GTK_OBJECT (sg)); gtk_object_sink (GTK_OBJECT (sg)); sg->model = bm; g_array_append_val (bm->groups, sg); } EShortcutGroup * e_shortcut_group_from_pos (EShortcutBarModel *bm, int group_num) { EShortcutGroup *group; if (group_num == -1) return NULL; group = g_array_index (bm->groups, EShortcutGroup *, group_num); return group; } EShortcut * e_shortcut_from_pos (EShortcutGroup *group, int item_num) { EShortcut *shortcut; g_return_val_if_fail (group != NULL, NULL); g_return_val_if_fail (E_IS_SHORTCUT_GROUP (group), NULL); if (item_num == -1) return NULL; g_return_val_if_fail (item_num < group->shortcuts->len, NULL); shortcut = g_array_index (group->shortcuts, EShortcut *, item_num); return shortcut; } static void populate_group (EShortcutBarModel *bm, EShortcutGroup *esg, EShortcutBar *shortcut_bar) { int group_num, i; const int items = esg->shortcuts->len; group_num = e_shortcut_bar_add_group (shortcut_bar, esg->title); e_shortcut_bar_set_view_type (shortcut_bar, group_num, esg->type); for (i = 0; i < items; i++){ EShortcut *shortcut = E_SHORTCUT (g_array_index (esg->shortcuts, EShortcut *, i)); EFolder *folder = shortcut->efolder; char *type = NULL; switch (folder->type){ case E_FOLDER_SUMMARY: type = "summary:"; break; case E_FOLDER_MAIL: type = "mail:"; break; case E_FOLDER_CONTACTS: type = "contacts:"; break; case E_FOLDER_CALENDAR: type = "calendar:"; break; case E_FOLDER_TASKS: type = "todo:"; break; case E_FOLDER_OTHER: type = "file:"; break; default: g_assert_not_reached (); } e_shortcut_bar_add_item (shortcut_bar, group_num, type, folder->name); } } static void populate_from_model (EShortcutBarModel *bm, EShortcutBar *shortcut_bar) { const int groups = bm->groups->len; int i; for (i = 0; i < groups; i++){ EShortcutGroup *esg; esg = g_array_index (bm->groups, EShortcutGroup *, i); populate_group (bm, esg, shortcut_bar); } } static struct { char *prefix, *path; GdkPixbuf *image; } shell_icons[] = { { "summary:", "evolution-today.png", NULL }, { "mail:", "evolution-inbox.png", NULL }, { "calendar:", "evolution-calendar.png", NULL }, { "contacts:", "evolution-contacts.png", NULL }, { "notes:", "evolution-notes.png", NULL }, { "todo:", "evolution-tasks.png", NULL } }; #define NSHELL_ICONS (sizeof (shell_icons) / sizeof (shell_icons[0])) static GdkPixbuf * shell_icon_cb (EShortcutBar *shortcut_bar, const gchar *url, gpointer data) { int i; for (i = 0; i < NSHELL_ICONS; i++) { if (!strncmp (shell_icons[i].prefix, url, strlen (shell_icons[i].prefix))) { if (!shell_icons[i].image) { char *pixmap_path; pixmap_path = g_strconcat (EVOLUTION_IMAGES "/", shell_icons[i].path, NULL); if (pixmap_path) shell_icons[i].image = gdk_pixbuf_new_from_file (pixmap_path); else { g_warning ("Couldn't find image: %s", pixmap_path); } g_free (pixmap_path); } return shell_icons[i].image; } } return NULL; } static void view_destroyed (EShortcutBar *shortcut_bar, EShortcutBarModel *bm) { bm->views = g_slist_remove (bm->views, shortcut_bar); } GtkWidget * e_shortcut_bar_view_new (EShortcutBarModel *bm) { GtkWidget *shortcut_bar; gtk_widget_push_visual (gdk_rgb_get_visual ()); gtk_widget_push_colormap (gdk_rgb_get_cmap ()); shortcut_bar = e_shortcut_bar_new (); e_shortcut_bar_set_icon_callback (E_SHORTCUT_BAR (shortcut_bar), shell_icon_cb, NULL); gtk_widget_pop_visual (); gtk_widget_pop_colormap (); populate_from_model (bm, E_SHORTCUT_BAR (shortcut_bar)); bm->views = g_slist_prepend (bm->views, shortcut_bar); gtk_signal_connect (GTK_OBJECT (shortcut_bar), "destroy", GTK_SIGNAL_FUNC (view_destroyed), bm); return shortcut_bar; } int e_group_num_from_group_ptr (EShortcutBarModel *bm, EShortcutGroup *group) { const int n = bm->groups->len; int i; for (i = 0; i < n; i++) if (g_array_index (bm->groups, EShortcutGroup *, i) == group) return i; return -1; } /* * Sets the view mode in all the views */ void e_shortcut_group_set_view_type (EShortcutGroup *group, EIconBarViewType type) { GSList *l; int group_num; g_return_if_fail (group != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (group)); group_num = e_group_num_from_group_ptr (group->model, group); g_assert (group_num != -1); group->type = type; for (l = group->model->views; l ; l = l->next){ EShortcutBar *shortcut_bar = l->data; e_shortcut_bar_set_view_type (shortcut_bar, group_num, type); } } gint e_shortcut_bar_model_add_group (EShortcutBarModel *model) { int id = -1; GSList *l = NULL; g_return_val_if_fail (model != NULL, -1); g_return_val_if_fail (E_IS_SHORTCUT_BAR_MODEL (model), -1); for (l = model->views; l; l = l->next){ EShortcutBar *shortcut_bar = l->data; id = e_shortcut_bar_add_group (shortcut_bar, _("New group")); } return id; } void e_shortcut_bar_model_remove_group (EShortcutBarModel *model, EShortcutGroup *sg) { GSList *l = NULL; int group_num; g_return_if_fail (model != NULL); g_return_if_fail (E_IS_SHORTCUT_BAR_MODEL (model)); g_return_if_fail (sg != NULL); g_return_if_fail (E_IS_SHORTCUT_GROUP (sg)); group_num = e_group_num_from_group_ptr (model, sg); for (l = model->views; l; l = l->next){ EShortcutBar *shortcut_bar = l->data; e_shortcut_bar_remove_group (shortcut_bar, group_num); } } E_MAKE_TYPE (e_shortcut, "EShortcut", EShortcut, e_shortcut_class_init, NULL, SHORTCUT_PARENT_TYPE); E_MAKE_TYPE (e_shortcut_group, "EShortcutGroup", EShortcutGroup, e_shortcut_group_class_init, e_shortcut_group_init, SHORTCUT_GROUP_PARENT_TYPE); E_MAKE_TYPE (e_shortcut_bar_model, "EShortcutBarModel", EShortcutBarModel, e_shortcut_bar_model_class_init, e_shortcut_bar_model_init, SHORTCUT_BAR_MODEL_PARENT_TYPE);