aboutsummaryrefslogtreecommitdiffstats
path: root/addressbook/gui
diff options
context:
space:
mode:
Diffstat (limited to 'addressbook/gui')
-rw-r--r--addressbook/gui/component/addressbook-component.c10
-rw-r--r--addressbook/gui/component/addressbook-view.c15
-rw-r--r--addressbook/gui/widgets/Makefile.am2
-rw-r--r--addressbook/gui/widgets/e-addressbook-view.c23
-rw-r--r--addressbook/gui/widgets/e-addressbook-view.h5
-rw-r--r--addressbook/gui/widgets/eab-menu.c257
-rw-r--r--addressbook/gui/widgets/eab-menu.h109
-rw-r--r--addressbook/gui/widgets/eab-popup.c13
-rw-r--r--addressbook/gui/widgets/eab-popup.h8
9 files changed, 438 insertions, 4 deletions
diff --git a/addressbook/gui/component/addressbook-component.c b/addressbook/gui/component/addressbook-component.c
index befd07098d..9e71580d0b 100644
--- a/addressbook/gui/component/addressbook-component.c
+++ b/addressbook/gui/component/addressbook-component.c
@@ -31,6 +31,9 @@
#include "addressbook-view.h"
#include "addressbook/gui/contact-editor/eab-editor.h"
#include "addressbook/gui/widgets/eab-gui-util.h"
+#include "e-util/e-plugin.h"
+#include "addressbook/gui/widgets/eab-popup.h"
+#include "addressbook/gui/widgets/eab-menu.h"
#include "widgets/misc/e-task-bar.h"
#include "widgets/misc/e-info-label.h"
@@ -246,6 +249,7 @@ static void
addressbook_component_init (AddressbookComponent *component)
{
AddressbookComponentPrivate *priv;
+ static int first = TRUE;
priv = g_new0 (AddressbookComponentPrivate, 1);
@@ -259,6 +263,12 @@ addressbook_component_init (AddressbookComponent *component)
#ifdef ENABLE_SMIME
smime_component_init ();
#endif
+
+ if (first) {
+ first = FALSE;
+ e_plugin_hook_register_type(eab_popup_hook_get_type());
+ e_plugin_hook_register_type(eab_menu_hook_get_type());
+ }
}
diff --git a/addressbook/gui/component/addressbook-view.c b/addressbook/gui/component/addressbook-view.c
index bd5b3b3c1e..ef4a24b5e0 100644
--- a/addressbook/gui/component/addressbook-view.c
+++ b/addressbook/gui/component/addressbook-view.c
@@ -62,6 +62,7 @@
#include "addressbook/printing/e-contact-print.h"
#include "addressbook/util/eab-book-util.h"
#include "addressbook/gui/widgets/eab-popup.h"
+#include "addressbook/gui/widgets/eab-menu.h"
#define PARENT_TYPE G_TYPE_OBJECT
static GObjectClass *parent_class = NULL;
@@ -93,6 +94,8 @@ struct _AddressbookViewPrivate {
ESourceList *source_list;
char *passwd;
EUserCreatableItemsHandler *creatable_items_handler;
+
+ EABMenu *menu;
};
enum DndTargetType {
@@ -337,14 +340,20 @@ update_command_state (EABView *eav, AddressbookView *view)
{
AddressbookViewPrivate *priv = view->priv;
BonoboUIComponent *uic;
+ EABMenuTargetSelect *target;
if (eav != get_current_view (view))
return;
g_object_ref (view);
+ target = eab_view_get_menu_target(eav, priv->menu);
+ e_menu_update_target((EMenu *)priv->menu, target);
+
uic = bonobo_control_get_ui_component (priv->folder_view_control);
+ /* TODO: this stuff can mostly be made to use the target bits instead */
+
if (bonobo_ui_component_get_container (uic) != CORBA_OBJECT_NIL) {
#define SET_SENSITIVE(verb,f) \
bonobo_ui_component_set_prop (uic, (verb), "sensitive", (f)(eav) ? "1" : "0", NULL)
@@ -464,9 +473,11 @@ control_activate_cb (BonoboControl *control,
if (activate) {
control_activate (control, uic, view);
+ e_menu_activate((EMenu *)view->priv->menu, uic, activate);
if (activate && v && v->model)
eab_model_force_folder_bar_message (v->model);
} else {
+ e_menu_activate((EMenu *)view->priv->menu, uic, activate);
bonobo_ui_component_unset_container (uic, NULL);
eab_view_discard_menus (v);
}
@@ -1087,6 +1098,7 @@ addressbook_view_init (AddressbookView *view)
G_CALLBACK (source_list_changed_cb), view);
priv->creatable_items_handler = e_user_creatable_items_handler_new ("contacts", NULL, NULL);
+ priv->menu = eab_menu_new("com.novell.evolution.addressbook.view");
g_signal_connect (priv->folder_view_control, "activate",
G_CALLBACK (control_activate_cb), view);
@@ -1176,6 +1188,9 @@ addressbook_view_dispose (GObject *object)
if (priv->creatable_items_handler)
g_object_unref (priv->creatable_items_handler);
+ if (priv->menu)
+ g_object_unref (priv->menu);
+
g_free (view->priv);
view->priv = NULL;
}
diff --git a/addressbook/gui/widgets/Makefile.am b/addressbook/gui/widgets/Makefile.am
index 504702b277..b54045a5c8 100644
--- a/addressbook/gui/widgets/Makefile.am
+++ b/addressbook/gui/widgets/Makefile.am
@@ -44,6 +44,8 @@ libeabwidgets_la_SOURCES = \
eab-contact-display.h \
eab-gui-util.c \
eab-gui-util.h \
+ eab-menu.c \
+ eab-menu.h \
eab-popup.c \
eab-popup.h \
eab-popup-control.c \
diff --git a/addressbook/gui/widgets/e-addressbook-view.c b/addressbook/gui/widgets/e-addressbook-view.c
index ed55f42a89..09e6f7626e 100644
--- a/addressbook/gui/widgets/e-addressbook-view.c
+++ b/addressbook/gui/widgets/e-addressbook-view.c
@@ -44,6 +44,7 @@
#include "addressbook/printing/e-contact-print-envelope.h"
#include "addressbook/gui/search/e-addressbook-search-dialog.h"
#include "addressbook/gui/widgets/eab-popup.h"
+#include "addressbook/gui/widgets/eab-menu.h"
#include "e-util/e-categories-master-list-wombat.h"
#include "e-util/e-sexp.h"
@@ -2171,3 +2172,25 @@ eab_view_can_move_to_folder (EABView *view)
{
return view ? eab_view_selection_nonempty (view) && eab_model_editable (view->model) : FALSE;
}
+
+EABMenuTargetSelect *
+eab_view_get_menu_target (EABView *view, EABMenu *menu)
+{
+ GPtrArray *cards = g_ptr_array_new();
+ ESelectionModel *selection_model;
+ EABMenuTargetSelect *t;
+
+ selection_model = get_selection_model (view);
+ if (selection_model) {
+ ContactAndBook cab;
+
+ cab.view = view;
+ cab.closure = cards;
+ e_selection_model_foreach(selection_model, get_card_1, &cab);
+ }
+
+ t = eab_menu_target_new_select(menu, view->book, !eab_model_editable(view->model), cards);
+ t->target.widget = (GtkWidget *)view;
+
+ return t;
+}
diff --git a/addressbook/gui/widgets/e-addressbook-view.h b/addressbook/gui/widgets/e-addressbook-view.h
index 810252ea6b..a55406915e 100644
--- a/addressbook/gui/widgets/e-addressbook-view.h
+++ b/addressbook/gui/widgets/e-addressbook-view.h
@@ -32,6 +32,9 @@
G_BEGIN_DECLS
+struct _EABMenu;
+struct _EABMenuTargetSelect;
+
/* EABView - A card displaying information about a contact.
*
* The following arguments are available:
@@ -154,6 +157,8 @@ gboolean eab_view_can_stop (EABView *view);
gboolean eab_view_can_copy_to_folder (EABView *view);
gboolean eab_view_can_move_to_folder (EABView *view);
+struct _EABMenuTargetSelect *eab_view_get_menu_target (EABView *view, struct _EABMenu *menu);
+
G_END_DECLS;
#endif /* __EAB_VIEW_H__ */
diff --git a/addressbook/gui/widgets/eab-menu.c b/addressbook/gui/widgets/eab-menu.c
new file mode 100644
index 0000000000..d398c769f2
--- /dev/null
+++ b/addressbook/gui/widgets/eab-menu.c
@@ -0,0 +1,257 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+/*
+ * Authors: Michael Zucchi <notzed@ximian.com>
+ *
+ * Copyright 2004 Ximian, Inc. (www.ximian.com)
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA.
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <string.h>
+#include <stdlib.h>
+
+#include <glib.h>
+
+#include <libebook/e-contact.h>
+
+#include "eab-menu.h"
+
+static void eabm_standard_menu_factory(EMenu *emp, void *data);
+
+static GObjectClass *eabm_parent;
+
+static void
+eabm_init(GObject *o)
+{
+ /*EABMenu *emp = (EABMenu *)o; */
+}
+
+static void
+eabm_finalise(GObject *o)
+{
+ ((GObjectClass *)eabm_parent)->finalize(o);
+}
+
+static void
+eabm_target_free(EMenu *ep, EMenuTarget *t)
+{
+ switch (t->type) {
+ case EAB_MENU_TARGET_SELECT: {
+ EABMenuTargetSelect *s = (EABMenuTargetSelect *)t;
+ int i;
+
+ for (i=0;i<s->cards->len;i++)
+ g_object_unref(s->cards->pdata[i]);
+ g_ptr_array_free(s->cards, TRUE);
+ g_object_unref(s->book);
+ break; }
+ }
+
+ ((EMenuClass *)eabm_parent)->target_free(ep, t);
+}
+
+static void
+eabm_class_init(GObjectClass *klass)
+{
+ klass->finalize = eabm_finalise;
+ ((EMenuClass *)klass)->target_free = eabm_target_free;
+
+ e_menu_class_add_factory((EMenuClass *)klass, NULL, (EMenuFactoryFunc)eabm_standard_menu_factory, NULL);
+}
+
+GType
+eab_menu_get_type(void)
+{
+ static GType type = 0;
+
+ if (type == 0) {
+ static const GTypeInfo info = {
+ sizeof(EABMenuClass),
+ NULL, NULL,
+ (GClassInitFunc)eabm_class_init,
+ NULL, NULL,
+ sizeof(EABMenu), 0,
+ (GInstanceInitFunc)eabm_init
+ };
+ eabm_parent = g_type_class_ref(e_menu_get_type());
+ type = g_type_register_static(e_menu_get_type(), "EABMenu", &info, 0);
+ }
+
+ return type;
+}
+
+EABMenu *eab_menu_new(const char *menuid)
+{
+ EABMenu *emp = g_object_new(eab_menu_get_type(), 0);
+
+ e_menu_construct(&emp->menu, menuid);
+
+ return emp;
+}
+
+/**
+ * eab_menu_target_new_select - create a menu target of the current selection.
+ * @eabp: Address book menu.
+ * @book: Book the cards belong to. May be NULL in which case cards must be an empty GPtrArray.
+ * @readonly: Book is read-only mode. FIXME: Why can't we just get this off the book?
+ * @cards: Cards selected. This will be freed on completion and the array indices unreferenced.
+ *
+ * Create a new selection menu target.
+ *
+ * Return value:
+ **/
+EABMenuTargetSelect *
+eab_menu_target_new_select(EABMenu *eabp, struct _EBook *book, int readonly, GPtrArray *cards)
+{
+ EABMenuTargetSelect *t = e_menu_target_new(&eabp->menu, EAB_MENU_TARGET_SELECT, sizeof(*t));
+ guint32 mask = ~0;
+ int has_email = FALSE, i;
+
+ /* FIXME: duplicated in eab-popup.c */
+
+ t->book = book;
+ if (book)
+ g_object_ref(book);
+ t->cards = cards;
+
+ for (i=0;i<cards->len && !has_email;i++) {
+ EContact *contact = cards->pdata[i];
+ GList *email;
+
+ email = e_contact_get(E_CONTACT(contact), E_CONTACT_EMAIL);
+ if (email) {
+ has_email = TRUE;
+
+ g_list_foreach(email, (GFunc)g_free, NULL);
+ g_list_free(email);
+ }
+ }
+
+ if (has_email)
+ mask &= ~EAB_MENU_SELECT_EMAIL;
+
+ if (!readonly)
+ mask &= ~EAB_MENU_SELECT_EDITABLE;
+
+ if (cards->len == 1)
+ mask &= ~EAB_MENU_SELECT_ONE;
+
+ if (cards->len > 1)
+ mask &= ~EAB_MENU_SELECT_MANY;
+
+ if (cards->len >= 1)
+ mask &= ~EAB_MENU_SELECT_ANY;
+
+ t->target.mask = mask;
+
+ return t;
+}
+
+static void
+eabm_standard_menu_factory(EMenu *emp, void *data)
+{
+ /* noop */
+}
+
+/* ********************************************************************** */
+
+/* menu plugin handler */
+
+/*
+<e-plugin
+ class="com.ximian.mail.plugin.popup:1.0"
+ id="com.ximian.mail.plugin.popup.item:1.0"
+ type="shlib"
+ location="/opt/gnome2/lib/camel/1.0/libcamelimap.so"
+ name="imap"
+ description="IMAP4 and IMAP4v1 mail store">
+ <hook class="com.ximian.mail.popupMenu:1.0"
+ handler="HandlePopup">
+ <menu id="any" target="select">
+ <item
+ type="item|toggle|radio|image|submenu|bar"
+ active
+ path="foo/bar"
+ label="label"
+ icon="foo"
+ mask="select_one"
+ activate="eabm_view_emacs"/>
+ </menu>
+ </extension>
+
+*/
+
+static void *eabmph_parent_class;
+#define eabmph ((EABMenuHook *)eph)
+
+static const EMenuHookTargetMask eabmph_select_masks[] = {
+ { "one", EAB_MENU_SELECT_ONE },
+ { "many", EAB_MENU_SELECT_MANY },
+ { "any", EAB_MENU_SELECT_ANY },
+ { "editable", EAB_MENU_SELECT_EDITABLE },
+ { "email", EAB_MENU_SELECT_EMAIL },
+ { 0 }
+};
+
+static const EMenuHookTargetMap eabmph_targets[] = {
+ { "select", EAB_MENU_TARGET_SELECT, eabmph_select_masks },
+ { 0 }
+};
+
+static void
+eabmph_finalise(GObject *o)
+{
+ /*EPluginHook *eph = (EPluginHook *)o;*/
+
+ ((GObjectClass *)eabmph_parent_class)->finalize(o);
+}
+
+static void
+eabmph_class_init(EPluginHookClass *klass)
+{
+ int i;
+
+ ((GObjectClass *)klass)->finalize = eabmph_finalise;
+ ((EPluginHookClass *)klass)->id = "com.novell.evolution.addressbook.bonobomenu:1.0";
+
+ for (i=0;eabmph_targets[i].type;i++)
+ e_menu_hook_class_add_target_map((EMenuHookClass *)klass, &eabmph_targets[i]);
+
+ /* FIXME: leaks parent set class? */
+ ((EMenuHookClass *)klass)->menu_class = g_type_class_ref(eab_menu_get_type());
+}
+
+GType
+eab_menu_hook_get_type(void)
+{
+ static GType type = 0;
+
+ if (!type) {
+ static const GTypeInfo info = {
+ sizeof(EABMenuHookClass), NULL, NULL, (GClassInitFunc) eabmph_class_init, NULL, NULL,
+ sizeof(EABMenuHook), 0, (GInstanceInitFunc) NULL,
+ };
+
+ eabmph_parent_class = g_type_class_ref(e_menu_hook_get_type());
+ type = g_type_register_static(e_menu_hook_get_type(), "EABMenuHook", &info, 0);
+ }
+
+ return type;
+}
diff --git a/addressbook/gui/widgets/eab-menu.h b/addressbook/gui/widgets/eab-menu.h
new file mode 100644
index 0000000000..0427d43fe1
--- /dev/null
+++ b/addressbook/gui/widgets/eab-menu.h
@@ -0,0 +1,109 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
+ *
+ * Authors: Michel Zucchi <notzed@ximian.com>
+ *
+ * Copyright 2004 Novell Inc. (www.novell.com)
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA.
+ *
+ */
+
+#ifndef __EAB_MENU_H__
+#define __EAB_MENU_H__
+
+#include <glib-object.h>
+
+#include "e-util/e-menu.h"
+
+#ifdef __cplusplus
+extern "C" {
+#pragma }
+#endif /* __cplusplus */
+
+struct _EBook;
+
+typedef struct _EABMenu EABMenu;
+typedef struct _EABMenuClass EABMenuClass;
+
+/* Current target description */
+enum _eab_menu_target_t {
+ EAB_MENU_TARGET_SELECT,
+};
+
+/**
+ * enum _eab_menu_target_select_t - EABMenuTargetSelect qualifiers.
+ *
+ * @EAB_MENU_SELECT_ONE: Only one item is selected.
+ * @EAB_MENU_SELECT_MANY: More than one item selected.
+ * @EAB_MENU_SELECT_ANY: One or more items selected.
+ * @EAB_MENU_SELECT_EDITABLE: Editable addressbook.
+ * @EAB_MENU_SELECT_EMAIL: Has an email address.
+ **/
+enum _eab_menu_target_select_t {
+ EAB_MENU_SELECT_ONE = 1<<0,
+ EAB_MENU_SELECT_MANY = 1<<1,
+ EAB_MENU_SELECT_ANY = 1<<2,
+ EAB_MENU_SELECT_EDITABLE = 1<<3,
+ EAB_MENU_SELECT_EMAIL = 1<<4,
+};
+
+typedef struct _EABMenuTargetSelect EABMenuTargetSelect;
+
+struct _EABMenuTargetSelect {
+ EMenuTarget target;
+
+ struct _EBook *book;
+ GPtrArray *cards;
+};
+
+typedef struct _EMenuItem EABMenuItem;
+
+/* The object */
+struct _EABMenu {
+ EMenu menu;
+
+ struct _EABMenuPrivate *priv;
+};
+
+struct _EABMenuClass {
+ EMenuClass menu_class;
+};
+
+GType eab_menu_get_type(void);
+
+EABMenu *eab_menu_new(const char *menuid);
+
+EABMenuTargetSelect *eab_menu_target_new_select(EABMenu *eabp, struct _EBook *book, int readonly, GPtrArray *cards);
+
+/* ********************************************************************** */
+
+typedef struct _EABMenuHook EABMenuHook;
+typedef struct _EABMenuHookClass EABMenuHookClass;
+
+struct _EABMenuHook {
+ EMenuHook hook;
+};
+
+struct _EABMenuHookClass {
+ EMenuHookClass hook_class;
+};
+
+GType eab_menu_hook_get_type(void);
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __EAB_MENU_H__ */
diff --git a/addressbook/gui/widgets/eab-popup.c b/addressbook/gui/widgets/eab-popup.c
index 79e9c2a57a..2a9a7e654b 100644
--- a/addressbook/gui/widgets/eab-popup.c
+++ b/addressbook/gui/widgets/eab-popup.c
@@ -130,6 +130,8 @@ eab_popup_target_new_select(EABPopup *eabp, struct _EBook *book, int readonly, G
guint32 mask = ~0;
int has_email = FALSE, i;
+ /* FIXME: duplicated in eab-menu.c */
+
t->book = book;
g_object_ref(book);
t->cards = cards;
@@ -156,9 +158,12 @@ eab_popup_target_new_select(EABPopup *eabp, struct _EBook *book, int readonly, G
if (cards->len == 1)
mask &= ~EAB_POPUP_SELECT_ONE;
- if (cards->len >= 1)
+ if (cards->len > 1)
mask &= ~EAB_POPUP_SELECT_MANY;
+ if (cards->len >= 1)
+ mask &= ~EAB_POPUP_SELECT_ANY;
+
t->target.mask = mask;
return t;
@@ -199,6 +204,9 @@ eab_popup_target_new_select_names(EABPopup *eabp, struct _ESelectNamesModel *mod
{
EABPopupTargetSelectNames *t = e_popup_target_new(&eabp->popup, EAB_POPUP_TARGET_SELECT_NAMES, sizeof(*t));
+ /* TODO: this is sort of not very useful, maybe the popup which uses it doesn't
+ need to be pluggable */
+
t->model = model;
g_object_ref(model);
t->row = row;
@@ -239,6 +247,9 @@ static void *eabph_parent_class;
static const EPopupHookTargetMask eabph_select_masks[] = {
{ "one", EAB_POPUP_SELECT_ONE },
{ "many", EAB_POPUP_SELECT_MANY },
+ { "any", EAB_POPUP_SELECT_ANY },
+ { "editable", EAB_POPUP_SELECT_EDITABLE },
+ { "email", EAB_POPUP_SELECT_EMAIL },
{ 0 }
};
diff --git a/addressbook/gui/widgets/eab-popup.h b/addressbook/gui/widgets/eab-popup.h
index 39df885de5..7fec4cc742 100644
--- a/addressbook/gui/widgets/eab-popup.h
+++ b/addressbook/gui/widgets/eab-popup.h
@@ -52,15 +52,17 @@ enum _eab_popup_target_t {
* enum _eab_popup_target_select_t - EABPopupTargetSelect qualifiers.
*
* @EAB_POPUP_SELECT_ONE: Only one item is selected.
- * @EAB_POPUP_SELECT_MANY: One ore more items are selected.
+ * @EAB_POPUP_SELECT_MANY: Two or more items are selected.
+ * @EAB_POPUP_SELECT_ANY: One or more items are selected.
* @EAB_POPUP_SELECT_EDITABLE: Read/writable source.
* @EAB_POPUP_SELECT_EMAIL: Has an email address.
**/
enum _eab_popup_target_select_t {
EAB_POPUP_SELECT_ONE = 1<<0,
EAB_POPUP_SELECT_MANY = 1<<1,
- EAB_POPUP_SELECT_EDITABLE = 1<<2,
- EAB_POPUP_SELECT_EMAIL = 1<<3,
+ EAB_POPUP_SELECT_ANY = 1<<2,
+ EAB_POPUP_SELECT_EDITABLE = 1<<3,
+ EAB_POPUP_SELECT_EMAIL = 1<<4,
};
/**