Blame gtk/gtklinkbutton.c

Packit 98cdb6
/* GTK - The GIMP Toolkit
Packit 98cdb6
 * gtklinkbutton.c - an hyperlink-enabled button
Packit 98cdb6
 * 
Packit 98cdb6
 * Copyright (C) 2006 Emmanuele Bassi <ebassi@gmail.com>
Packit 98cdb6
 * All rights reserved.
Packit 98cdb6
 *
Packit 98cdb6
 * Based on gnome-href code by:
Packit 98cdb6
 * 	James Henstridge <james@daa.com.au>
Packit 98cdb6
 * 
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Library General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Library General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Library General Public
Packit 98cdb6
 * License along with this library; if not, write to the Free Software
Packit 98cdb6
 * Foundation, Inc., 59 Temple Place - Suite 330, Cambridge, MA 02139, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
Packit 98cdb6
#include <string.h>
Packit 98cdb6
Packit 98cdb6
#include "gtkclipboard.h"
Packit 98cdb6
#include "gtkdnd.h"
Packit 98cdb6
#include "gtkimagemenuitem.h"
Packit 98cdb6
#include "gtklabel.h"
Packit 98cdb6
#include "gtkmain.h"
Packit 98cdb6
#include "gtkmenu.h"
Packit 98cdb6
#include "gtkmenuitem.h"
Packit 98cdb6
#include "gtkstock.h"
Packit 98cdb6
#include "gtkshow.h"
Packit 98cdb6
#include "gtktooltip.h"
Packit 98cdb6
#include "gtklinkbutton.h"
Packit 98cdb6
#include "gtkprivate.h"
Packit 98cdb6
Packit 98cdb6
#include "gtkintl.h"
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
struct _GtkLinkButtonPrivate
Packit 98cdb6
{
Packit 98cdb6
  gchar *uri;
Packit 98cdb6
Packit 98cdb6
  gboolean visited;
Packit 98cdb6
Packit 98cdb6
  GtkWidget *popup_menu;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
enum
Packit 98cdb6
{
Packit 98cdb6
  PROP_0,
Packit 98cdb6
  PROP_URI,
Packit 98cdb6
  PROP_VISITED
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
#define GTK_LINK_BUTTON_GET_PRIVATE(obj)	(G_TYPE_INSTANCE_GET_PRIVATE ((obj), GTK_TYPE_LINK_BUTTON, GtkLinkButtonPrivate))
Packit 98cdb6
Packit 98cdb6
static void     gtk_link_button_finalize     (GObject          *object);
Packit 98cdb6
static void     gtk_link_button_get_property (GObject          *object,
Packit 98cdb6
					      guint             prop_id,
Packit 98cdb6
					      GValue           *value,
Packit 98cdb6
					      GParamSpec       *pspec);
Packit 98cdb6
static void     gtk_link_button_set_property (GObject          *object,
Packit 98cdb6
					      guint             prop_id,
Packit 98cdb6
					      const GValue     *value,
Packit 98cdb6
					      GParamSpec       *pspec);
Packit 98cdb6
static void     gtk_link_button_add          (GtkContainer     *container,
Packit 98cdb6
					      GtkWidget        *widget);
Packit 98cdb6
static gboolean gtk_link_button_button_press (GtkWidget        *widget,
Packit 98cdb6
					      GdkEventButton   *event);
Packit 98cdb6
static void     gtk_link_button_clicked      (GtkButton        *button);
Packit 98cdb6
static gboolean gtk_link_button_popup_menu   (GtkWidget        *widget);
Packit 98cdb6
static void     gtk_link_button_style_set    (GtkWidget        *widget,
Packit 98cdb6
					      GtkStyle         *old_style);
Packit 98cdb6
static gboolean gtk_link_button_enter_cb     (GtkWidget        *widget,
Packit 98cdb6
					      GdkEventCrossing *event,
Packit 98cdb6
					      gpointer          user_data);
Packit 98cdb6
static gboolean gtk_link_button_leave_cb     (GtkWidget        *widget,
Packit 98cdb6
					      GdkEventCrossing *event,
Packit 98cdb6
					      gpointer          user_data);
Packit 98cdb6
static void gtk_link_button_drag_data_get_cb (GtkWidget        *widget,
Packit 98cdb6
					      GdkDragContext   *context,
Packit 98cdb6
					      GtkSelectionData *selection,
Packit 98cdb6
					      guint             _info,
Packit 98cdb6
					      guint             _time,
Packit 98cdb6
					      gpointer          user_data);
Packit 98cdb6
static gboolean gtk_link_button_query_tooltip_cb (GtkWidget    *widget,
Packit 98cdb6
                                                  gint          x,
Packit 98cdb6
                                                  gint          y,
Packit 98cdb6
                                                  gboolean      keyboard_tip,
Packit 98cdb6
                                                  GtkTooltip   *tooltip,
Packit 98cdb6
                                                  gpointer      data);
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
static const GtkTargetEntry link_drop_types[] = {
Packit 98cdb6
  { "text/uri-list", 0, 0 },
Packit 98cdb6
  { "_NETSCAPE_URL", 0, 0 }
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static const GdkColor default_link_color = { 0, 0, 0, 0xeeee };
Packit 98cdb6
static const GdkColor default_visited_link_color = { 0, 0x5555, 0x1a1a, 0x8b8b };
Packit 98cdb6
Packit 98cdb6
static GtkLinkButtonUriFunc uri_func = NULL;
Packit 98cdb6
static gpointer uri_func_data = NULL;
Packit 98cdb6
static GDestroyNotify uri_func_destroy = NULL;
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GtkLinkButton, gtk_link_button, GTK_TYPE_BUTTON)
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_class_init (GtkLinkButtonClass *klass)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit 98cdb6
  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
Packit 98cdb6
  GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
Packit 98cdb6
  GtkButtonClass *button_class = GTK_BUTTON_CLASS (klass);
Packit 98cdb6
  
Packit 98cdb6
  gobject_class->set_property = gtk_link_button_set_property;
Packit 98cdb6
  gobject_class->get_property = gtk_link_button_get_property;
Packit 98cdb6
  gobject_class->finalize = gtk_link_button_finalize;
Packit 98cdb6
  
Packit 98cdb6
  widget_class->button_press_event = gtk_link_button_button_press;
Packit 98cdb6
  widget_class->popup_menu = gtk_link_button_popup_menu;
Packit 98cdb6
  widget_class->style_set = gtk_link_button_style_set;
Packit 98cdb6
  
Packit 98cdb6
  container_class->add = gtk_link_button_add;
Packit 98cdb6
Packit 98cdb6
  button_class->clicked = gtk_link_button_clicked;
Packit 98cdb6
Packit 98cdb6
  /**
Packit 98cdb6
   * GtkLinkButton:uri
Packit 98cdb6
   * 
Packit 98cdb6
   * The URI bound to this button. 
Packit 98cdb6
   *
Packit 98cdb6
   * Since: 2.10
Packit 98cdb6
   */
