2008-07-01 22:57:50 +00:00
|
|
|
/* GTK - The GIMP Toolkit
|
2006-01-24 03:35:53 +00:00
|
|
|
* gtklinkbutton.c - an hyperlink-enabled button
|
2011-01-04 22:32:12 +00:00
|
|
|
*
|
2006-01-24 03:35:53 +00:00
|
|
|
* Copyright (C) 2006 Emmanuele Bassi <ebassi@gmail.com>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Based on gnome-href code by:
|
2011-01-04 22:32:12 +00:00
|
|
|
* James Henstridge <james@daa.com.au>
|
|
|
|
*
|
2006-01-24 03:35:53 +00:00
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
2012-02-27 13:01:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
2006-01-24 03:35:53 +00:00
|
|
|
*/
|
|
|
|
|
2010-09-24 17:41:29 +00:00
|
|
|
/**
|
|
|
|
* SECTION:gtklinkbutton
|
|
|
|
* @Title: GtkLinkButton
|
|
|
|
* @Short_description: Create buttons bound to a URL
|
|
|
|
* @See_also: #GtkButton
|
|
|
|
*
|
|
|
|
* A GtkLinkButton is a #GtkButton with a hyperlink, similar to the one
|
|
|
|
* used by web browsers, which triggers an action when clicked. It is useful
|
|
|
|
* to show quick links to resources.
|
|
|
|
*
|
|
|
|
* A link button is created by calling either gtk_link_button_new() or
|
|
|
|
* gtk_link_button_new_with_label(). If using the former, the URI you pass
|
|
|
|
* to the constructor is used as a label for the widget.
|
|
|
|
*
|
|
|
|
* The URI bound to a GtkLinkButton can be set specifically using
|
|
|
|
* gtk_link_button_set_uri(), and retrieved using gtk_link_button_get_uri().
|
|
|
|
*
|
|
|
|
* By default, GtkLinkButton calls gtk_show_uri() when the button is
|
|
|
|
* clicked. This behaviour can be overridden by connecting to the
|
2010-10-14 14:22:19 +00:00
|
|
|
* #GtkLinkButton::activate-link signal and returning %TRUE from the
|
|
|
|
* signal handler.
|
2015-10-30 04:50:48 +00:00
|
|
|
*
|
|
|
|
* # CSS nodes
|
|
|
|
*
|
|
|
|
* GtkLinkButton has a single CSS node with name button. To differentiate
|
|
|
|
* it from a plain #GtkButton, it gets the .link style class.
|
2010-09-24 17:41:29 +00:00
|
|
|
*/
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2010-09-14 01:33:06 +00:00
|
|
|
#include "gtklinkbutton.h"
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "gtkclipboard.h"
|
|
|
|
#include "gtkdnd.h"
|
|
|
|
#include "gtklabel.h"
|
2011-10-23 11:57:07 +00:00
|
|
|
#include "gtkmain.h"
|
2010-10-14 14:22:19 +00:00
|
|
|
#include "gtkmarshalers.h"
|
2006-01-24 03:35:53 +00:00
|
|
|
#include "gtkmenu.h"
|
|
|
|
#include "gtkmenuitem.h"
|
2010-09-14 01:33:06 +00:00
|
|
|
#include "gtksizerequest.h"
|
2008-12-08 02:47:21 +00:00
|
|
|
#include "gtkshow.h"
|
2008-02-16 05:28:22 +00:00
|
|
|
#include "gtktooltip.h"
|
2011-09-18 22:32:52 +00:00
|
|
|
#include "gtkprivate.h"
|
2006-01-24 03:35:53 +00:00
|
|
|
#include "gtkintl.h"
|
|
|
|
|
2011-06-27 03:48:54 +00:00
|
|
|
#include "a11y/gtklinkbuttonaccessible.h"
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
struct _GtkLinkButtonPrivate
|
|
|
|
{
|
|
|
|
gchar *uri;
|
|
|
|
|
|
|
|
gboolean visited;
|
|
|
|
|
|
|
|
GtkWidget *popup_menu;
|
2016-12-02 11:22:41 +00:00
|
|
|
GtkGesture *click_gesture;
|
2006-01-24 03:35:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
2008-08-07 11:37:57 +00:00
|
|
|
PROP_URI,
|
|
|
|
PROP_VISITED
|
2006-01-24 03:35:53 +00:00
|
|
|
};
|
|
|
|
|
2010-10-14 14:22:19 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
ACTIVATE_LINK,
|
|
|
|
|
|
|
|
LAST_SIGNAL
|
|
|
|
};
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
static void gtk_link_button_finalize (GObject *object);
|
|
|
|
static void gtk_link_button_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec);
|
|
|
|
static void gtk_link_button_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec);
|
|
|
|
static void gtk_link_button_clicked (GtkButton *button);
|
|
|
|
static gboolean gtk_link_button_popup_menu (GtkWidget *widget);
|
2011-09-16 14:04:44 +00:00
|
|
|
static void gtk_link_button_unrealize (GtkWidget *widget);
|
2006-01-24 03:35:53 +00:00
|
|
|
static gboolean gtk_link_button_enter_cb (GtkWidget *widget,
|
|
|
|
GdkEventCrossing *event,
|
|
|
|
gpointer user_data);
|
|
|
|
static gboolean gtk_link_button_leave_cb (GtkWidget *widget,
|
|
|
|
GdkEventCrossing *event,
|
|
|
|
gpointer user_data);
|
|
|
|
static void gtk_link_button_drag_data_get_cb (GtkWidget *widget,
|
|
|
|
GdkDragContext *context,
|
|
|
|
GtkSelectionData *selection,
|
|
|
|
guint _info,
|
|
|
|
guint _time,
|
|
|
|
gpointer user_data);
|
2008-02-16 05:28:22 +00:00
|
|
|
static gboolean gtk_link_button_query_tooltip_cb (GtkWidget *widget,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
gboolean keyboard_tip,
|
|
|
|
GtkTooltip *tooltip,
|
|
|
|
gpointer data);
|
2016-12-02 11:22:41 +00:00
|
|
|
static void gtk_link_button_pressed_cb (GtkGestureMultiPress *gesture,
|
|
|
|
int n_press,
|
|
|
|
double x,
|
|
|
|
double y,
|
|
|
|
gpointer user_data);
|
|
|
|
|
2010-10-14 14:22:19 +00:00
|
|
|
static gboolean gtk_link_button_activate_link (GtkLinkButton *link_button);
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
static const GtkTargetEntry link_drop_types[] = {
|
2016-10-15 20:24:42 +00:00
|
|
|
{ (char *) "text/uri-list", 0, 0 },
|
|
|
|
{ (char *) "_NETSCAPE_URL", 0, 0 }
|
2006-01-24 03:35:53 +00:00
|
|
|
};
|
|
|
|
|
2010-10-14 14:22:19 +00:00
|
|
|
static guint link_signals[LAST_SIGNAL] = { 0, };
|
|
|
|
|
2013-06-27 19:02:52 +00:00
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (GtkLinkButton, gtk_link_button, GTK_TYPE_BUTTON)
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_class_init (GtkLinkButtonClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
GtkButtonClass *button_class = GTK_BUTTON_CLASS (klass);
|
2015-07-05 17:31:28 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
gobject_class->set_property = gtk_link_button_set_property;
|
|
|
|
gobject_class->get_property = gtk_link_button_get_property;
|
|
|
|
gobject_class->finalize = gtk_link_button_finalize;
|
2015-07-05 17:31:28 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
widget_class->popup_menu = gtk_link_button_popup_menu;
|
2011-09-16 14:04:44 +00:00
|
|
|
widget_class->unrealize = gtk_link_button_unrealize;
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
button_class->clicked = gtk_link_button_clicked;
|
|
|
|
|
2010-10-14 14:22:19 +00:00
|
|
|
klass->activate_link = gtk_link_button_activate_link;
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
/**
|
2012-04-13 00:07:28 +00:00
|
|
|
* GtkLinkButton:uri:
|
|
|
|
*
|
|
|
|
* The URI bound to this button.
|
2006-01-24 03:35:53 +00:00
|
|
|
*
|
|
|
|
* Since: 2.10
|
|
|
|
*/
|
|
|
|
g_object_class_install_property (gobject_class,
|
2014-06-09 13:04:33 +00:00
|
|
|
PROP_URI,
|
|
|
|
g_param_spec_string ("uri",
|
|
|
|
P_("URI"),
|
|
|
|
P_("The URI bound to this button"),
|
|
|
|
NULL,
|
|
|
|
GTK_PARAM_READWRITE));
|
|
|
|
|
2008-08-07 11:37:57 +00:00
|
|
|
/**
|
2012-04-13 00:07:28 +00:00
|
|
|
* GtkLinkButton:visited:
|
|
|
|
*
|
2008-08-07 11:37:57 +00:00
|
|
|
* The 'visited' state of this button. A visited link is drawn in a
|
|
|
|
* different color.
|
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
*/
|
|
|
|
g_object_class_install_property (gobject_class,
|
2014-06-09 13:04:33 +00:00
|
|
|
PROP_VISITED,
|
|
|
|
g_param_spec_boolean ("visited",
|
2008-08-07 11:37:57 +00:00
|
|
|
P_("Visited"),
|
|
|
|
P_("Whether this link has been visited."),
|
|
|
|
FALSE,
|
2014-06-09 13:04:33 +00:00
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY));
|
2010-10-14 14:22:19 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* GtkLinkButton::activate-link:
|
|
|
|
* @button: the #GtkLinkButton that emitted the signal
|
|
|
|
*
|
|
|
|
* The ::activate-link signal is emitted each time the #GtkLinkButton
|
|
|
|
* has been clicked.
|
|
|
|
*
|
|
|
|
* The default handler will call gtk_show_uri() with the URI stored inside
|
|
|
|
* the #GtkLinkButton:uri property.
|
|
|
|
*
|
|
|
|
* To override the default behavior, you can connect to the ::activate-link
|
|
|
|
* signal and stop the propagation of the signal by returning %TRUE from
|
|
|
|
* your handler.
|
|
|
|
*/
|
|
|
|
link_signals[ACTIVATE_LINK] =
|
|
|
|
g_signal_new (I_("activate-link"),
|
|
|
|
G_TYPE_FROM_CLASS (klass),
|
|
|
|
G_SIGNAL_RUN_LAST,
|
|
|
|
G_STRUCT_OFFSET (GtkLinkButtonClass, activate_link),
|
|
|
|
_gtk_boolean_handled_accumulator, NULL,
|
|
|
|
_gtk_marshal_BOOLEAN__VOID,
|
|
|
|
G_TYPE_BOOLEAN, 0);
|
2011-06-27 03:48:54 +00:00
|
|
|
|
|
|
|
gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_LINK_BUTTON_ACCESSIBLE);
|
2015-10-30 04:50:48 +00:00
|
|
|
gtk_widget_class_set_css_name (widget_class, "button");
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_init (GtkLinkButton *link_button)
|
|
|
|
{
|
2016-12-02 11:22:41 +00:00
|
|
|
GtkLinkButtonPrivate *priv = gtk_link_button_get_instance_private (link_button);
|
2015-10-30 04:50:48 +00:00
|
|
|
GtkStyleContext *context;
|
|
|
|
|
2016-12-02 11:22:41 +00:00
|
|
|
link_button->priv = priv;
|
2010-06-28 04:10:15 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
gtk_button_set_relief (GTK_BUTTON (link_button), GTK_RELIEF_NONE);
|
2013-10-08 10:50:08 +00:00
|
|
|
gtk_widget_set_state_flags (GTK_WIDGET (link_button), GTK_STATE_FLAG_LINK, FALSE);
|
|
|
|
|
2008-08-07 11:37:57 +00:00
|
|
|
g_signal_connect (link_button, "enter-notify-event",
|
2006-01-24 03:35:53 +00:00
|
|
|
G_CALLBACK (gtk_link_button_enter_cb), NULL);
|
2008-08-07 11:37:57 +00:00
|
|
|
g_signal_connect (link_button, "leave-notify-event",
|
2006-01-24 03:35:53 +00:00
|
|
|
G_CALLBACK (gtk_link_button_leave_cb), NULL);
|
2008-08-07 11:37:57 +00:00
|
|
|
g_signal_connect (link_button, "drag-data-get",
|
2006-01-24 03:35:53 +00:00
|
|
|
G_CALLBACK (gtk_link_button_drag_data_get_cb), NULL);
|
2008-08-07 11:37:57 +00:00
|
|
|
|
2008-02-16 05:28:22 +00:00
|
|
|
g_object_set (link_button, "has-tooltip", TRUE, NULL);
|
|
|
|
g_signal_connect (link_button, "query-tooltip",
|
|
|
|
G_CALLBACK (gtk_link_button_query_tooltip_cb), NULL);
|
2013-10-08 10:50:08 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
/* enable drag source */
|
|
|
|
gtk_drag_source_set (GTK_WIDGET (link_button),
|
|
|
|
GDK_BUTTON1_MASK,
|
|
|
|
link_drop_types, G_N_ELEMENTS (link_drop_types),
|
|
|
|
GDK_ACTION_COPY);
|
2015-10-30 04:50:48 +00:00
|
|
|
|
2016-12-02 11:22:41 +00:00
|
|
|
priv->click_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (link_button));
|
|
|
|
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (priv->click_gesture), FALSE);
|
|
|
|
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (priv->click_gesture), 0);
|
|
|
|
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (priv->click_gesture), GTK_PHASE_BUBBLE);
|
|
|
|
g_signal_connect (priv->click_gesture, "pressed", G_CALLBACK (gtk_link_button_pressed_cb),
|
|
|
|
link_button);
|
|
|
|
|
2015-10-30 04:50:48 +00:00
|
|
|
context = gtk_widget_get_style_context (GTK_WIDGET (link_button));
|
|
|
|
gtk_style_context_add_class (context, "link");
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
|
|
|
|
|
|
|
|
g_free (link_button->priv->uri);
|
2016-12-02 11:22:41 +00:00
|
|
|
g_object_unref (link_button->priv->click_gesture);
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_link_button_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_URI:
|
|
|
|
g_value_set_string (value, link_button->priv->uri);
|
|
|
|
break;
|
2008-08-07 11:37:57 +00:00
|
|
|
case PROP_VISITED:
|
|
|
|
g_value_set_boolean (value, link_button->priv->visited);
|
|
|
|
break;
|
2006-01-24 03:35:53 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_URI:
|
|
|
|
gtk_link_button_set_uri (link_button, g_value_get_string (value));
|
|
|
|
break;
|
2008-08-07 11:37:57 +00:00
|
|
|
case PROP_VISITED:
|
|
|
|
gtk_link_button_set_visited (link_button, g_value_get_boolean (value));
|
|
|
|
break;
|
2006-01-24 03:35:53 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_hand_cursor (GtkWidget *widget,
|
|
|
|
gboolean show_hand)
|
|
|
|
{
|
|
|
|
GdkDisplay *display;
|
|
|
|
GdkCursor *cursor;
|
|
|
|
|
|
|
|
display = gtk_widget_get_display (widget);
|
|
|
|
|
|
|
|
cursor = NULL;
|
|
|
|
if (show_hand)
|
2015-05-10 02:50:02 +00:00
|
|
|
cursor = gdk_cursor_new_from_name (display, "pointer");
|
2006-01-24 03:35:53 +00:00
|
|
|
|
2010-08-11 21:09:12 +00:00
|
|
|
gdk_window_set_cursor (gtk_widget_get_window (widget), cursor);
|
2006-01-24 03:35:53 +00:00
|
|
|
gdk_display_flush (display);
|
|
|
|
|
|
|
|
if (cursor)
|
2010-12-20 02:14:38 +00:00
|
|
|
g_object_unref (cursor);
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
|
2011-09-16 14:04:44 +00:00
|
|
|
static void
|
|
|
|
gtk_link_button_unrealize (GtkWidget *widget)
|
|
|
|
{
|
|
|
|
set_hand_cursor (widget, FALSE);
|
|
|
|
|
|
|
|
GTK_WIDGET_CLASS (gtk_link_button_parent_class)->unrealize (widget);
|
|
|
|
}
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
static void
|
|
|
|
popup_menu_detach (GtkWidget *attach_widget,
|
|
|
|
GtkMenu *menu)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (attach_widget);
|
|
|
|
|
|
|
|
link_button->priv->popup_menu = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
copy_activate_cb (GtkWidget *widget,
|
|
|
|
GtkLinkButton *link_button)
|
|
|
|
{
|
|
|
|
GtkLinkButtonPrivate *priv = link_button->priv;
|
|
|
|
|
|
|
|
gtk_clipboard_set_text (gtk_widget_get_clipboard (GTK_WIDGET (link_button),
|
|
|
|
GDK_SELECTION_CLIPBOARD),
|
|
|
|
priv->uri, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_do_popup (GtkLinkButton *link_button,
|
2016-07-12 18:08:36 +00:00
|
|
|
const GdkEvent *event)
|
2006-01-24 03:35:53 +00:00
|
|
|
{
|
|
|
|
GtkLinkButtonPrivate *priv = link_button->priv;
|
|
|
|
|
2010-03-02 06:16:02 +00:00
|
|
|
if (gtk_widget_get_realized (GTK_WIDGET (link_button)))
|
2006-01-24 03:35:53 +00:00
|
|
|
{
|
|
|
|
GtkWidget *menu_item;
|
2013-08-18 14:45:24 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
if (priv->popup_menu)
|
|
|
|
gtk_widget_destroy (priv->popup_menu);
|
|
|
|
|
|
|
|
priv->popup_menu = gtk_menu_new ();
|
2013-08-18 14:45:24 +00:00
|
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (priv->popup_menu),
|
|
|
|
GTK_STYLE_CLASS_CONTEXT_MENU);
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
gtk_menu_attach_to_widget (GTK_MENU (priv->popup_menu),
|
|
|
|
GTK_WIDGET (link_button),
|
|
|
|
popup_menu_detach);
|
|
|
|
|
2013-06-24 23:28:53 +00:00
|
|
|
menu_item = gtk_menu_item_new_with_mnemonic (_("Copy URL"));
|
2006-01-24 03:35:53 +00:00
|
|
|
g_signal_connect (menu_item, "activate",
|
|
|
|
G_CALLBACK (copy_activate_cb), link_button);
|
|
|
|
gtk_widget_show (menu_item);
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (priv->popup_menu), menu_item);
|
2013-08-18 14:45:24 +00:00
|
|
|
|
2016-07-12 18:08:36 +00:00
|
|
|
if (event && gdk_event_triggers_context_menu (event))
|
|
|
|
gtk_menu_popup_at_pointer (GTK_MENU (priv->popup_menu), event);
|
2006-01-24 03:35:53 +00:00
|
|
|
else
|
|
|
|
{
|
2016-07-12 18:08:36 +00:00
|
|
|
gtk_menu_popup_at_widget (GTK_MENU (priv->popup_menu),
|
|
|
|
GTK_WIDGET (link_button),
|
|
|
|
GDK_GRAVITY_SOUTH,
|
|
|
|
GDK_GRAVITY_NORTH_WEST,
|
|
|
|
event);
|
|
|
|
|
|
|
|
gtk_menu_shell_select_first (GTK_MENU_SHELL (priv->popup_menu), FALSE);
|
|
|
|
}
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-02 11:22:41 +00:00
|
|
|
static void
|
|
|
|
gtk_link_button_pressed_cb (GtkGestureMultiPress *gesture,
|
|
|
|
int n_press,
|
|
|
|
double x,
|
|
|
|
double y,
|
|
|
|
gpointer user_data)
|
2006-01-24 03:35:53 +00:00
|
|
|
{
|
2016-12-02 11:22:41 +00:00
|
|
|
GtkLinkButton *link_button = user_data;
|
|
|
|
GtkLinkButtonPrivate *priv = gtk_link_button_get_instance_private (link_button);
|
|
|
|
GdkEventSequence *sequence = gtk_gesture_single_get_current_sequence (GTK_GESTURE_SINGLE (gesture));
|
|
|
|
const GdkEvent *event = gtk_gesture_get_last_event (GTK_GESTURE (gesture), sequence);
|
2006-01-24 03:35:53 +00:00
|
|
|
|
2016-12-02 11:22:41 +00:00
|
|
|
if (!gtk_widget_has_focus (GTK_WIDGET (link_button)))
|
|
|
|
gtk_widget_grab_focus (GTK_WIDGET (link_button));
|
2011-09-27 13:37:28 +00:00
|
|
|
|
2016-12-02 11:22:41 +00:00
|
|
|
if (gdk_event_triggers_context_menu (event) &&
|
|
|
|
priv->uri != NULL)
|
|
|
|
{
|
|
|
|
gtk_link_button_do_popup (link_button, event);
|
|
|
|
gtk_gesture_set_state (GTK_GESTURE (gesture), GTK_EVENT_SEQUENCE_CLAIMED);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_gesture_set_state (GTK_GESTURE (gesture), GTK_EVENT_SEQUENCE_DENIED);
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-14 14:22:19 +00:00
|
|
|
static gboolean
|
|
|
|
gtk_link_button_activate_link (GtkLinkButton *link_button)
|
2006-01-24 03:35:53 +00:00
|
|
|
{
|
2016-07-13 15:23:47 +00:00
|
|
|
GtkWidget *toplevel;
|
2010-09-22 01:55:05 +00:00
|
|
|
GError *error;
|
2006-01-24 03:35:53 +00:00
|
|
|
|
2016-07-13 15:23:47 +00:00
|
|
|
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (link_button));
|
2008-12-08 02:39:28 +00:00
|
|
|
|
2010-09-22 01:55:05 +00:00
|
|
|
error = NULL;
|
2016-07-13 15:23:47 +00:00
|
|
|
gtk_show_uri_on_window (GTK_WINDOW (toplevel), link_button->priv->uri, GDK_CURRENT_TIME, &error);
|
2010-09-22 01:55:05 +00:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
g_warning ("Unable to show '%s': %s",
|
|
|
|
link_button->priv->uri,
|
|
|
|
error->message);
|
|
|
|
g_error_free (error);
|
2010-10-14 14:22:19 +00:00
|
|
|
|
|
|
|
return FALSE;
|
2008-12-08 02:39:28 +00:00
|
|
|
}
|
2006-01-28 06:20:30 +00:00
|
|
|
|
2008-08-07 11:37:57 +00:00
|
|
|
gtk_link_button_set_visited (link_button, TRUE);
|
2010-10-14 14:22:19 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_clicked (GtkButton *button)
|
|
|
|
{
|
|
|
|
gboolean retval = FALSE;
|
|
|
|
|
|
|
|
g_signal_emit (button, link_signals[ACTIVATE_LINK], 0, &retval);
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_link_button_popup_menu (GtkWidget *widget)
|
|
|
|
{
|
|
|
|
gtk_link_button_do_popup (GTK_LINK_BUTTON (widget), NULL);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_link_button_enter_cb (GtkWidget *widget,
|
|
|
|
GdkEventCrossing *crossing,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
set_hand_cursor (widget, TRUE);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gtk_link_button_leave_cb (GtkWidget *widget,
|
|
|
|
GdkEventCrossing *crossing,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
set_hand_cursor (widget, FALSE);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_link_button_drag_data_get_cb (GtkWidget *widget,
|
|
|
|
GdkDragContext *context,
|
|
|
|
GtkSelectionData *selection,
|
|
|
|
guint _info,
|
|
|
|
guint _time,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
|
|
|
|
gchar *uri;
|
|
|
|
|
|
|
|
uri = g_strdup_printf ("%s\r\n", link_button->priv->uri);
|
|
|
|
gtk_selection_data_set (selection,
|
2010-12-15 23:54:49 +00:00
|
|
|
gtk_selection_data_get_target (selection),
|
2006-01-24 03:35:53 +00:00
|
|
|
8,
|
|
|
|
(guchar *) uri,
|
|
|
|
strlen (uri));
|
|
|
|
|
|
|
|
g_free (uri);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_link_button_new:
|
|
|
|
* @uri: a valid URI
|
|
|
|
*
|
|
|
|
* Creates a new #GtkLinkButton with the URI as its text.
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
* Returns: a new link button widget.
|
2006-01-24 03:35:53 +00:00
|
|
|
*
|
|
|
|
* Since: 2.10
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_link_button_new (const gchar *uri)
|
|
|
|
{
|
|
|
|
gchar *utf8_uri = NULL;
|
|
|
|
GtkWidget *retval;
|
|
|
|
|
|
|
|
g_return_val_if_fail (uri != NULL, NULL);
|
|
|
|
|
|
|
|
if (g_utf8_validate (uri, -1, NULL))
|
|
|
|
{
|
|
|
|
utf8_uri = g_strdup (uri);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GError *conv_err = NULL;
|
|
|
|
|
|
|
|
utf8_uri = g_locale_to_utf8 (uri, -1, NULL, NULL, &conv_err);
|
|
|
|
if (conv_err)
|
|
|
|
{
|
2015-09-20 06:32:48 +00:00
|
|
|
g_warning ("Attempting to convert URI '%s' to UTF-8, but failed "
|
2016-02-28 16:06:25 +00:00
|
|
|
"with error: %s",
|
2006-01-24 03:35:53 +00:00
|
|
|
uri,
|
|
|
|
conv_err->message);
|
|
|
|
g_error_free (conv_err);
|
|
|
|
|
|
|
|
utf8_uri = g_strdup (_("Invalid URI"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
retval = g_object_new (GTK_TYPE_LINK_BUTTON,
|
2008-02-11 09:55:29 +00:00
|
|
|
"label", utf8_uri,
|
2008-02-16 05:28:22 +00:00
|
|
|
"uri", uri,
|
2006-01-24 03:35:53 +00:00
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_free (utf8_uri);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_link_button_new_with_label:
|
|
|
|
* @uri: a valid URI
|
2009-12-10 10:23:40 +00:00
|
|
|
* @label: (allow-none): the text of the button
|
2006-01-24 03:35:53 +00:00
|
|
|
*
|
|
|
|
* Creates a new #GtkLinkButton containing a label.
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
* Returns: (transfer none): a new link button widget.
|
2006-01-24 03:35:53 +00:00
|
|
|
*
|
|
|
|
* Since: 2.10
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_link_button_new_with_label (const gchar *uri,
|
|
|
|
const gchar *label)
|
|
|
|
{
|
|
|
|
GtkWidget *retval;
|
|
|
|
|
2006-04-01 00:25:00 +00:00
|
|
|
g_return_val_if_fail (uri != NULL, NULL);
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
if (!label)
|
|
|
|
return gtk_link_button_new (uri);
|
|
|
|
|
|
|
|
retval = g_object_new (GTK_TYPE_LINK_BUTTON,
|
|
|
|
"label", label,
|
|
|
|
"uri", uri,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2008-02-16 05:28:22 +00:00
|
|
|
static gboolean
|
|
|
|
gtk_link_button_query_tooltip_cb (GtkWidget *widget,
|
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
gboolean keyboard_tip,
|
|
|
|
GtkTooltip *tooltip,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
|
|
|
|
const gchar *label, *uri;
|
|
|
|
|
|
|
|
label = gtk_button_get_label (GTK_BUTTON (link_button));
|
|
|
|
uri = link_button->priv->uri;
|
|
|
|
|
2008-10-30 08:49:18 +00:00
|
|
|
if (!gtk_widget_get_tooltip_text (widget)
|
|
|
|
&& !gtk_widget_get_tooltip_markup (widget)
|
|
|
|
&& label && *label != '\0' && uri && strcmp (label, uri) != 0)
|
2008-02-16 05:28:22 +00:00
|
|
|
{
|
|
|
|
gtk_tooltip_set_text (tooltip, uri);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-08 10:50:08 +00:00
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
/**
|
|
|
|
* gtk_link_button_set_uri:
|
|
|
|
* @link_button: a #GtkLinkButton
|
|
|
|
* @uri: a valid URI
|
|
|
|
*
|
2008-08-07 11:37:57 +00:00
|
|
|
* Sets @uri as the URI where the #GtkLinkButton points. As a side-effect
|
2014-02-07 19:03:49 +00:00
|
|
|
* this unsets the “visited” state of the button.
|
2006-01-24 03:35:53 +00:00
|
|
|
*
|
|
|
|
* Since: 2.10
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_link_button_set_uri (GtkLinkButton *link_button,
|
|
|
|
const gchar *uri)
|
|
|
|
{
|
2008-02-16 05:28:22 +00:00
|
|
|
GtkLinkButtonPrivate *priv;
|
2006-01-24 03:35:53 +00:00
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_LINK_BUTTON (link_button));
|
|
|
|
g_return_if_fail (uri != NULL);
|
|
|
|
|
2008-02-16 05:28:22 +00:00
|
|
|
priv = link_button->priv;
|
|
|
|
|
|
|
|
g_free (priv->uri);
|
|
|
|
priv->uri = g_strdup (uri);
|
|
|
|
|
2006-01-24 03:35:53 +00:00
|
|
|
g_object_notify (G_OBJECT (link_button), "uri");
|
2008-08-07 11:37:57 +00:00
|
|
|
|
|
|
|
gtk_link_button_set_visited (link_button, FALSE);
|
2006-01-24 03:35:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_link_button_get_uri:
|
|
|
|
* @link_button: a #GtkLinkButton
|
|
|
|
*
|
|
|
|
* Retrieves the URI set using gtk_link_button_set_uri().
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
* Returns: a valid URI. The returned string is owned by the link button
|
2006-01-24 03:35:53 +00:00
|
|
|
* and should not be modified or freed.
|
|
|
|
*
|
|
|
|
* Since: 2.10
|
|
|
|
*/
|
2011-06-06 18:13:44 +00:00
|
|
|
const gchar *
|
2006-01-24 03:35:53 +00:00
|
|
|
gtk_link_button_get_uri (GtkLinkButton *link_button)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_LINK_BUTTON (link_button), NULL);
|
|
|
|
|
|
|
|
return link_button->priv->uri;
|
|
|
|
}
|
|
|
|
|
2008-08-07 11:37:57 +00:00
|
|
|
/**
|
|
|
|
* gtk_link_button_set_visited:
|
|
|
|
* @link_button: a #GtkLinkButton
|
2014-02-07 19:03:49 +00:00
|
|
|
* @visited: the new “visited” state
|
2008-08-07 11:37:57 +00:00
|
|
|
*
|
2014-02-07 19:03:49 +00:00
|
|
|
* Sets the “visited” state of the URI where the #GtkLinkButton
|
2008-08-07 11:37:57 +00:00
|
|
|
* points. See gtk_link_button_get_visited() for more details.
|
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_link_button_set_visited (GtkLinkButton *link_button,
|
|
|
|
gboolean visited)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_LINK_BUTTON (link_button));
|
|
|
|
|
|
|
|
visited = visited != FALSE;
|
|
|
|
|
|
|
|
if (link_button->priv->visited != visited)
|
|
|
|
{
|
|
|
|
link_button->priv->visited = visited;
|
|
|
|
|
2013-10-08 10:50:08 +00:00
|
|
|
if (visited)
|
|
|
|
{
|
|
|
|
gtk_widget_unset_state_flags (GTK_WIDGET (link_button), GTK_STATE_FLAG_LINK);
|
|
|
|
gtk_widget_set_state_flags (GTK_WIDGET (link_button), GTK_STATE_FLAG_VISITED, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_widget_unset_state_flags (GTK_WIDGET (link_button), GTK_STATE_FLAG_VISITED);
|
|
|
|
gtk_widget_set_state_flags (GTK_WIDGET (link_button), GTK_STATE_FLAG_LINK, FALSE);
|
|
|
|
}
|
2008-08-07 11:37:57 +00:00
|
|
|
|
|
|
|
g_object_notify (G_OBJECT (link_button), "visited");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_link_button_get_visited:
|
|
|
|
* @link_button: a #GtkLinkButton
|
|
|
|
*
|
2014-02-07 19:03:49 +00:00
|
|
|
* Retrieves the “visited” state of the URI where the #GtkLinkButton
|
2008-08-07 11:37:57 +00:00
|
|
|
* points. The button becomes visited when it is clicked. If the URI
|
2014-02-07 19:03:49 +00:00
|
|
|
* is changed on the button, the “visited” state is unset again.
|
2008-08-07 11:37:57 +00:00
|
|
|
*
|
|
|
|
* The state may also be changed using gtk_link_button_set_visited().
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
* Returns: %TRUE if the link has been visited, %FALSE otherwise
|
2008-08-07 11:37:57 +00:00
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gtk_link_button_get_visited (GtkLinkButton *link_button)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_LINK_BUTTON (link_button), FALSE);
|
|
|
|
|
|
|
|
return link_button->priv->visited;
|
|
|
|
}
|