aboutsummaryrefslogtreecommitdiffstats
path: root/addressbook
diff options
context:
space:
mode:
Diffstat (limited to 'addressbook')
-rw-r--r--addressbook/gui/widgets/eab-contact-display.c469
1 files changed, 195 insertions, 274 deletions
diff --git a/addressbook/gui/widgets/eab-contact-display.c b/addressbook/gui/widgets/eab-contact-display.c
index 0ade284323..692b6f003e 100644
--- a/addressbook/gui/widgets/eab-contact-display.c
+++ b/addressbook/gui/widgets/eab-contact-display.c
@@ -46,14 +46,6 @@
struct _EABContactDisplayPrivate {
EContact *contact;
EABContactDisplayMode mode;
-
- GtkUIManager *ui_manager;
- GtkActionGroup *email_actions;
- GtkActionGroup *uri_actions;
- GtkWidget *invisible;
-
- gchar *selected_uri;
- gchar *clipboard_uri;
};
enum {
@@ -98,11 +90,13 @@ common_location [] =
static const gchar *ui =
"<ui>"
-" <popup>"
-" <menuitem action='open-link'/>"
-" <menuitem action='copy-link'/>"
-" <menuitem action='send-message'/>"
-" <menuitem action='copy-address'/>"
+" <popup name='context'>"
+" <placeholder name='custom-actions-1'>"
+" <menuitem action='contact-send-message'/>"
+" </placeholder>"
+" <placeholder name='custom-actions-2'>"
+" <menuitem action='contact-mailto-copy'/>"
+" </placeholder>"
" </popup>"
"</ui>";
@@ -110,135 +104,85 @@ static gpointer parent_class;
static guint signals[LAST_SIGNAL];
static void
-action_copy_address_cb (GtkAction *action,
- EABContactDisplay *display)
+contact_display_emit_send_message (EABContactDisplay *display,
+ gint email_num)
{
+ EDestination *destination;
+ EContact *contact;
+
+ g_return_if_fail (email_num >= 0);
+
+ destination = e_destination_new ();
+ contact = eab_contact_display_get_contact (display);
+ e_destination_set_contact (destination, contact, email_num);
+ g_signal_emit (display, signals[SEND_MESSAGE], 0, destination);
+ g_object_unref (destination);
+}
+
+static void
+action_contact_mailto_copy_cb (GtkAction *action,
+ EABContactDisplay *display)
+{
+ GtkClipboard *clipboard;
+ EWebView *web_view;
EContact *contact;
GList *list;
+ const gchar *text;
const gchar *uri;
- gchar *html;
gint index;
- uri = display->priv->selected_uri;
+ web_view = E_WEB_VIEW (display);
+ uri = e_web_view_get_selected_uri (web_view);
+ g_return_if_fail (uri != NULL);
+
index = atoi (uri + strlen ("internal-mailto:"));
- contact = eab_contact_display_get_contact (display);
+ g_return_if_fail (index >= 0);
+ contact = eab_contact_display_get_contact (display);
list = e_contact_get (contact, E_CONTACT_EMAIL);
- html = e_text_to_html (g_list_nth_data (list, index), 0);
- g_list_foreach (list, (GFunc) g_free, NULL);
- g_list_free (list);
-
- display->priv->clipboard_uri = html;
- display->priv->selected_uri = NULL;
+ text = g_list_nth_data (list, index);
- gtk_selection_owner_set (
- display->priv->invisible, GDK_SELECTION_PRIMARY,
- gtk_get_current_event_time ());
- gtk_selection_owner_set (
- display->priv->invisible, GDK_SELECTION_CLIPBOARD,
- gtk_get_current_event_time ());
-}
-
-static void
-action_copy_link_cb (GtkAction *action,
- EABContactDisplay *display)
-{
- display->priv->clipboard_uri = display->priv->selected_uri;
- display->priv->selected_uri = NULL;
-
- gtk_selection_owner_set (
- display->priv->invisible, GDK_SELECTION_PRIMARY,
- gtk_get_current_event_time ());
- gtk_selection_owner_set (
- display->priv->invisible, GDK_SELECTION_CLIPBOARD,
- gtk_get_current_event_time ());
-}
+ clipboard = gtk_clipboard_get (GDK_SELECTION_CLIPBOARD);
+ gtk_clipboard_set_text (clipboard, text, -1);
+ gtk_clipboard_store (clipboard);
-static void
-action_open_link_cb (GtkAction *action,
- EABContactDisplay *display)
-{
- /* XXX Pass a parent window. */
- e_show_uri (NULL, display->priv->selected_uri);
+ g_list_foreach (list, (GFunc) g_free, NULL);
+ g_list_free (list);
}
static void
-action_send_message_cb (GtkAction *action,
- EABContactDisplay *display)
+action_contact_send_message_cb (GtkAction *action,
+ EABContactDisplay *display)
{
- EDestination *destination;
- EContact *contact;
+ EWebView *web_view;
const gchar *uri;
- gint row;
+ gint index;
- uri = display->priv->selected_uri;
- row = atoi (uri + strlen ("internal-mailto:"));
- g_return_if_fail (row >= 0);
+ web_view = E_WEB_VIEW (display);
+ uri = e_web_view_get_selected_uri (web_view);
+ g_return_if_fail (uri != NULL);
- destination = e_destination_new ();
- contact = eab_contact_display_get_contact (display);
- e_destination_set_contact (destination, contact, row);
- g_signal_emit (display, signals[SEND_MESSAGE], 0, destination);
- g_object_unref (destination);
+ index = atoi (uri + strlen ("internal-mailto:"));
+ contact_display_emit_send_message (display, index);
}
-static GtkActionEntry email_entries[] = {
+static GtkActionEntry internal_mailto_entries[] = {
- { "copy-address",
- NULL,
+ { "contact-mailto-copy",
+ GTK_STOCK_COPY,
N_("Copy _Email Address"),
NULL,
- NULL,
- G_CALLBACK (action_copy_address_cb) },
+ N_("Copy the email address to the clipboard"),
+ G_CALLBACK (action_contact_mailto_copy_cb) },
- { "send-message",
- NULL,
+ { "contact-send-message",
+ "mail-message-new",
N_("_Send New Message To..."),
NULL,
- NULL,
- G_CALLBACK (action_send_message_cb) }
+ N_("Send a mail message to this address"),
+ G_CALLBACK (action_contact_send_message_cb) }
};
-static GtkActionEntry uri_entries[] = {
-
- { "copy-link",
- NULL,
- N_("_Copy Link Location"),
- NULL,
- NULL,
- G_CALLBACK (action_copy_link_cb) },
-
- { "open-link",
- NULL,
- N_("_Open Link in Browser"),
- NULL,
- NULL,
- G_CALLBACK (action_open_link_cb) }
-};
-
-static void
-contact_display_selection_get (EABContactDisplay *display,
- GtkSelectionData *data,
- guint info,
- guint time_stamp)
-{
- if (display->priv->clipboard_uri == NULL)
- return;
-
- gtk_selection_data_set (
- data, data->target, 8,
- (guchar *) display->priv->clipboard_uri,
- strlen (display->priv->clipboard_uri));
-}
-
-static void
-contact_display_selection_clear_event (EABContactDisplay *display,
- GdkEventSelection *event)
-{
- g_free (display->priv->clipboard_uri);
- display->priv->clipboard_uri = NULL;
-}
-
static void
render_name_value (GtkHTMLStream *html_stream, const gchar *label, const gchar *str, const gchar *icon, guint html_flags)
{
@@ -823,54 +767,6 @@ eab_contact_display_render_compact (EABContactDisplay *display,
gtk_html_end (GTK_HTML (display), html_stream, GTK_HTML_STREAM_OK);
}
-static gint
-contact_display_button_press_event (GtkWidget *widget,
- GdkEvent *event,
- EABContactDisplay *display)
-{
- GtkUIManager *ui_manager;
- GtkActionGroup *action_group;
- GtkWidget *menu;
- gboolean has_email;
- gchar *uri;
-
- if (event->button.button != 3)
- return FALSE;
-
- uri = gtk_html_get_url_at (
- GTK_HTML (widget),
- event->button.x, event->button.y);
-
- if (uri == NULL)
- return FALSE;
-
- g_free (display->priv->selected_uri);
- display->priv->selected_uri = uri;
-
- ui_manager = display->priv->ui_manager;
- menu = gtk_ui_manager_get_widget (ui_manager, "/popup");
- g_return_val_if_fail (GTK_IS_MENU (menu), FALSE);
-
- has_email = (g_ascii_strncasecmp (uri, "internal-mailto:", 16) == 0);
-
- /* Show the appropriate actions. */
- action_group = display->priv->email_actions;
- gtk_action_group_set_visible (action_group, has_email);
- action_group = display->priv->uri_actions;
- gtk_action_group_set_visible (action_group, !has_email);
-
- if (event != NULL)
- gtk_menu_popup (
- GTK_MENU (menu), NULL, NULL, NULL, NULL,
- event->button.button, event->button.time);
- else
- gtk_menu_popup (
- GTK_MENU (menu), NULL, NULL, NULL, NULL,
- 0, gtk_get_current_event_time ());
-
- return TRUE;
-}
-
static void
contact_display_set_property (GObject *object,
guint property_id,
@@ -929,26 +825,6 @@ contact_display_dispose (GObject *object)
priv->contact = NULL;
}
- if (priv->ui_manager != NULL) {
- g_object_unref (priv->ui_manager);
- priv->ui_manager = NULL;
- }
-
- if (priv->email_actions != NULL) {
- g_object_unref (priv->email_actions);
- priv->email_actions = NULL;
- }
-
- if (priv->uri_actions != NULL) {
- g_object_unref (priv->uri_actions);
- priv->uri_actions = NULL;
- }
-
- if (priv->invisible != NULL) {
- g_object_unref (priv->invisible);
- priv->invisible = NULL;
- }
-
/* Chain up to parent's dispose() method. */
G_OBJECT_CLASS (parent_class)->dispose (object);
}
@@ -960,30 +836,35 @@ contact_display_finalize (GObject *object)
priv = EAB_CONTACT_DISPLAY_GET_PRIVATE (object);
- g_free (priv->selected_uri);
- g_free (priv->clipboard_uri);
-
/* Chain up to parent's finalize() method. */
G_OBJECT_CLASS (parent_class)->finalize (object);
}
static void
contact_display_url_requested (GtkHTML *html,
- const gchar *url,
+ const gchar *uri,
GtkHTMLStream *handle)
{
- EABContactDisplayPrivate *priv;
+ EABContactDisplay *display;
+ GtkHTMLClass *class;
+ gsize length;
- priv = EAB_CONTACT_DISPLAY_GET_PRIVATE (html);
+ display = EAB_CONTACT_DISPLAY (html);
+ class = GTK_HTML_CLASS (parent_class);
- if (strcmp (url, "internal-contact-photo:") == 0) {
+ /* internal-contact-photo: */
+ if (strcmp (uri, "internal-contact-photo:") == 0) {
EContactPhoto *photo;
+ EContact *contact;
- photo = e_contact_get (priv->contact, E_CONTACT_PHOTO);
- if (!photo)
- photo = e_contact_get (priv->contact, E_CONTACT_LOGO);
+ contact = eab_contact_display_get_contact (display);
+ photo = e_contact_get (contact, E_CONTACT_PHOTO);
+ if (photo == NULL)
+ photo = e_contact_get (contact, E_CONTACT_LOGO);
- gtk_html_stream_write (handle, (gchar *)photo->data.inlined.data, photo->data.inlined.length);
+ gtk_html_stream_write (
+ handle, (gchar *) photo->data.inlined.data,
+ photo->data.inlined.length);
gtk_html_end (html, handle, GTK_HTML_STREAM_OK);
@@ -992,26 +873,39 @@ contact_display_url_requested (GtkHTML *html,
return;
}
- if (strncmp (url, "evo-icon:", strlen ("evo-icon:")) == 0) {
- gchar *data;
- gsize data_length;
- gchar *filename;
-
- filename = e_icon_factory_get_icon_filename (url + strlen ("evo-icon:"), GTK_ICON_SIZE_MENU);
- if (g_file_get_contents (filename, &data, &data_length, NULL)) {
- gtk_html_stream_write (handle, data, data_length);
- g_free (data);
+ /* evo-icon:<<themed-icon-name>> */
+ length = strlen ("evo-icon:");
+ if (g_ascii_strncasecmp (uri, "evo-icon:", length) == 0) {
+ GtkIconTheme *icon_theme;
+ GtkIconInfo *icon_info;
+ const gchar *filename;
+ gchar *icon_uri;
+ GError *error = NULL;
+
+ icon_theme = gtk_icon_theme_get_default ();
+ icon_info = gtk_icon_theme_lookup_icon (
+ icon_theme, uri + length, GTK_ICON_SIZE_MENU, 0);
+ g_return_if_fail (icon_info != NULL);
+
+ filename = gtk_icon_info_get_filename (icon_info);
+ icon_uri = g_filename_to_uri (filename, NULL, &error);
+
+ if (error != NULL) {
+ g_warning ("%s", error->message);
+ g_error_free (error);
}
- gtk_html_stream_close (handle, GTK_HTML_STREAM_OK);
+ /* Chain up with the URI for the icon file. */
+ class->url_requested (html, icon_uri, handle);
- g_free (filename);
+ gtk_icon_info_free (icon_info);
+ g_free (icon_uri);
return;
}
- /* Chain up to parent's url_requested() method. */
- GTK_HTML_CLASS (parent_class)->url_requested (html, url, handle);
+ /* Chain up to parent's uri_requested() method. */
+ class->url_requested (html, uri, handle);
}
static void
@@ -1019,38 +913,93 @@ contact_display_link_clicked (GtkHTML *html,
const gchar *uri)
{
EABContactDisplay *display;
+ gsize length;
display = EAB_CONTACT_DISPLAY (html);
-#ifdef HANDLE_MAILTO_INTERNALLY
- if (!strncmp (uri, "internal-mailto:", strlen ("internal-mailto:"))) {
- EDestination *destination;
- EContact *contact;
- gint email_num;
-
- email_num = atoi (uri + strlen ("internal-mailto:"));
- if (email_num == -1)
- return;
-
- destination = e_destination_new ();
- contact = eab_contact_display_get_contact (display);
- e_destination_set_contact (destination, contact, email_num);
- g_signal_emit (display, signals[SEND_MESSAGE], 0, destination);
- g_object_unref (destination);
+ length = strlen ("internal-mailto:");
+ if (g_ascii_strncasecmp (uri, "internal-mailto:", length) == 0) {
+ gint index;
+ index = atoi (uri + length);
+ contact_display_emit_send_message (display, index);
return;
}
-#endif
/* Chain up to parent's link_clicked() method. */
GTK_HTML_CLASS (parent_class)->link_clicked (html, uri);
}
static void
+contact_display_on_url (GtkHTML *html,
+ const gchar *uri)
+{
+ EABContactDisplay *display;
+ EContact *contact;
+ const gchar *name;
+ gchar *message;
+
+ if (uri == NULL || *uri == '\0')
+ goto chainup;
+
+ if (!g_str_has_prefix (uri, "internal-mailto:"))
+ goto chainup;
+
+ display = EAB_CONTACT_DISPLAY (html);
+ contact = eab_contact_display_get_contact (display);
+
+ name = e_contact_get_const (contact, E_CONTACT_FILE_AS);
+ if (name == NULL)
+ e_contact_get_const (contact, E_CONTACT_FULL_NAME);
+ g_return_if_fail (name != NULL);
+
+ message = g_strdup_printf (_("Click to mail %s"), name);
+ e_web_view_status_message (E_WEB_VIEW (html), message);
+ g_free (message);
+
+ return;
+
+chainup:
+ /* Chain up to parent's on_url() method. */
+ GTK_HTML_CLASS (parent_class)->on_url (html, uri);
+}
+
+static void
+contact_display_update_actions (EWebView *web_view)
+{
+ GtkActionGroup *action_group;
+ gboolean scheme_is_internal_mailto;
+ gboolean visible;
+ const gchar *group_name;
+ const gchar *uri;
+
+ /* Chain up to parent's update_actions() method. */
+ E_WEB_VIEW_CLASS (parent_class)->update_actions (web_view);
+
+ uri = e_web_view_get_selected_uri (web_view);
+
+ scheme_is_internal_mailto = (uri == NULL) ? FALSE :
+ (g_ascii_strncasecmp (uri, "internal-mailto:", 16) == 0);
+
+ /* Override how EWebView treats internal-mailto URIs. */
+ group_name = "uri";
+ action_group = e_web_view_get_action_group (web_view, group_name);
+ visible = gtk_action_group_get_visible (action_group);
+ visible &= !scheme_is_internal_mailto;
+ gtk_action_group_set_visible (action_group, visible);
+
+ group_name = "internal-mailto";
+ visible = scheme_is_internal_mailto;
+ action_group = e_web_view_get_action_group (web_view, group_name);
+ gtk_action_group_set_visible (action_group, visible);
+}
+
+static void
eab_contact_display_class_init (EABContactDisplayClass *class)
{
GObjectClass *object_class;
GtkHTMLClass *html_class;
+ EWebViewClass *web_view_class;
parent_class = g_type_class_peek_parent (class);
g_type_class_add_private (class, sizeof (EABContactDisplayPrivate));
@@ -1064,6 +1013,10 @@ eab_contact_display_class_init (EABContactDisplayClass *class)
html_class = GTK_HTML_CLASS (class);
html_class->url_requested = contact_display_url_requested;
html_class->link_clicked = contact_display_link_clicked;
+ html_class->on_url = contact_display_on_url;
+
+ web_view_class = E_WEB_VIEW_CLASS (class);
+ web_view_class->update_actions = contact_display_update_actions;
g_object_class_install_property (
object_class,
@@ -1102,65 +1055,33 @@ eab_contact_display_class_init (EABContactDisplayClass *class)
static void
eab_contact_display_init (EABContactDisplay *display)
{
+ EWebView *web_view;
+ GtkUIManager *ui_manager;
GtkActionGroup *action_group;
- GtkHTML *html;
- const gchar *id;
+ const gchar *domain = GETTEXT_PACKAGE;
+ GError *error = NULL;
display->priv = EAB_CONTACT_DISPLAY_GET_PRIVATE (display);
display->priv->mode = EAB_CONTACT_DISPLAY_RENDER_NORMAL;
- display->priv->ui_manager = gtk_ui_manager_new ();
- display->priv->invisible = gtk_invisible_new ();
- g_object_ref_sink (display->priv->invisible);
+ web_view = E_WEB_VIEW (display);
+ ui_manager = e_web_view_get_ui_manager (web_view);
+
+ action_group = gtk_action_group_new ("internal-mailto");
+ gtk_action_group_set_translation_domain (action_group, domain);
+ gtk_ui_manager_insert_action_group (ui_manager, action_group, 0);
+ g_object_unref (action_group);
- action_group = gtk_action_group_new ("email");
- gtk_action_group_set_translation_domain (
- action_group, GETTEXT_PACKAGE);
- gtk_action_group_add_actions (
- action_group, email_entries,
- G_N_ELEMENTS (email_entries), display);
- gtk_ui_manager_insert_action_group (
- display->priv->ui_manager, action_group, 0);
- display->priv->email_actions = action_group;
-
- action_group = gtk_action_group_new ("uri");
- gtk_action_group_set_translation_domain (
- action_group, GETTEXT_PACKAGE);
gtk_action_group_add_actions (
- action_group, uri_entries,
- G_N_ELEMENTS (uri_entries), display);
- gtk_ui_manager_insert_action_group (
- display->priv->ui_manager, action_group, 0);
- display->priv->uri_actions = action_group;
-
- gtk_ui_manager_add_ui_from_string (
- display->priv->ui_manager, ui, -1, NULL);
-
- html = GTK_HTML (display);
- gtk_html_construct (html);
- gtk_html_set_editable (html, FALSE);
- gtk_html_set_default_content_type (html, "text/html; charset=utf-8");
-
- g_signal_connect (
- display, "button-press-event",
- G_CALLBACK (contact_display_button_press_event), display);
-
- g_signal_connect_swapped (
- display->priv->invisible, "selection-get",
- G_CALLBACK (contact_display_selection_get), display);
- g_signal_connect_swapped (
- display->priv->invisible, "selection-clear-event",
- G_CALLBACK (contact_display_selection_clear_event), display);
- gtk_selection_add_target (
- display->priv->invisible,
- GDK_SELECTION_PRIMARY, GDK_SELECTION_TYPE_STRING, 0);
- gtk_selection_add_target (
- display->priv->invisible,
- GDK_SELECTION_CLIPBOARD, GDK_SELECTION_TYPE_STRING, 1);
-
- id = "org.gnome.evolution.contact-display";
- e_plugin_ui_register_manager (display->priv->ui_manager, id, display);
- e_plugin_ui_enable_manager (display->priv->ui_manager, id);
+ action_group, internal_mailto_entries,
+ G_N_ELEMENTS (internal_mailto_entries), display);
+
+ /* Because we are loading from a hard-coded string, there is
+ * no chance of I/O errors. Failure here implies a malformed
+ * UI definition. Full stop. */
+ gtk_ui_manager_add_ui_from_string (ui_manager, ui, -1, &error);
+ if (error != NULL)
+ g_error ("%s", error->message);
}
GType