Packit 98cdb6
  g_object_class_install_property (gobject_class,
Packit 98cdb6
  				   PROP_URI,
Packit 98cdb6
  				   g_param_spec_string ("uri",
Packit 98cdb6
  				   			P_("URI"),
Packit 98cdb6
  				   			P_("The URI bound to this button"),
Packit 98cdb6
  				   			NULL,
Packit 98cdb6
  				   			G_PARAM_READWRITE));
Packit 98cdb6
  /**
Packit 98cdb6
   * GtkLinkButton:visited
Packit 98cdb6
   * 
Packit 98cdb6
   * The 'visited' state of this button. A visited link is drawn in a
Packit 98cdb6
   * different color.
Packit 98cdb6
   *
Packit 98cdb6
   * Since: 2.14
Packit 98cdb6
   */
Packit 98cdb6
  g_object_class_install_property (gobject_class,
Packit 98cdb6
  				   PROP_VISITED,
Packit 98cdb6
  				   g_param_spec_boolean ("visited",
Packit 98cdb6
                                                         P_("Visited"),
Packit 98cdb6
                                                         P_("Whether this link has been visited."),
Packit 98cdb6
                                                         FALSE,
Packit 98cdb6
                                                         G_PARAM_READWRITE));
Packit 98cdb6
  
