aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--addressbook/ChangeLog10
-rw-r--r--addressbook/gui/widgets/eab-contact-display.c190
-rw-r--r--addressbook/gui/widgets/eab-popup.c39
-rw-r--r--addressbook/gui/widgets/eab-popup.h15
4 files changed, 250 insertions, 4 deletions
diff --git a/addressbook/ChangeLog b/addressbook/ChangeLog
index 6021880dab..ce849de229 100644
--- a/addressbook/ChangeLog
+++ b/addressbook/ChangeLog
@@ -1,3 +1,13 @@
+2006-02-18 M Victor Aloysius J <victoraloy@gmail.com>
+
+ ** Fixes bug #317512
+
+ * gui/eab-contact-editot.c:
+ * gui/eab-popup.h:
+ * gui/eab-popup.c:
+ Hyperlinked email-addresses and websites in preview pane support right
+ click mouse options.
+
2006-04-21 Simon Zheng <simon.zheng@sun.com>
Fix for bug #337604
diff --git a/addressbook/gui/widgets/eab-contact-display.c b/addressbook/gui/widgets/eab-contact-display.c
index 4d7b39debe..897fd79ff5 100644
--- a/addressbook/gui/widgets/eab-contact-display.c
+++ b/addressbook/gui/widgets/eab-contact-display.c
@@ -25,6 +25,7 @@
#endif
#include "eab-contact-display.h"
+#include "eab-popup.h"
#include "eab-gui-util.h"
#include "e-util/e-html-utils.h"
@@ -42,6 +43,9 @@
struct _EABContactDisplayPrivate {
EContact *contact;
+
+ GtkWidget *invisible;
+ char *selection_uri;
};
@@ -62,6 +66,157 @@ struct _EABContactDisplayPrivate {
#define MAX_COMPACT_IMAGE_DIMENSION 48
static void
+eab_uri_popup_link_open(EPopup *ep, EPopupItem *item, void *data)
+{
+ EABPopupTargetURI *t = (EABPopupTargetURI *)ep->target;
+ GError *err = NULL;
+
+ gnome_url_show(t->uri, &err);
+ if (err) {
+ g_warning("gnome_url_show: %s", err->message);
+ g_error_free(err);
+ }
+}
+
+static void
+eab_uri_popup_email_address_copy(EPopup *ep, EPopupItem *item, void *data)
+{
+ EABContactDisplay *display = data;
+ struct _EABContactDisplayPrivate *p = display->priv;
+ EABPopupTargetURI *t = (EABPopupTargetURI *)ep->target;
+ const char *url;
+ char *html=NULL;
+ int i=0;
+ GList *email_list, *l;
+ url = t->uri;
+ int email_num = atoi (url + strlen ("internal-mailto:"));
+
+ email_list = e_contact_get (p->contact, E_CONTACT_EMAIL);
+ for (l = email_list; l; l=l->next) {
+ if(i==email_num)
+ html = e_text_to_html (l->data, 0);
+ i++;
+ }
+
+ g_free(p->selection_uri);
+ p->selection_uri = g_strdup(html);
+ g_free (html);
+
+ gtk_selection_owner_set(p->invisible, GDK_SELECTION_PRIMARY, gtk_get_current_event_time());
+ gtk_selection_owner_set(p->invisible, GDK_SELECTION_CLIPBOARD, gtk_get_current_event_time());
+}
+
+static void
+eab_uri_popup_link_copy(EPopup *ep, EPopupItem *pitem, void *data)
+{
+ EABContactDisplay *display = data;
+ struct _EABContactDisplayPrivate *p = display->priv;
+
+ g_free(p->selection_uri);
+ p->selection_uri = g_strdup(pitem->user_data);
+
+ gtk_selection_owner_set(p->invisible, GDK_SELECTION_PRIMARY, gtk_get_current_event_time());
+ gtk_selection_owner_set(p->invisible, GDK_SELECTION_CLIPBOARD, gtk_get_current_event_time());
+}
+
+static void
+eab_uri_popup_address_send(EPopup *ep, EPopupItem *item, void *data)
+{
+ EABPopupTargetURI *t = (EABPopupTargetURI *)ep->target;
+ const char *url;
+ EABContactDisplay *display = data;
+ struct _EABContactDisplayPrivate *p = display->priv;
+
+ url = t->uri;
+ int mail_num = atoi (url + strlen ("internal-mailto:"));
+
+ if (mail_num == -1)
+ return;
+
+ eab_send_contact (p->contact, mail_num, EAB_DISPOSITION_AS_TO);
+
+}
+
+static void
+eab_selection_get(GtkWidget *widget, GtkSelectionData *data, guint info, guint time_stamp, EABContactDisplay *display)
+{
+ struct _EABContactDisplayPrivate *p = display->priv;
+
+ if (p->selection_uri == NULL)
+ return;
+
+ gtk_selection_data_set(data, data->target, 8, p->selection_uri, strlen(p->selection_uri));
+}
+
+static void
+eab_selection_clear_event(GtkWidget *widget, GdkEventSelection *event, EABContactDisplay *display)
+{
+#if 0
+ struct _EABContactDisplayPrivate *p = display->priv;
+
+ g_free(p->selection_uri);
+ p->selection_uri = NULL;
+#endif
+}
+
+static EPopupItem eab_uri_popups[] = {
+ { E_POPUP_ITEM, "05.open", N_("_Open Link in Browser"), eab_uri_popup_link_open, NULL, NULL, EAB_POPUP_URI_NOT_MAILTO },
+ { E_POPUP_ITEM, "10.copy", N_("_Copy Link Location"), eab_uri_popup_link_copy, NULL, NULL, EAB_POPUP_URI_NOT_MAILTO },
+ { E_POPUP_ITEM, "15.send", N_("_Send New Message To ..."), eab_uri_popup_address_send, NULL, NULL, EAB_POPUP_URI_MAILTO},
+ { E_POPUP_ITEM, "20.copy", N_("Copy _Email Address"), eab_uri_popup_email_address_copy, NULL, "gtk-copy", EAB_POPUP_URI_MAILTO},
+ };
+
+
+static void
+eab_uri_popup_free(EPopup *ep, GSList *list, void *data)
+{
+ while (list){
+ GSList *n = list->next;
+ struct _EPopupItem *item = list->data;
+
+ g_free(item->user_data);
+ item->user_data = NULL;
+ g_slist_free_1(list);
+
+ list = n;
+ }
+}
+
+static int
+eab_uri_popup_event(EABContactDisplay *display, GdkEvent *event, const char *uri)
+{
+ EABPopup *emp;
+ EABPopupTargetURI *t ;
+ GtkMenu *menu;
+
+
+ emp = eab_popup_new("org.gnome.evolution.addressbook.contactdisplay.popup");
+
+ GSList *menus = NULL;
+ int i;
+
+ t = eab_popup_target_new_uri(emp, uri);
+ t->target.widget = (GtkWidget *)display;
+
+ for (i=0;i<sizeof(eab_uri_popups)/sizeof(eab_uri_popups[0]);i++) {
+ eab_uri_popups[i].user_data = g_strdup(t->uri);
+ menus = g_slist_prepend(menus, &eab_uri_popups[i]);
+ }
+ e_popup_add_items((EPopup *)emp, menus, NULL, eab_uri_popup_free, display);
+
+
+ menu = e_popup_create_menu_once((EPopup *)emp,(EPopupTarget*)t, 0);
+
+ if (event == NULL) {
+ gtk_menu_popup(menu, NULL, NULL, NULL, NULL, 0, gtk_get_current_event_time());
+ } else {
+ gtk_menu_popup(menu, NULL, NULL, NULL, NULL, event->button.button, event->button.time);
+ }
+
+ return TRUE;
+}
+
+static void
on_url_requested (GtkHTML *html, const char *url, GtkHTMLStream *handle,
EABContactDisplay *display)
{
@@ -106,7 +261,7 @@ on_link_clicked (GtkHTML *html, const char *url, EABContactDisplay *display)
if (mail_num == -1)
return;
-
+
eab_send_contact (display->priv->contact, mail_num, EAB_DISPOSITION_AS_TO);
return;
@@ -624,6 +779,26 @@ eab_contact_display_render (EABContactDisplay *display, EContact *contact,
}
}
+static int
+eab_html_press_event (GtkWidget *widget, GdkEvent *event,EABContactDisplay *display)
+{
+ char *uri;
+ gboolean res = FALSE;
+
+
+ if (event->button.button!= 3 )
+ return FALSE;
+
+ uri = gtk_html_get_url_at (GTK_HTML (widget), event->button.x, event->button.y);
+ if (uri){
+ eab_uri_popup_event(display,event,uri);
+ }
+
+ g_free(uri);
+
+ return res;
+}
+
GtkWidget*
eab_contact_display_new (void)
{
@@ -631,7 +806,8 @@ eab_contact_display_new (void)
display = g_object_new (EAB_TYPE_CONTACT_DISPLAY, NULL);
- display->priv = g_new0 (EABContactDisplayPrivate, 1);
+ struct _EABContactDisplayPrivate *p;
+ p=display->priv = g_new0 (EABContactDisplayPrivate, 1);
gtk_html_set_default_content_type (GTK_HTML (display), "text/html; charset=utf-8");
@@ -643,6 +819,15 @@ eab_contact_display_new (void)
g_signal_connect (display, "link_clicked",
G_CALLBACK (on_link_clicked),
display);
+ g_signal_connect(display, "button_press_event",
+ G_CALLBACK(eab_html_press_event),
+ display);
+ p->invisible = gtk_invisible_new();
+ g_signal_connect(p->invisible, "selection_get", G_CALLBACK(eab_selection_get), display);
+ g_signal_connect(p->invisible, "selection_clear_event", G_CALLBACK(eab_selection_clear_event), display);
+ gtk_selection_add_target(p->invisible, GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0);
+ gtk_selection_add_target(p->invisible, GDK_SELECTION_CLIPBOARD, GDK_SELECTION_TYPE_STRING, 1);
+
#if 0
g_signal_connect (display, "object_requested",
G_CALLBACK (on_object_requested),
@@ -662,7 +847,6 @@ eab_contact_display_new (void)
return GTK_WIDGET (display);
}
-
static void
eab_contact_display_init (GObject *object)
{
diff --git a/addressbook/gui/widgets/eab-popup.c b/addressbook/gui/widgets/eab-popup.c
index 4a745a41f2..8da9241624 100644
--- a/addressbook/gui/widgets/eab-popup.c
+++ b/addressbook/gui/widgets/eab-popup.c
@@ -60,6 +60,11 @@ eabp_target_free(EPopup *ep, EPopupTarget *t)
g_object_unref(s->book);
break; }
+ case EAB_POPUP_TARGET_URI: {
+ EABPopupTargetURI *s = (EABPopupTargetURI *)t;
+
+ g_free(s->uri);
+ break; }
case EAB_POPUP_TARGET_SOURCE: {
EABPopupTargetSource *s = (EABPopupTargetSource *)t;
@@ -129,6 +134,32 @@ EABPopup *eab_popup_new(const char *menuid)
*
* Return value:
**/
+
+
+EABPopupTargetURI *
+eab_popup_target_new_uri(EABPopup *emp, const char *uri)
+{
+ EABPopupTargetURI *t = e_popup_target_new(&emp->popup, EAB_POPUP_TARGET_URI, sizeof(*t));
+ guint32 mask = ~0;
+
+ t->uri = g_strdup(uri);
+
+ if (g_ascii_strncasecmp(uri, "http:", 5) == 0
+ || g_ascii_strncasecmp(uri, "https:", 6) == 0)
+ mask &= ~EAB_POPUP_URI_HTTP;
+ if (g_ascii_strncasecmp(uri, "internal-mailto:", 16) == 0)
+ mask &= ~EAB_POPUP_URI_MAILTO;
+ else
+ mask &= ~EAB_POPUP_URI_NOT_MAILTO;
+
+ t->target.mask = mask;
+
+ return t;
+}
+
+
+
+
EABPopupTargetSelect *
eab_popup_target_new_select(EABPopup *eabp, struct _EBook *book, int readonly, GPtrArray *cards)
{
@@ -276,12 +307,20 @@ static const EPopupHookTargetMask eabph_source_masks[] = {
{ 0 }
};
+static const EPopupHookTargetMask eabph_uri_masks[] = {
+ { "http", EAB_POPUP_URI_HTTP },
+ { "internal-mailto", EAB_POPUP_URI_MAILTO },
+ { "notmailto", EAB_POPUP_URI_NOT_MAILTO },
+ { 0 }
+};
+
static const EPopupHookTargetMask eabph_select_names_masks[] = {
{ 0 }
};
static const EPopupHookTargetMap eabph_targets[] = {
{ "select", EAB_POPUP_TARGET_SELECT, eabph_select_masks },
+ { "uri", EAB_POPUP_TARGET_URI, eabph_uri_masks },
{ "source", EAB_POPUP_TARGET_SOURCE, eabph_source_masks },
{ "select-names", EAB_POPUP_TARGET_SELECT_NAMES, eabph_select_names_masks },
{ 0 }
diff --git a/addressbook/gui/widgets/eab-popup.h b/addressbook/gui/widgets/eab-popup.h
index 11e46f4c8c..1ae069dac3 100644
--- a/addressbook/gui/widgets/eab-popup.h
+++ b/addressbook/gui/widgets/eab-popup.h
@@ -46,6 +46,7 @@ typedef struct _EABPopupClass EABPopupClass;
**/
enum _eab_popup_target_t {
EAB_POPUP_TARGET_SELECT,
+ EAB_POPUP_TARGET_URI,
EAB_POPUP_TARGET_SOURCE,
EAB_POPUP_TARGET_SELECT_NAMES,
};
@@ -69,6 +70,11 @@ enum _eab_popup_target_select_t {
EAB_POPUP_CONTACT = 1<<6,
};
+enum _eab_popup_target_uri_t {
+ EAB_POPUP_URI_HTTP = 1<<0,
+ EAB_POPUP_URI_MAILTO = 1<<1,
+ EAB_POPUP_URI_NOT_MAILTO = 1<<2,
+};
/**
* enum _eab_popup_target_source_t - EABPopupTargetSource qualifiers.
*
@@ -85,7 +91,7 @@ enum _eab_popup_target_source_t {
typedef struct _EABPopupTargetSelect EABPopupTargetSelect;
typedef struct _EABPopupTargetSource EABPopupTargetSource;
typedef struct _EABPopupTargetSelectNames EABPopupTargetSelectNames;
-
+typedef struct _EABPopupTargetURI EABPopupTargetURI;
/**
* struct _EABPopupTargetSelect - A list of address cards.
*
@@ -103,6 +109,12 @@ struct _EABPopupTargetSelect {
GPtrArray *cards;
};
+
+struct _EABPopupTargetURI {
+ EPopupTarget target;
+ char *uri;
+};
+
/**
* struct _EABPopupTargetSource - A source target.
*
@@ -156,6 +168,7 @@ GType eab_popup_get_type(void);
EABPopup *eab_popup_new(const char *menuid);
EABPopupTargetSelect *eab_popup_target_new_select(EABPopup *eabp, struct _EBook *book, int readonly, GPtrArray *cards);
+EABPopupTargetURI *eab_popup_target_new_uri(EABPopup *emp, const char *uri);
EABPopupTargetSource *eab_popup_target_new_source(EABPopup *eabp, struct _ESourceSelector *selector);
#ifdef ADAPTED_TO_E_NAME_SELECTOR