1 /* GTK - The GIMP Toolkit
2 * gtklinkbutton.c - an hyperlink-enabled button
4 * Copyright (C) 2006 Emmanuele Bassi <ebassi@gmail.com>
7 * Based on gnome-href code by:
8 * James Henstridge <james@daa.com.au>
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Library General Public
12 * License as published by the Free Software Foundation; either
13 * version 2 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Library General Public License for more details.
20 * You should have received a copy of the GNU Library General Public
21 * License along with this library; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place - Suite 330, Cambridge, MA 02139, USA.
29 #include <gdk/gdkcolor.h>
30 #include <gdk/gdkcursor.h>
31 #include <gdk/gdkdisplay.h>
33 #include "gtkclipboard.h"
35 #include "gtkimagemenuitem.h"
39 #include "gtkmenuitem.h"
42 #include "gtklinkbutton.h"
48 struct _GtkLinkButtonPrivate
54 GtkWidget *popup_menu;
64 #define GTK_LINK_BUTTON_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), GTK_TYPE_LINK_BUTTON, GtkLinkButtonPrivate))
66 static void gtk_link_button_finalize (GObject *object);
67 static void gtk_link_button_get_property (GObject *object,
71 static void gtk_link_button_set_property (GObject *object,
75 static void gtk_link_button_add (GtkContainer *container,
77 static gboolean gtk_link_button_button_press (GtkWidget *widget,
78 GdkEventButton *event);
79 static void gtk_link_button_clicked (GtkButton *button);
80 static gboolean gtk_link_button_popup_menu (GtkWidget *widget);
81 static void gtk_link_button_style_set (GtkWidget *widget,
83 static gboolean gtk_link_button_enter_cb (GtkWidget *widget,
84 GdkEventCrossing *event,
86 static gboolean gtk_link_button_leave_cb (GtkWidget *widget,
87 GdkEventCrossing *event,
89 static void gtk_link_button_drag_data_get_cb (GtkWidget *widget,
90 GdkDragContext *context,
91 GtkSelectionData *selection,
97 static const GtkTargetEntry link_drop_types[] = {
98 { "text/uri-list", 0, 0 },
99 { "_NETSCAPE_URL", 0, 0 }
102 static const GdkColor default_link_color = { 0, 0, 0, 0xeeee };
103 static const GdkColor default_visited_link_color = { 0, 0x5555, 0x1a1a, 0x8b8b };
105 static GtkLinkButtonUriFunc uri_func = NULL;
106 static gpointer uri_func_data = NULL;
107 static GDestroyNotify uri_func_destroy = NULL;
109 G_DEFINE_TYPE (GtkLinkButton, gtk_link_button, GTK_TYPE_BUTTON)
112 gtk_link_button_class_init (GtkLinkButtonClass *klass)
114 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
115 GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
116 GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
117 GtkButtonClass *button_class = GTK_BUTTON_CLASS (klass);
119 gobject_class->set_property = gtk_link_button_set_property;
120 gobject_class->get_property = gtk_link_button_get_property;
121 gobject_class->finalize = gtk_link_button_finalize;
123 widget_class->button_press_event = gtk_link_button_button_press;
124 widget_class->popup_menu = gtk_link_button_popup_menu;
125 widget_class->style_set = gtk_link_button_style_set;
127 container_class->add = gtk_link_button_add;
129 button_class->clicked = gtk_link_button_clicked;
134 * The URI bound to this button.
138 g_object_class_install_property (gobject_class,
140 g_param_spec_string ("uri",
142 _("The URI bound to this button"),
143 "http://www.gtk.org",
146 g_type_class_add_private (gobject_class, sizeof (GtkLinkButtonPrivate));
150 gtk_link_button_init (GtkLinkButton *link_button)
152 link_button->priv = GTK_LINK_BUTTON_GET_PRIVATE (link_button),
154 gtk_button_set_relief (GTK_BUTTON (link_button), GTK_RELIEF_NONE);
156 g_signal_connect (link_button, "enter_notify_event",
157 G_CALLBACK (gtk_link_button_enter_cb), NULL);
158 g_signal_connect (link_button, "leave_notify_event",
159 G_CALLBACK (gtk_link_button_leave_cb), NULL);
160 g_signal_connect (link_button, "drag_data_get",
161 G_CALLBACK (gtk_link_button_drag_data_get_cb), NULL);
163 /* enable drag source */
164 gtk_drag_source_set (GTK_WIDGET (link_button),
166 link_drop_types, G_N_ELEMENTS (link_drop_types),
171 gtk_link_button_finalize (GObject *object)
173 GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
175 g_free (link_button->priv->uri);
177 G_OBJECT_CLASS (gtk_link_button_parent_class)->finalize (object);
181 gtk_link_button_get_property (GObject *object,
186 GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
191 g_value_set_string (value, link_button->priv->uri);
194 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
200 gtk_link_button_set_property (GObject *object,
205 GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
210 gtk_link_button_set_uri (link_button, g_value_get_string (value));
213 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
219 set_link_color (GtkLinkButton *link_button)
221 GdkColor *link_color = NULL;
224 label = gtk_bin_get_child (GTK_BIN (link_button));
226 if (link_button->priv->visited)
228 gtk_widget_style_get (GTK_WIDGET (link_button),
229 "visited-link-color", &link_color, NULL);
231 link_color = &default_visited_link_color;
235 gtk_widget_style_get (GTK_WIDGET (link_button),
236 "link-color", &link_color, NULL);
238 link_color = &default_link_color;
241 gtk_widget_modify_fg (label, GTK_STATE_NORMAL, link_color);
242 gtk_widget_modify_fg (label, GTK_STATE_ACTIVE, link_color);
243 gtk_widget_modify_fg (label, GTK_STATE_PRELIGHT, link_color);
244 gtk_widget_modify_fg (label, GTK_STATE_SELECTED, link_color);
246 if (link_color != &default_link_color &&
247 link_color != &default_visited_link_color)
248 gdk_color_free (link_color);
252 set_link_underline (GtkLinkButton *link_button)
256 label = gtk_bin_get_child (GTK_BIN (link_button));
257 if (GTK_IS_LABEL (label))
259 PangoAttrList *attributes;
260 PangoAttribute *uline;
262 uline = pango_attr_underline_new (PANGO_UNDERLINE_SINGLE);
263 uline->start_index = 0;
264 uline->end_index = G_MAXUINT;
265 attributes = pango_attr_list_new ();
266 pango_attr_list_insert (attributes, uline);
267 gtk_label_set_attributes (GTK_LABEL (label), attributes);
268 pango_attr_list_unref (attributes);
273 gtk_link_button_add (GtkContainer *container,
276 GTK_CONTAINER_CLASS (gtk_link_button_parent_class)->add (container, widget);
278 set_link_underline (GTK_LINK_BUTTON (container));
282 gtk_link_button_style_set (GtkWidget *widget,
285 GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
287 set_link_color (link_button);
291 set_hand_cursor (GtkWidget *widget,
297 display = gtk_widget_get_display (widget);
301 cursor = gdk_cursor_new_for_display (display, GDK_HAND2);
303 gdk_window_set_cursor (widget->window, cursor);
304 gdk_display_flush (display);
307 gdk_cursor_unref (cursor);
311 popup_menu_detach (GtkWidget *attach_widget,
314 GtkLinkButton *link_button = GTK_LINK_BUTTON (attach_widget);
316 link_button->priv->popup_menu = NULL;
320 popup_position_func (GtkMenu *menu,
326 GtkLinkButton *link_button = GTK_LINK_BUTTON (user_data);
327 GtkLinkButtonPrivate *priv = link_button->priv;
328 GtkWidget *widget = GTK_WIDGET (link_button);
329 GdkScreen *screen = gtk_widget_get_screen (widget);
332 GdkRectangle monitor;
334 g_return_if_fail (GTK_WIDGET_REALIZED (link_button));
336 gdk_window_get_origin (widget->window, x, y);
338 gtk_widget_size_request (priv->popup_menu, &req);
340 *x += widget->allocation.width / 2;
341 *y += widget->allocation.height;
343 monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
344 gtk_menu_set_monitor (menu, monitor_num);
345 gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
347 *x = CLAMP (*x, monitor.x, monitor.x + MAX (0, monitor.width - req.width));
348 *y = CLAMP (*y, monitor.y, monitor.y + MAX (0, monitor.height - req.height));
354 copy_activate_cb (GtkWidget *widget,
355 GtkLinkButton *link_button)
357 GtkLinkButtonPrivate *priv = link_button->priv;
359 gtk_clipboard_set_text (gtk_widget_get_clipboard (GTK_WIDGET (link_button),
360 GDK_SELECTION_CLIPBOARD),
365 gtk_link_button_do_popup (GtkLinkButton *link_button,
366 GdkEventButton *event)
368 GtkLinkButtonPrivate *priv = link_button->priv;
374 button = event->button;
380 time = gtk_get_current_event_time ();
383 if (GTK_WIDGET_REALIZED (link_button))
385 GtkWidget *menu_item;
387 if (priv->popup_menu)
388 gtk_widget_destroy (priv->popup_menu);
390 priv->popup_menu = gtk_menu_new ();
392 gtk_menu_attach_to_widget (GTK_MENU (priv->popup_menu),
393 GTK_WIDGET (link_button),
396 menu_item = gtk_image_menu_item_new_with_mnemonic (_("Copy URL"));
397 gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item),
398 gtk_image_new_from_stock (GTK_STOCK_COPY,
399 GTK_ICON_SIZE_MENU));
400 g_signal_connect (menu_item, "activate",
401 G_CALLBACK (copy_activate_cb), link_button);
402 gtk_widget_show (menu_item);
403 gtk_menu_shell_append (GTK_MENU_SHELL (priv->popup_menu), menu_item);
406 gtk_menu_popup (GTK_MENU (priv->popup_menu), NULL, NULL,
411 gtk_menu_popup (GTK_MENU (priv->popup_menu), NULL, NULL,
412 popup_position_func, link_button,
414 gtk_menu_shell_select_first (GTK_MENU_SHELL (priv->popup_menu), FALSE);
420 gtk_link_button_button_press (GtkWidget *widget,
421 GdkEventButton *event)
423 if (!GTK_WIDGET_HAS_FOCUS (widget))
424 gtk_widget_grab_focus (widget);
426 if ((event->button == 3) && (event->type == GDK_BUTTON_PRESS))
428 gtk_link_button_do_popup (GTK_LINK_BUTTON (widget), event);
433 if (GTK_WIDGET_CLASS (gtk_link_button_parent_class)->button_press_event)
434 return (* GTK_WIDGET_CLASS (gtk_link_button_parent_class)->button_press_event) (widget, event);
440 gtk_link_button_clicked (GtkButton *button)
442 GtkLinkButton *link_button = GTK_LINK_BUTTON (button);
445 (* uri_func) (link_button, link_button->priv->uri, uri_func_data);
447 link_button->priv->visited = TRUE;
449 set_link_color (link_button);
453 gtk_link_button_popup_menu (GtkWidget *widget)
455 gtk_link_button_do_popup (GTK_LINK_BUTTON (widget), NULL);
461 gtk_link_button_enter_cb (GtkWidget *widget,
462 GdkEventCrossing *crossing,
465 set_hand_cursor (widget, TRUE);
471 gtk_link_button_leave_cb (GtkWidget *widget,
472 GdkEventCrossing *crossing,
475 set_hand_cursor (widget, FALSE);
481 gtk_link_button_drag_data_get_cb (GtkWidget *widget,
482 GdkDragContext *context,
483 GtkSelectionData *selection,
488 GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
491 uri = g_strdup_printf ("%s\r\n", link_button->priv->uri);
492 gtk_selection_data_set (selection,
502 * gtk_link_button_new:
505 * Creates a new #GtkLinkButton with the URI as its text.
507 * Return value: a new link button widget.
512 gtk_link_button_new (const gchar *uri)
514 gchar *utf8_uri = NULL;
517 g_return_val_if_fail (uri != NULL, NULL);
519 if (g_utf8_validate (uri, -1, NULL))
521 utf8_uri = g_strdup (uri);
525 GError *conv_err = NULL;
527 utf8_uri = g_locale_to_utf8 (uri, -1, NULL, NULL, &conv_err);
530 g_warning ("Attempting to convert URI `%s' to UTF-8, but failed "
534 g_error_free (conv_err);
536 utf8_uri = g_strdup (_("Invalid URI"));
540 retval = g_object_new (GTK_TYPE_LINK_BUTTON,
551 * gtk_link_button_new_with_label:
553 * @label: the text of the button
555 * Creates a new #GtkLinkButton containing a label.
557 * Return value: a new link button widget.
562 gtk_link_button_new_with_label (const gchar *uri,
567 g_return_val_if_fail (uri != NULL, NULL);
570 return gtk_link_button_new (uri);
572 retval = g_object_new (GTK_TYPE_LINK_BUTTON,
581 * gtk_link_button_set_uri:
582 * @link_button: a #GtkLinkButton
585 * Sets @uri as the URI where the #GtkLinkButton points.
590 gtk_link_button_set_uri (GtkLinkButton *link_button,
595 g_return_if_fail (GTK_IS_LINK_BUTTON (link_button));
596 g_return_if_fail (uri != NULL);
598 tmp = link_button->priv->uri;
599 link_button->priv->uri = g_strdup (uri);
602 link_button->priv->visited = FALSE;
604 g_object_notify (G_OBJECT (link_button), "uri");
608 * gtk_link_button_get_uri:
609 * @link_button: a #GtkLinkButton
611 * Retrieves the URI set using gtk_link_button_set_uri().
613 * Return value: a valid URI. The returned string is owned by the link button
614 * and should not be modified or freed.
618 G_CONST_RETURN gchar *
619 gtk_link_button_get_uri (GtkLinkButton *link_button)
621 g_return_val_if_fail (GTK_IS_LINK_BUTTON (link_button), NULL);
623 return link_button->priv->uri;
627 * gtk_link_button_set_uri_hook:
628 * @func: a function called each time a #GtkLinkButton is clicked, or %NULL
629 * @data: user data to be passed to @func, or %NULL
630 * @destroy: a #GDestroyNotify that gets called when @data is no longer needed, or %NULL
632 * Sets @func as the function that should be invoked every time a user clicks
633 * a #GtkLinkButton. This function is called before every callback registered
634 * for the "clicked" signal.
636 * Return value: the previously set hook function.
641 gtk_link_button_set_uri_hook (GtkLinkButtonUriFunc func,
643 GDestroyNotify destroy)
645 GtkLinkButtonUriFunc old_uri_func;
647 if (uri_func_destroy)
648 (* uri_func_destroy) (uri_func_data);
650 old_uri_func = uri_func;
653 uri_func_data = data;
654 uri_func_destroy = destroy;
659 #define __GTK_LINK_BUTTON_C__
660 #include "gtkaliasdef.c"