Packit 98cdb6
  g_type_class_add_private (gobject_class, sizeof (GtkLinkButtonPrivate));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_init (GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  link_button->priv = GTK_LINK_BUTTON_GET_PRIVATE (link_button),
Packit 98cdb6
  
Packit 98cdb6
  gtk_button_set_relief (GTK_BUTTON (link_button), GTK_RELIEF_NONE);
Packit 98cdb6
  
Packit 98cdb6
  g_signal_connect (link_button, "enter-notify-event",
Packit 98cdb6
  		    G_CALLBACK (gtk_link_button_enter_cb), NULL);
Packit 98cdb6
  g_signal_connect (link_button, "leave-notify-event",
Packit 98cdb6
  		    G_CALLBACK (gtk_link_button_leave_cb), NULL);
Packit 98cdb6
  g_signal_connect (link_button, "drag-data-get",
Packit 98cdb6
  		    G_CALLBACK (gtk_link_button_drag_data_get_cb), NULL);
Packit 98cdb6
Packit 98cdb6
  g_object_set (link_button, "has-tooltip", TRUE, NULL);
Packit 98cdb6
  g_signal_connect (link_button, "query-tooltip",
Packit 98cdb6
                    G_CALLBACK (gtk_link_button_query_tooltip_cb), NULL);
Packit 98cdb6
  
Packit 98cdb6
  /* enable drag source */
Packit 98cdb6
  gtk_drag_source_set (GTK_WIDGET (link_button),
Packit 98cdb6
  		       GDK_BUTTON1_MASK,
Packit 98cdb6
  		       link_drop_types, G_N_ELEMENTS (link_drop_types),
Packit 98cdb6
  		       GDK_ACTION_COPY);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
Packit 98cdb6
  
Packit 98cdb6
  g_free (link_button->priv->uri);
Packit 98cdb6
  
Packit 98cdb6
  G_OBJECT_CLASS (gtk_link_button_parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_get_property (GObject    *object,
Packit 98cdb6
			      guint       prop_id,
Packit 98cdb6
			      GValue     *value,
Packit 98cdb6
			      GParamSpec *pspec)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
Packit 98cdb6
  
Packit 98cdb6
  switch (prop_id)
Packit 98cdb6
    {
Packit 98cdb6
    case PROP_URI:
Packit 98cdb6
      g_value_set_string (value, link_button->priv->uri);
Packit 98cdb6
      break;
Packit 98cdb6
    case PROP_VISITED:
Packit 98cdb6
      g_value_set_boolean (value, link_button->priv->visited);
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_set_property (GObject      *object,
Packit 98cdb6
			      guint         prop_id,
Packit 98cdb6
			      const GValue *value,
Packit 98cdb6
			      GParamSpec   *pspec)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (object);
Packit 98cdb6
  
Packit 98cdb6
  switch (prop_id)
Packit 98cdb6
    {
Packit 98cdb6
    case PROP_URI:
Packit 98cdb6
      gtk_link_button_set_uri (link_button, g_value_get_string (value));
Packit 98cdb6
      break;
Packit 98cdb6
    case PROP_VISITED:
Packit 98cdb6
      gtk_link_button_set_visited (link_button, g_value_get_boolean (value));
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_link_color (GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  GdkColor *link_color = NULL;
Packit 98cdb6
  GtkWidget *label;
Packit 98cdb6
Packit 98cdb6
  label = gtk_bin_get_child (GTK_BIN (link_button));
Packit 98cdb6
  if (!GTK_IS_LABEL (label))
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  if (link_button->priv->visited)
Packit 98cdb6
    {
Packit 98cdb6
      gtk_widget_style_get (GTK_WIDGET (link_button),
Packit 98cdb6
			    "visited-link-color", &link_color, NULL);
Packit 98cdb6
      if (!link_color)
Packit 98cdb6
	link_color = (GdkColor *) &default_visited_link_color;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gtk_widget_style_get (GTK_WIDGET (link_button),
Packit 98cdb6
			    "link-color", &link_color, NULL);
Packit 98cdb6
      if (!link_color)
Packit 98cdb6
	link_color = (GdkColor *) &default_link_color;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gtk_widget_modify_fg (label, GTK_STATE_NORMAL, link_color);
Packit 98cdb6
  gtk_widget_modify_fg (label, GTK_STATE_ACTIVE, link_color);
Packit 98cdb6
  gtk_widget_modify_fg (label, GTK_STATE_PRELIGHT, link_color);
Packit 98cdb6
  gtk_widget_modify_fg (label, GTK_STATE_SELECTED, link_color);
Packit 98cdb6
Packit 98cdb6
  if (link_color != &default_link_color &&
Packit 98cdb6
      link_color != &default_visited_link_color)
Packit 98cdb6
    gdk_color_free (link_color);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_link_underline (GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  GtkWidget *label;
Packit 98cdb6
  
Packit 98cdb6
  label = gtk_bin_get_child (GTK_BIN (link_button));
Packit 98cdb6
  if (GTK_IS_LABEL (label))
Packit 98cdb6
    {
Packit 98cdb6
      PangoAttrList *attributes;
Packit 98cdb6
      PangoAttribute *uline;
Packit 98cdb6
Packit 98cdb6
      uline = pango_attr_underline_new (PANGO_UNDERLINE_SINGLE);
Packit 98cdb6
      uline->start_index = 0;
Packit 98cdb6
      uline->end_index = G_MAXUINT;
Packit 98cdb6
      attributes = pango_attr_list_new ();
Packit 98cdb6
      pango_attr_list_insert (attributes, uline); 
Packit 98cdb6
      gtk_label_set_attributes (GTK_LABEL (label), attributes);
Packit 98cdb6
      pango_attr_list_unref (attributes);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_add (GtkContainer *container,
Packit 98cdb6
		     GtkWidget    *widget)
Packit 98cdb6
{
Packit 98cdb6
  GTK_CONTAINER_CLASS (gtk_link_button_parent_class)->add (container, widget);
Packit 98cdb6
Packit 98cdb6
  set_link_color (GTK_LINK_BUTTON (container));
Packit 98cdb6
  set_link_underline (GTK_LINK_BUTTON (container));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_style_set (GtkWidget *widget,
Packit 98cdb6
			   GtkStyle  *old_style)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
Packit 98cdb6
Packit 98cdb6
  set_link_color (link_button);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_hand_cursor (GtkWidget *widget,
Packit 98cdb6
		 gboolean   show_hand)
Packit 98cdb6
{
Packit 98cdb6
  GdkDisplay *display;
Packit 98cdb6
  GdkCursor *cursor;
Packit 98cdb6
Packit 98cdb6
  display = gtk_widget_get_display (widget);
Packit 98cdb6
Packit 98cdb6
  cursor = NULL;
Packit 98cdb6
  if (show_hand)
Packit 98cdb6
    cursor = gdk_cursor_new_for_display (display, GDK_HAND2);
Packit 98cdb6
Packit 98cdb6
  gdk_window_set_cursor (widget->window, cursor);
Packit 98cdb6
  gdk_display_flush (display);
Packit 98cdb6
Packit 98cdb6
  if (cursor)
Packit 98cdb6
    gdk_cursor_unref (cursor);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
popup_menu_detach (GtkWidget *attach_widget,
Packit 98cdb6
		   GtkMenu   *menu)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (attach_widget);
Packit 98cdb6
Packit 98cdb6
  link_button->priv->popup_menu = NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
popup_position_func (GtkMenu  *menu,
Packit 98cdb6
		     gint     *x,
Packit 98cdb6
		     gint     *y,
Packit 98cdb6
		     gboolean *push_in,
Packit 98cdb6
		     gpointer  user_data)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (user_data);
Packit 98cdb6
  GtkLinkButtonPrivate *priv = link_button->priv;
Packit 98cdb6
  GtkWidget *widget = GTK_WIDGET (link_button);
Packit 98cdb6
  GdkScreen *screen = gtk_widget_get_screen (widget);
Packit 98cdb6
  GtkRequisition req;
Packit 98cdb6
  gint monitor_num;
Packit 98cdb6
  GdkRectangle monitor;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (gtk_widget_get_realized (widget));
Packit 98cdb6
Packit 98cdb6
  gdk_window_get_origin (widget->window, x, y);
Packit 98cdb6
Packit 98cdb6
  gtk_widget_size_request (priv->popup_menu, &req;;
Packit 98cdb6
Packit 98cdb6
  *x += widget->allocation.width / 2;
Packit 98cdb6
  *y += widget->allocation.height;
Packit 98cdb6
Packit 98cdb6
  monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
Packit 98cdb6
  gtk_menu_set_monitor (menu, monitor_num);
Packit 98cdb6
  gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
Packit 98cdb6
Packit 98cdb6
  *x = CLAMP (*x, monitor.x, monitor.x + MAX (0, monitor.width - req.width));
Packit 98cdb6
  *y = CLAMP (*y, monitor.y, monitor.y + MAX (0, monitor.height - req.height));
Packit 98cdb6
Packit 98cdb6
  *push_in = FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
copy_activate_cb (GtkWidget     *widget,
Packit 98cdb6
		  GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButtonPrivate *priv = link_button->priv;
Packit 98cdb6
  
Packit 98cdb6
  gtk_clipboard_set_text (gtk_widget_get_clipboard (GTK_WIDGET (link_button),
Packit 98cdb6
			  			    GDK_SELECTION_CLIPBOARD),
Packit 98cdb6
		  	  priv->uri, -1);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_do_popup (GtkLinkButton  *link_button,
Packit 98cdb6
			  GdkEventButton *event)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButtonPrivate *priv = link_button->priv;
Packit 98cdb6
  gint button;
Packit 98cdb6
  guint time;
Packit 98cdb6
  
Packit 98cdb6
  if (event)
Packit 98cdb6
    {
Packit 98cdb6
      button = event->button;
Packit 98cdb6
      time = event->time;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      button = 0;
Packit 98cdb6
      time = gtk_get_current_event_time ();
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (gtk_widget_get_realized (GTK_WIDGET (link_button)))
Packit 98cdb6
    {
Packit 98cdb6
      GtkWidget *menu_item;
Packit 98cdb6
      
Packit 98cdb6
      if (priv->popup_menu)
Packit 98cdb6
	gtk_widget_destroy (priv->popup_menu);
Packit 98cdb6
Packit 98cdb6
      priv->popup_menu = gtk_menu_new ();
Packit 98cdb6
      
Packit 98cdb6
      gtk_menu_attach_to_widget (GTK_MENU (priv->popup_menu),
Packit 98cdb6
		      		 GTK_WIDGET (link_button),
Packit 98cdb6
				 popup_menu_detach);
Packit 98cdb6
Packit 98cdb6
      menu_item = gtk_image_menu_item_new_with_mnemonic (_("Copy URL"));
Packit 98cdb6
      gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item),
Packit 98cdb6
		      		     gtk_image_new_from_stock (GTK_STOCK_COPY,
Packit 98cdb6
							       GTK_ICON_SIZE_MENU));
Packit 98cdb6
      g_signal_connect (menu_item, "activate",
Packit 98cdb6
		        G_CALLBACK (copy_activate_cb), link_button);
Packit 98cdb6
      gtk_widget_show (menu_item);
Packit 98cdb6
      gtk_menu_shell_append (GTK_MENU_SHELL (priv->popup_menu), menu_item);
Packit 98cdb6
      
Packit 98cdb6
      if (button)
Packit 98cdb6
        gtk_menu_popup (GTK_MENU (priv->popup_menu), NULL, NULL,
Packit 98cdb6
		        NULL, NULL,
Packit 98cdb6
			button, time);
Packit 98cdb6
      else
Packit 98cdb6
        {
Packit 98cdb6
          gtk_menu_popup (GTK_MENU (priv->popup_menu), NULL, NULL,
Packit 98cdb6
			  popup_position_func, link_button,
Packit 98cdb6
			  button, time);
Packit 98cdb6
	  gtk_menu_shell_select_first (GTK_MENU_SHELL (priv->popup_menu), FALSE);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_link_button_button_press (GtkWidget      *widget,
Packit 98cdb6
			      GdkEventButton *event)
Packit 98cdb6
{
Packit 98cdb6
  if (!gtk_widget_has_focus (widget))
Packit 98cdb6
    gtk_widget_grab_focus (widget);
Packit 98cdb6
Packit 98cdb6
  if (_gtk_button_event_triggers_context_menu (event))
Packit 98cdb6
    {
Packit 98cdb6
      gtk_link_button_do_popup (GTK_LINK_BUTTON (widget), event);
Packit 98cdb6
      
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (GTK_WIDGET_CLASS (gtk_link_button_parent_class)->button_press_event)
Packit 98cdb6
    return GTK_WIDGET_CLASS (gtk_link_button_parent_class)->button_press_event (widget, event);
Packit 98cdb6
  
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_clicked (GtkButton *button)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (button);
Packit 98cdb6
Packit 98cdb6
  if (uri_func)
Packit 98cdb6
    (* uri_func) (link_button, link_button->priv->uri, uri_func_data);
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      GdkScreen *screen;
Packit 98cdb6
      GError *error;
Packit 98cdb6
Packit 98cdb6
      if (gtk_widget_has_screen (GTK_WIDGET (button)))
Packit 98cdb6
        screen = gtk_widget_get_screen (GTK_WIDGET (button));
Packit 98cdb6
      else
Packit 98cdb6
        screen = NULL;
Packit 98cdb6
Packit 98cdb6
      error = NULL;
Packit 98cdb6
      gtk_show_uri (screen, link_button->priv->uri, GDK_CURRENT_TIME, &error);
Packit 98cdb6
      if (error)
Packit 98cdb6
        {
Packit 98cdb6
          g_warning ("Unable to show '%s': %s",
Packit 98cdb6
                     link_button->priv->uri,
Packit 98cdb6
                     error->message);
Packit 98cdb6
          g_error_free (error);
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gtk_link_button_set_visited (link_button, TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_link_button_popup_menu (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  gtk_link_button_do_popup (GTK_LINK_BUTTON (widget), NULL);
Packit 98cdb6
Packit 98cdb6
  return TRUE; 
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_link_button_enter_cb (GtkWidget        *widget,
Packit 98cdb6
			  GdkEventCrossing *crossing,
Packit 98cdb6
			  gpointer          user_data)
Packit 98cdb6
{
Packit 98cdb6
  set_hand_cursor (widget, TRUE);
Packit 98cdb6
  
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_link_button_leave_cb (GtkWidget        *widget,
Packit 98cdb6
			  GdkEventCrossing *crossing,
Packit 98cdb6
			  gpointer          user_data)
Packit 98cdb6
{
Packit 98cdb6
  set_hand_cursor (widget, FALSE);
Packit 98cdb6
  
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_link_button_drag_data_get_cb (GtkWidget        *widget,
Packit 98cdb6
				  GdkDragContext   *context,
Packit 98cdb6
				  GtkSelectionData *selection,
Packit 98cdb6
				  guint             _info,
Packit 98cdb6
				  guint             _time,
Packit 98cdb6
				  gpointer          user_data)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
Packit 98cdb6
  gchar *uri;
Packit 98cdb6
  
Packit 98cdb6
  uri = g_strdup_printf ("%s\r\n", link_button->priv->uri);
Packit 98cdb6
  gtk_selection_data_set (selection,
Packit 98cdb6
  			  selection->target,
Packit 98cdb6
  			  8,
Packit 98cdb6
  			  (guchar *) uri,
Packit 98cdb6
			  strlen (uri));
Packit 98cdb6
  
Packit 98cdb6
  g_free (uri);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_new:
Packit 98cdb6
 * @uri: a valid URI
Packit 98cdb6
 *
Packit 98cdb6
 * Creates a new #GtkLinkButton with the URI as its text.
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: a new link button widget.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 */
Packit 98cdb6
GtkWidget *
Packit 98cdb6
gtk_link_button_new (const gchar *uri)
Packit 98cdb6
{
Packit 98cdb6
  gchar *utf8_uri = NULL;
Packit 98cdb6
  GtkWidget *retval;
Packit 98cdb6
  
Packit 98cdb6
  g_return_val_if_fail (uri != NULL, NULL);
Packit 98cdb6
  
Packit 98cdb6
  if (g_utf8_validate (uri, -1, NULL))
Packit 98cdb6
    {
Packit 98cdb6
      utf8_uri = g_strdup (uri);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      GError *conv_err = NULL;
Packit 98cdb6
    
Packit 98cdb6
      utf8_uri = g_locale_to_utf8 (uri, -1, NULL, NULL, &conv_err);
Packit 98cdb6
      if (conv_err)
Packit 98cdb6
        {
Packit 98cdb6
          g_warning ("Attempting to convert URI `%s' to UTF-8, but failed "
Packit 98cdb6
                     "with error: %s\n",
Packit 98cdb6
                     uri,
Packit 98cdb6
                     conv_err->message);
Packit 98cdb6
          g_error_free (conv_err);
Packit 98cdb6
        
Packit 98cdb6
          utf8_uri = g_strdup (_("Invalid URI"));
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
  
Packit 98cdb6
  retval = g_object_new (GTK_TYPE_LINK_BUTTON,
Packit 98cdb6
  			 "label", utf8_uri,
Packit 98cdb6
  			 "uri", uri,
Packit 98cdb6
  			 NULL);
Packit 98cdb6
  
Packit 98cdb6
  g_free (utf8_uri);
Packit 98cdb6
  
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_new_with_label:
Packit 98cdb6
 * @uri: a valid URI
Packit 98cdb6
 * @label: (allow-none): the text of the button
Packit 98cdb6
 *
Packit 98cdb6
 * Creates a new #GtkLinkButton containing a label.
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: (transfer none): a new link button widget.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 */
Packit 98cdb6
GtkWidget *
Packit 98cdb6
gtk_link_button_new_with_label (const gchar *uri,
Packit 98cdb6
				const gchar *label)
Packit 98cdb6
{
Packit 98cdb6
  GtkWidget *retval;
Packit 98cdb6
  
Packit 98cdb6
  g_return_val_if_fail (uri != NULL, NULL);
Packit 98cdb6
  
Packit 98cdb6
  if (!label)
Packit 98cdb6
    return gtk_link_button_new (uri);
Packit 98cdb6
Packit 98cdb6
  retval = g_object_new (GTK_TYPE_LINK_BUTTON,
Packit 98cdb6
		         "label", label,
Packit 98cdb6
			 "uri", uri,
Packit 98cdb6
			 NULL);
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean 
Packit 98cdb6
gtk_link_button_query_tooltip_cb (GtkWidget    *widget,
Packit 98cdb6
                                  gint          x,
Packit 98cdb6
                                  gint          y,
Packit 98cdb6
                                  gboolean      keyboard_tip,
Packit 98cdb6
                                  GtkTooltip   *tooltip,
Packit 98cdb6
                                  gpointer      data)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButton *link_button = GTK_LINK_BUTTON (widget);
Packit 98cdb6
  const gchar *label, *uri;
Packit 98cdb6
Packit 98cdb6
  label = gtk_button_get_label (GTK_BUTTON (link_button));
Packit 98cdb6
  uri = link_button->priv->uri;
Packit 98cdb6
Packit 98cdb6
  if (!gtk_widget_get_tooltip_text (widget)
Packit 98cdb6
    && !gtk_widget_get_tooltip_markup (widget)
Packit 98cdb6
    && label && *label != '\0' && uri && strcmp (label, uri) != 0)
Packit 98cdb6
    {
Packit 98cdb6
      gtk_tooltip_set_text (tooltip, uri);
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_set_uri:
Packit 98cdb6
 * @link_button: a #GtkLinkButton
Packit 98cdb6
 * @uri: a valid URI
Packit 98cdb6
 *
Packit 98cdb6
 * Sets @uri as the URI where the #GtkLinkButton points. As a side-effect
Packit 98cdb6
 * this unsets the 'visited' state of the button.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_link_button_set_uri (GtkLinkButton *link_button,
Packit 98cdb6
			 const gchar   *uri)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButtonPrivate *priv;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (GTK_IS_LINK_BUTTON (link_button));
Packit 98cdb6
  g_return_if_fail (uri != NULL);
Packit 98cdb6
Packit 98cdb6
  priv = link_button->priv;
Packit 98cdb6
Packit 98cdb6
  g_free (priv->uri);
Packit 98cdb6
  priv->uri = g_strdup (uri);
Packit 98cdb6
Packit 98cdb6
  g_object_notify (G_OBJECT (link_button), "uri");
Packit 98cdb6
Packit 98cdb6
  gtk_link_button_set_visited (link_button, FALSE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_get_uri:
Packit 98cdb6
 * @link_button: a #GtkLinkButton
Packit 98cdb6
 *
Packit 98cdb6
 * Retrieves the URI set using gtk_link_button_set_uri().
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: a valid URI.  The returned string is owned by the link button
Packit 98cdb6
 *   and should not be modified or freed.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 */
Packit 98cdb6
const gchar *
Packit 98cdb6
gtk_link_button_get_uri (GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_LINK_BUTTON (link_button), NULL);
Packit 98cdb6
  
Packit 98cdb6
  return link_button->priv->uri;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_set_uri_hook:
Packit 98cdb6
 * @func: (allow-none): a function called each time a #GtkLinkButton is clicked, or %NULL
Packit 98cdb6
 * @data: (allow-none): user data to be passed to @func, or %NULL
Packit 98cdb6
 * @destroy: (allow-none): a #GDestroyNotify that gets called when @data is no longer needed, or %NULL
Packit 98cdb6
 *
Packit 98cdb6
 * Sets @func as the function that should be invoked every time a user clicks
Packit 98cdb6
 * a #GtkLinkButton. This function is called before every callback registered
Packit 98cdb6
 * for the "clicked" signal.
Packit 98cdb6
 *
Packit 98cdb6
 * If no uri hook has been set, GTK+ defaults to calling gtk_show_uri().
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: the previously set hook function.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.10
Packit 98cdb6
 *
Packit 98cdb6
 * Deprecated: 2.24: Use the #GtkButton::clicked signal instead
Packit 98cdb6
 */
Packit 98cdb6
GtkLinkButtonUriFunc
Packit 98cdb6
gtk_link_button_set_uri_hook (GtkLinkButtonUriFunc func,
Packit 98cdb6
			      gpointer             data,
Packit 98cdb6
			      GDestroyNotify       destroy)
Packit 98cdb6
{
Packit 98cdb6
  GtkLinkButtonUriFunc old_uri_func;
Packit 98cdb6
Packit 98cdb6
  if (uri_func_destroy)
Packit 98cdb6
    (* uri_func_destroy) (uri_func_data);
Packit 98cdb6
Packit 98cdb6
  old_uri_func = uri_func;
Packit 98cdb6
Packit 98cdb6
  uri_func = func;
Packit 98cdb6
  uri_func_data = data;
Packit 98cdb6
  uri_func_destroy = destroy;
Packit 98cdb6
Packit 98cdb6
  return old_uri_func;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_set_visited:
Packit 98cdb6
 * @link_button: a #GtkLinkButton
Packit 98cdb6
 * @visited: the new 'visited' state
Packit 98cdb6
 *
Packit 98cdb6
 * Sets the 'visited' state of the URI where the #GtkLinkButton
Packit 98cdb6
 * points.  See gtk_link_button_get_visited() for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.14
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_link_button_set_visited (GtkLinkButton *link_button,
Packit 98cdb6
                             gboolean       visited)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (GTK_IS_LINK_BUTTON (link_button));
Packit 98cdb6
Packit 98cdb6
  visited = visited != FALSE;
Packit 98cdb6
Packit 98cdb6
  if (link_button->priv->visited != visited)
Packit 98cdb6
    {
Packit 98cdb6
      link_button->priv->visited = visited;
Packit 98cdb6
Packit 98cdb6
      set_link_color (link_button);
Packit 98cdb6
Packit 98cdb6
      g_object_notify (G_OBJECT (link_button), "visited");
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_link_button_get_visited:
Packit 98cdb6
 * @link_button: a #GtkLinkButton
Packit 98cdb6
 *
Packit 98cdb6
 * Retrieves the 'visited' state of the URI where the #GtkLinkButton
Packit 98cdb6
 * points. The button becomes visited when it is clicked. If the URI
Packit 98cdb6
 * is changed on the button, the 'visited' state is unset again.
Packit 98cdb6
 *
Packit 98cdb6
 * The state may also be changed using gtk_link_button_set_visited().
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: %TRUE if the link has been visited, %FALSE otherwise
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.14
Packit 98cdb6
 */
Packit 98cdb6
gboolean
Packit 98cdb6
gtk_link_button_get_visited (GtkLinkButton *link_button)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_LINK_BUTTON (link_button), FALSE);
Packit 98cdb6
  
Packit 98cdb6
  return link_button->priv->visited;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
#define __GTK_LINK_BUTTON_C__
Packit 98cdb6
#include "gtkaliasdef.c"