/*
* Copyright (C) 2002 Christophe Fergeau
* Copyright (C) 2003, 2004 Marco Pesenti Gritti
* Copyright (C) 2003, 2004, 2005 Christian Persch
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Id$
*/
#include "config.h"
#include "ephy-notebook.h"
#include "ephy-stock-icons.h"
#include "eel-gconf-extensions.h"
#include "ephy-prefs.h"
#include "ephy-marshal.h"
#include "ephy-file-helpers.h"
#include "ephy-dnd.h"
#include "ephy-embed.h"
#include "ephy-window.h"
#include "ephy-shell.h"
#include "ephy-spinner.h"
#include "ephy-link.h"
#include "ephy-debug.h"
#include <glib-object.h>
#include <gtk/gtkeventbox.h>
#include <gtk/gtknotebook.h>
#include <gtk/gtkhbox.h>
#include <gtk/gtklabel.h>
#include <gtk/gtkwidget.h>
#include <gtk/gtktooltips.h>
#include <gtk/gtkmain.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkimage.h>
#include <gtk/gtkbutton.h>
#include <gtk/gtkiconfactory.h>
#include <glib/gi18n.h>
#include <libgnomevfs/gnome-vfs-uri.h>
#define TAB_WIDTH_N_CHARS 15
#define AFTER_ALL_TABS -1
#define NOT_IN_APP_WINDOWS -2
#define INSANE_NUMBER_OF_URLS 20
/* Until https://bugzilla.mozilla.org/show_bug.cgi?id=296002 is fixed */
#define KEEP_TAB_IN_SAME_TOPLEVEL
#define EPHY_NOTEBOOK_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_NOTEBOOK, EphyNotebookPrivate))
struct _EphyNotebookPrivate
{
GList *focused_pages;
GtkTooltips *title_tips;
guint tabs_vis_notifier_id;
gulong motion_notify_handler_id;
gulong grab_notify_handler_id;
gulong toplevel_grab_broken_handler_id;
gulong toplevel_motion_notify_handler_id;
gulong toplevel_button_release_handler_id;
int x_start;
int y_start;
guint drag_in_progress : 1;
guint show_tabs : 1;
guint dnd_enabled : 1;
};
static void ephy_notebook_init (EphyNotebook *notebook);
static void ephy_notebook_class_init (EphyNotebookClass *klass);
static void ephy_notebook_finalize (GObject *object);
static void move_tab_to_another_notebook (EphyNotebook *src,
EphyNotebook *dest,
GdkEventMotion *event,
int dest_position);
static void move_tab (EphyNotebook *notebook,
int dest_position);
/* Local variables */
static GdkCursor *cursor = NULL;
static const GtkTargetEntry url_drag_types [] =
{
{ EPHY_DND_URI_LIST_TYPE, 0, 0 },
{ EPHY_DND_URL_TYPE, 0, 1 }
};
enum
{
PROP_0,
PROP_DND_ENABLED,
PROP_SHOW_TABS
};
enum
{
TAB_ADDED,
TAB_REMOVED,
TABS_REORDERED,
TAB_DETACHED,
TAB_DELETE,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL] = { 0 };
static GObjectClass *parent_class = NULL;
GType
ephy_notebook_get_type (void)
{
static GType type = 0;
if (G_UNLIKELY (type == 0))
{
static const GTypeInfo our_info =
{
sizeof (EphyNotebookClass),
NULL, /* base_init */
NULL, /* base_finalize */
(GClassInitFunc) ephy_notebook_class_init,
NULL,
NULL, /* class_data */
sizeof (EphyNotebook),
0, /* n_preallocs */
(GInstanceInitFunc) ephy_notebook_init
};
static const GInterfaceInfo link_info =
{
NULL,
NULL,
NULL
};
type = g_type_register_static (GTK_TYPE_NOTEBOOK,
"EphyNotebook",
&our_info, 0);
g_type_add_interface_static (type,
EPHY_TYPE_LINK,
&link_info);
}
return type;
}
void
ephy_notebook_set_dnd_enabled (EphyNotebook *notebook,
gboolean enabled)
{
EphyNotebookPrivate *priv = notebook->priv;
priv->dnd_enabled = enabled;
/* FIXME abort any DNDs in progress */
g_object_notify (G_OBJECT (notebook), "dnd-enabled");
}
static void
ephy_notebook_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
EphyNotebook *notebook = EPHY_NOTEBOOK (object);
EphyNotebookPrivate *priv = notebook->priv;
switch (prop_id)
{
case PROP_DND_ENABLED:
g_value_set_boolean (value, priv->dnd_enabled);
break;
case PROP_SHOW_TABS:
g_value_set_boolean (value, priv->show_tabs);
break;
}
}
static void
ephy_notebook_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyNotebook *notebook = EPHY_NOTEBOOK (object);
switch (prop_id)
{
case PROP_DND_ENABLED:
ephy_notebook_set_dnd_enabled (notebook, g_value_get_boolean (value));
break;
case PROP_SHOW_TABS:
ephy_notebook_set_show_tabs (notebook, g_value_get_boolean (value));
break;
}
}
static void
ephy_notebook_class_init (EphyNotebookClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
parent_class = g_type_class_peek_parent (klass);
object_class->finalize = ephy_notebook_finalize;
object_class->get_property = ephy_notebook_get_property;
object_class->set_property = ephy_notebook_set_property;
signals[TAB_ADDED] =
g_signal_new ("tab_added",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyNotebookClass, tab_added),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE,
1,
EPHY_TYPE_TAB);
signals[TAB_REMOVED] =
g_signal_new ("tab_removed",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyNotebookClass, tab_removed),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE,
1,
EPHY_TYPE_TAB);
signals[TAB_DETACHED] =
g_signal_new ("tab_detached",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyNotebookClass, tab_detached),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE,
1,
EPHY_TYPE_TAB);
signals[TABS_REORDERED] =
g_signal_new ("tabs_reordered",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyNotebookClass, tabs_reordered),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE,
0);
signals[TAB_DELETE] =
g_signal_new ("tab_delete",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyNotebookClass, tab_delete),
g_signal_accumulator_true_handled, NULL,
ephy_marshal_BOOLEAN__OBJECT,
G_TYPE_BOOLEAN,
1,
EPHY_TYPE_TAB);
g_object_class_install_property (object_class,
PROP_DND_ENABLED,
g_param_spec_boolean ("dnd-enabled",
"DND enabled",
"Whether the notebook allows tab reordering by DND",
TRUE,
G_PARAM_READWRITE));
g_object_class_install_property (object_class,
PROP_SHOW_TABS,
g_param_spec_boolean ("show-tabs",
"Show tabs",
"Whether the notebook shows the tabs bar",
TRUE,
G_PARAM_READWRITE));
g_type_class_add_private (object_class, sizeof (EphyNotebookPrivate));
}
static EphyNotebook *
find_notebook_at_pointer (gint abs_x, gint abs_y)
{
GdkWindow *win_at_pointer, *toplevel_win;
gpointer toplevel = NULL;
gint x, y;
/* FIXME multi-head */
win_at_pointer = gdk_window_at_pointer (&x, &y);
if (win_at_pointer == NULL)
{
/* We are outside all windows containing a notebook */
return NULL;
}
toplevel_win = gdk_window_get_toplevel (win_at_pointer);
/* get the GtkWidget which owns the toplevel GdkWindow */
gdk_window_get_user_data (toplevel_win, &toplevel);
/* toplevel should be an EphyWindow */
if (toplevel != NULL && EPHY_IS_WINDOW (toplevel))
{
return EPHY_NOTEBOOK (ephy_window_get_notebook
(EPHY_WINDOW (toplevel)));
}
return NULL;
}
static gboolean
is_in_notebook_window (EphyNotebook *notebook,
gint abs_x, gint abs_y)
{
EphyNotebook *nb_at_pointer;
nb_at_pointer = find_notebook_at_pointer (abs_x, abs_y);
return nb_at_pointer == notebook;
}
static gint
find_tab_num_at_pos (EphyNotebook *notebook, gint abs_x, gint abs_y)
{
GtkPositionType tab_pos;
int page_num = 0;
GtkNotebook *nb = GTK_NOTEBOOK (notebook);
GtkWidget *page;
tab_pos = gtk_notebook_get_tab_pos (GTK_NOTEBOOK (notebook));
if (GTK_NOTEBOOK (notebook)->first_tab == NULL)
{
return AFTER_ALL_TABS;
}
/* For some reason unfullscreen + quick click can
cause a wrong click event to be reported to the tab */
if (!is_in_notebook_window(notebook, abs_x, abs_y))
{
return NOT_IN_APP_WINDOWS;
}
while ((page = gtk_notebook_get_nth_page (nb, page_num)))
{
GtkWidget *tab;
gint max_x, max_y;
gint x_root, y_root;
tab = gtk_notebook_get_tab_label (nb, page);
g_return_val_if_fail (tab != NULL, -1);
if (!GTK_WIDGET_MAPPED (GTK_WIDGET (tab)))
{
page_num++;
continue;
}
gdk_window_get_origin (GDK_WINDOW (tab->window),
&x_root, &y_root);
max_x = x_root + tab->allocation.x + tab->allocation.width;
max_y = y_root + tab->allocation.y + tab->allocation.height;
if (((tab_pos == GTK_POS_TOP)
|| (tab_pos == GTK_POS_BOTTOM))
&&(abs_x<=max_x))
{
return page_num;
}
else if (((tab_pos == GTK_POS_LEFT)
|| (tab_pos == GTK_POS_RIGHT))
&& (abs_y<=max_y))
{
return page_num;
}
page_num++;
}
return AFTER_ALL_TABS;
}
static gint find_notebook_and_tab_at_pos (gint abs_x, gint abs_y,
EphyNotebook **notebook,
gint *page_num)
{
*notebook = find_notebook_at_pointer (abs_x, abs_y);
if (*notebook == NULL)
{
return NOT_IN_APP_WINDOWS;
}
*page_num = find_tab_num_at_pos (*notebook, abs_x, abs_y);
if (*page_num < 0)
{
return *page_num;
}
else
{
return 0;
}
}
void
ephy_notebook_move_tab (EphyNotebook *src,
EphyNotebook *dest,
EphyTab *tab,
int dest_position)
{
if (dest == NULL || src == dest)
{
gtk_notebook_reorder_child
(GTK_NOTEBOOK (src), GTK_WIDGET (tab), dest_position);
if (src->priv->drag_in_progress == FALSE)
{
g_signal_emit (G_OBJECT (src), signals[TABS_REORDERED], 0);
}
}
#ifndef KEEP_TAB_IN_SAME_TOPLEVEL
else
{
/* make sure the tab isn't destroyed while we move it */
g_object_ref (tab);
ephy_notebook_remove_tab (src, tab);
ephy_notebook_add_tab (dest, tab, dest_position, TRUE);
g_object_unref (tab);
}
#endif
}
static void
drag_stop (EphyNotebook *notebook,
guint32 time)
{
EphyNotebookPrivate *priv = notebook->priv;
GtkWidget *widget = GTK_WIDGET (notebook);
GtkWidget *toplevel, *child;
if (priv->drag_in_progress)
{
LOG ("Drag Stop");
toplevel = gtk_widget_get_toplevel (widget);
g_return_if_fail (GTK_WIDGET_TOPLEVEL (toplevel));
child = gtk_bin_get_child (GTK_BIN (toplevel));
g_return_if_fail (child != NULL);
/* disconnect the signals before ungrabbing! */
if (priv->toplevel_grab_broken_handler_id != 0)
{
g_signal_handler_disconnect (toplevel,
priv->toplevel_grab_broken_handler_id);
priv->toplevel_grab_broken_handler_id = 0;
}
if (priv->grab_notify_handler_id != 0)
{
g_signal_handler_disconnect (notebook,
priv->grab_notify_handler_id);
priv->grab_notify_handler_id = 0;
}
if (priv->toplevel_motion_notify_handler_id != 0)
{
g_signal_handler_disconnect (toplevel,
priv->toplevel_motion_notify_handler_id);
priv->toplevel_motion_notify_handler_id = 0;
}
if (priv->toplevel_button_release_handler_id != 0)
{
g_signal_handler_disconnect (toplevel,
priv->toplevel_button_release_handler_id);
priv->toplevel_button_release_handler_id = 0;
}
/* ungrab the pointer if it's grabbed */
/* FIXME multihead */
if (gdk_pointer_is_grabbed ())
{
gdk_pointer_ungrab (time);
}
gtk_grab_remove (toplevel);
g_signal_emit (G_OBJECT (notebook), signals[TABS_REORDERED], 0);
}
if (priv->motion_notify_handler_id != 0)
{
g_signal_handler_disconnect (notebook,
priv->motion_notify_handler_id);
priv->motion_notify_handler_id = 0;
}
priv->drag_in_progress = FALSE;
}
static gboolean
grab_broken_event_cb (GtkWidget *widget,
GdkEventGrabBroken *event,
EphyNotebook *notebook)
{
LOG ("Grab Broken [%p]", notebook);
drag_stop (notebook, GDK_CURRENT_TIME /* FIXME? */);
return FALSE;
}
static void
grab_notify_cb (GtkWidget *widget,
gboolean was_grabbed,
EphyNotebook *notebook)
{
LOG ("Grab Notify [%p, was-grabbed:%s]", notebook, was_grabbed ? "t" : "f");
drag_stop (notebook, GDK_CURRENT_TIME /* FIXME? */);
}
static gboolean
toplevel_motion_notify_cb (GtkWidget *toplevel,
GdkEventMotion *event,
EphyNotebook *notebook)
{
EphyNotebook *dest = NULL;
int page_num, result;
result = find_notebook_and_tab_at_pos ((gint)event->x_root,
(gint)event->y_root,
&dest, &page_num);
if (result != NOT_IN_APP_WINDOWS)
{
if (dest != notebook)
{
move_tab_to_another_notebook (notebook, dest,
event, page_num);
}
else
{
g_assert (page_num >= -1);
move_tab (notebook, page_num);
}
}
return FALSE;
}
static gboolean
toplevel_button_release_cb (GtkWidget *toplevel,
GdkEventButton *event,
EphyNotebook *notebook)
{
#ifndef KEEP_TAB_IN_SAME_TOPLEVEL
EphyNotebookPrivate *priv = notebook->priv;
if (priv->drag_in_progress)
{
gint cur_page_num;
GtkWidget *cur_page;
cur_page_num =
gtk_notebook_get_current_page (GTK_NOTEBOOK (notebook));
cur_page = gtk_notebook_get_nth_page (GTK_NOTEBOOK (notebook),
cur_page_num);
if (!is_in_notebook_window (notebook, event->x_root, event->y_root)
&& gtk_notebook_get_n_pages (GTK_NOTEBOOK (notebook)) > 1)
{
/* Tab was detached */
g_signal_emit (G_OBJECT (notebook),
signals[TAB_DETACHED], 0, cur_page);
}
}
#endif
/* This must be called even if a drag isn't happening */
drag_stop (notebook, event->time);
return FALSE;
}
static gboolean
drag_start (EphyNotebook *notebook,
guint32 time)
{
EphyNotebookPrivate *priv = notebook->priv;
GtkWidget *widget = GTK_WIDGET (notebook);
GtkWidget *toplevel, *child;
/* FIXME multihead */
if (priv->drag_in_progress || gdk_pointer_is_grabbed ()) return FALSE;
LOG ("Drag Start");
priv->drag_in_progress = TRUE;
/* get a new cursor, if necessary */
/* FIXME multi-head */
if (!cursor) cursor = gdk_cursor_new (GDK_FLEUR);
toplevel = gtk_widget_get_toplevel (widget);
g_return_val_if_fail (GTK_WIDGET_TOPLEVEL (toplevel), FALSE);
child = gtk_bin_get_child (GTK_BIN (toplevel));
g_return_val_if_fail (child != NULL, FALSE);
/* grab the pointer */
gtk_grab_add (toplevel);
/* FIXME multi-head */
if (gdk_pointer_grab (toplevel->window,
FALSE,
GDK_BUTTON1_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
NULL, cursor, time) != GDK_GRAB_SUCCESS)
{
drag_stop (notebook, time);
return FALSE;
}
g_return_val_if_fail (priv->toplevel_grab_broken_handler_id == 0, FALSE);
g_return_val_if_fail (priv->toplevel_motion_notify_handler_id == 0, FALSE);
g_return_val_if_fail (priv->toplevel_button_release_handler_id == 0, FALSE);
g_return_val_if_fail (priv->grab_notify_handler_id == 0, FALSE);
priv->toplevel_grab_broken_handler_id =
g_signal_connect (toplevel, "grab-broken-event",
G_CALLBACK (grab_broken_event_cb), notebook);
priv->toplevel_motion_notify_handler_id =
g_signal_connect (toplevel, "motion-notify-event",
G_CALLBACK (toplevel_motion_notify_cb), notebook);
priv->toplevel_button_release_handler_id =
g_signal_connect (toplevel, "button-release-event",
G_CALLBACK (toplevel_button_release_cb), notebook);
priv->grab_notify_handler_id =
g_signal_connect (notebook, "grab-notify",
G_CALLBACK (grab_notify_cb), notebook);
return TRUE;
}
/* this function is only called during dnd, we don't need to emit TABS_REORDERED
* here, instead we do it on drag_stop
*/
static void
move_tab (EphyNotebook *notebook,
int dest_position)
{
gint cur_page_num;
cur_page_num = gtk_notebook_get_current_page (GTK_NOTEBOOK (notebook));
if (dest_position != cur_page_num)
{
GtkWidget *cur_tab;
cur_tab = gtk_notebook_get_nth_page (GTK_NOTEBOOK (notebook),
cur_page_num);
ephy_notebook_move_tab (EPHY_NOTEBOOK (notebook), NULL,
EPHY_TAB (cur_tab),
dest_position);
}
}
static gboolean
motion_notify_cb (EphyNotebook *notebook,
GdkEventMotion *event,
gpointer data)
{
EphyNotebookPrivate *priv = notebook->priv;
if (priv->drag_in_progress == FALSE)
{
if (gtk_drag_check_threshold (GTK_WIDGET (notebook),
notebook->priv->x_start,
notebook->priv->y_start,
event->x_root, event->y_root))
{
return drag_start (notebook, event->time);
}
return FALSE;
}
return FALSE;
}
static void
move_tab_to_another_notebook (EphyNotebook *src,
EphyNotebook *dest,
GdkEventMotion *event,
int dest_position)
{
#ifndef KEEP_TAB_IN_SAME_TOPLEVEL
EphyTab *tab;
int cur_page;
/* This is getting tricky, the tab was dragged in a notebook
* in another window of the same app, we move the tab
* to that new notebook, and let this notebook handle the
* drag
*/
g_assert (EPHY_IS_NOTEBOOK (dest));
g_assert (dest != src);
/* Check if the dest notebook can accept the drag */
if (!dest->priv->dnd_enabled) return;
cur_page = gtk_notebook_get_current_page (GTK_NOTEBOOK (src));
tab = EPHY_TAB (gtk_notebook_get_nth_page (GTK_NOTEBOOK (src), cur_page));
/* stop drag in origin window */
*/
drag_stop (src, event->time);
ephy_notebook_move_tab (src, dest, tab, dest_position);
/* start drag handling in dest notebook */
dest->priv->motion_notify_handler_id =
g_signal_connect (G_OBJECT (dest),
"motion-notify-event",
G_CALLBACK (motion_notify_cb),
NULL);
drag_start (dest, event->time);
#endif /* KEEP_TAB_IN_SAME_TOPLEVEL */
}
static gboolean
button_release_cb (EphyNotebook *notebook,
GdkEventButton *event,
gpointer data)
{
/* This must be called even if a drag isn't happening */
drag_stop (notebook, event->time);
return FALSE;
}
static gboolean
button_press_cb (EphyNotebook *notebook,
GdkEventButton *event,
gpointer data)
{
EphyNotebookPrivate *priv = notebook->priv;
int tab_clicked;
if (!notebook->priv->dnd_enabled) return FALSE;
tab_clicked = find_tab_num_at_pos (notebook, event->x_root, event->y_root);
/* FIXME: how could there be a drag in progress!? */
if (notebook->priv->drag_in_progress)
{
return TRUE;
}
if ((event->button == 1) && (event->type == GDK_BUTTON_PRESS)
&& (tab_clicked >= 0))
{
priv->x_start = event->x_root;
priv->y_start = event->y_root;
priv->motion_notify_handler_id =
g_signal_connect (notebook, "motion-notify-event",
G_CALLBACK (motion_notify_cb), NULL);
}
else if (GDK_BUTTON_PRESS == event->type && 3 == event->button)
{
if (tab_clicked == -1)
{
/* consume event, so that we don't pop up the context menu when
* the mouse if not over a tab label
*/
return TRUE;
}
else
{
/* switch to the page the mouse is over, but don't consume the event */
gtk_notebook_set_current_page (GTK_NOTEBOOK (notebook), tab_clicked);
}
}
return FALSE;
}
GtkWidget *
ephy_notebook_new (void)
{
return GTK_WIDGET (g_object_new (EPHY_TYPE_NOTEBOOK, NULL));
}
static void
ephy_notebook_switch_page_cb (GtkNotebook *notebook,
GtkNotebookPage *page,
guint page_num,
gpointer data)
{
EphyNotebook *nb = EPHY_NOTEBOOK (notebook);
GtkWidget *child;
child = gtk_notebook_get_nth_page (notebook, page_num);
/* Remove the old page, we dont want to grow unnecessarily
* the list */
if (nb->priv->focused_pages)
{
nb->priv->focused_pages =
g_list_remove (nb->priv->focused_pages, child);
}
nb->priv->focused_pages = g_list_append (nb->priv->focused_pages,
child);
}
static void
notebook_drag_data_received_cb (GtkWidget* widget, GdkDragContext *context,
gint x, gint y, GtkSelectionData *selection_data,
guint info, guint time, EphyTab *tab)
{
EphyWindow *window;
GtkWidget *notebook;
g_signal_stop_emission_by_name (widget, "drag_data_received");
if (eel_gconf_get_boolean (CONF_LOCKDOWN_DISABLE_ARBITRARY_URL)) return;
if (selection_data->length <= 0 || selection_data->data == NULL) return;
window = EPHY_WINDOW (gtk_widget_get_toplevel (widget));
notebook = ephy_window_get_notebook (window);
if (selection_data->target == gdk_atom_intern (EPHY_DND_URL_TYPE, FALSE))
{
char **split;
/* URL_TYPE has format: url \n title */
split = g_strsplit ((const gchar *)selection_data->data, "\n", 2);
if (split != NULL && split[0] != NULL && split[0][0] != '\0')
{
ephy_link_open (EPHY_LINK (notebook), split[0], tab,
tab ? 0 : EPHY_LINK_NEW_TAB);
}
g_strfreev (split);
}
else if (selection_data->target == gdk_atom_intern (EPHY_DND_URI_LIST_TYPE, FALSE))
{
char **uris;
int i;
uris = gtk_selection_data_get_uris (selection_data);
if (uris == NULL) return;
for (i = 0; uris[i] != NULL && i < INSANE_NUMBER_OF_URLS; i++)
{
tab = ephy_link_open (EPHY_LINK (notebook), uris[i],
tab, i == 0 ? 0 : EPHY_LINK_NEW_TAB);
}
g_strfreev (uris);
}
else
{
g_return_if_reached ();
}
}
/*
* update_tabs_visibility: Hide tabs if there is only one tab
* and the pref is not set.
*/
static void
update_tabs_visibility (EphyNotebook *nb, gboolean before_inserting)
{
gboolean show_tabs;
guint num;
num = gtk_notebook_get_n_pages (GTK_NOTEBOOK (nb));
if (before_inserting) num++;
show_tabs = (eel_gconf_get_boolean (CONF_ALWAYS_SHOW_TABS_BAR) || num > 1) &&
nb->priv->show_tabs == TRUE;
gtk_notebook_set_show_tabs (GTK_NOTEBOOK (nb), show_tabs);
}
static void
tabs_visibility_notifier (GConfClient *client,
guint cnxn_id,
GConfEntry *entry,
EphyNotebook *nb)
{
update_tabs_visibility (nb, FALSE);
}
static void
ephy_notebook_init (EphyNotebook *notebook)
{
notebook->priv = EPHY_NOTEBOOK_GET_PRIVATE (notebook);
gtk_notebook_set_scrollable (GTK_NOTEBOOK (notebook), TRUE);
gtk_notebook_set_show_border (GTK_NOTEBOOK (notebook), FALSE);
gtk_notebook_set_show_tabs (GTK_NOTEBOOK (notebook), FALSE);
notebook->priv->title_tips = gtk_tooltips_new ();
g_object_ref (G_OBJECT (notebook->priv->title_tips));
gtk_object_sink (GTK_OBJECT (notebook->priv->title_tips));
notebook->priv->show_tabs = TRUE;
notebook->priv->dnd_enabled = TRUE;
g_signal_connect (notebook, "button-press-event",
(GCallback)button_press_cb, NULL);
g_signal_connect (notebook, "button-release-event",
(GCallback)button_release_cb, NULL);
gtk_widget_add_events (GTK_WIDGET (notebook), GDK_BUTTON1_MOTION_MASK);
g_signal_connect_after (G_OBJECT (notebook), "switch_page",
G_CALLBACK (ephy_notebook_switch_page_cb),
NULL);
/* Set up drag-and-drop target */
g_signal_connect (G_OBJECT(notebook), "drag_data_received",
G_CALLBACK(notebook_drag_data_received_cb),
NULL);
gtk_drag_dest_set (GTK_WIDGET(notebook), GTK_DEST_DEFAULT_MOTION |
GTK_DEST_DEFAULT_DROP,
url_drag_types, G_N_ELEMENTS (url_drag_types),
GDK_ACTION_MOVE | GDK_ACTION_COPY);
notebook->priv->tabs_vis_notifier_id = eel_gconf_notification_add
(CONF_ALWAYS_SHOW_TABS_BAR,
(GConfClientNotifyFunc)tabs_visibility_notifier, notebook);
}
static void
ephy_notebook_finalize (GObject *object)
{
EphyNotebook *notebook = EPHY_NOTEBOOK (object);
eel_gconf_notification_remove (notebook->priv->tabs_vis_notifier_id);
if (notebook->priv->focused_pages)
{
g_list_free (notebook->priv->focused_pages);
}
g_object_unref (notebook->priv->title_tips);
G_OBJECT_CLASS (parent_class)->finalize (object);
}
static void
sync_load_status (EphyTab *tab, GParamSpec *pspec, GtkWidget *proxy)
{
GtkWidget *spinner, *icon;
spinner = GTK_WIDGET (g_object_get_data (G_OBJECT (proxy), "spinner"));
icon = GTK_WIDGET (g_object_get_data (G_OBJECT (proxy), "icon"));
g_return_if_fail (spinner != NULL && icon != NULL);
if (ephy_tab_get_load_status (tab))
{
gtk_widget_hide (icon);
gtk_widget_show (spinner);
ephy_spinner_start (EPHY_SPINNER (spinner));
}
else
{
ephy_spinner_stop (EPHY_SPINNER (spinner));
gtk_widget_hide (spinner);
gtk_widget_show (icon);
}
}
static void
sync_icon (EphyTab *tab,
GParamSpec *pspec,
GtkWidget *proxy)
{
GtkImage *icon;
icon = GTK_IMAGE (g_object_get_data (G_OBJECT (proxy), "icon"));
g_return_if_fail (icon != NULL);
gtk_image_set_from_pixbuf (icon, ephy_tab_get_icon (tab));
}
static void
sync_label (EphyTab *tab, GParamSpec *pspec, GtkWidget *proxy)
{
GtkWidget *label, *ebox;
GtkTooltips *tips;
const char *title;
ebox = GTK_WIDGET (g_object_get_data (G_OBJECT (proxy), "label-ebox"));
tips = GTK_TOOLTIPS (g_object_get_data (G_OBJECT (proxy), "tooltips"));
label = GTK_WIDGET (g_object_get_data (G_OBJECT (proxy), "label"));
g_return_if_fail (ebox != NULL && tips != NULL && label != NULL);
title = ephy_tab_get_title (tab);
if (title)
{
gtk_label_set_text (GTK_LABEL (label), title);
gtk_tooltips_set_tip (tips, ebox, title, NULL);
}
}
static void
close_button_clicked_cb (GtkWidget *widget, GtkWidget *tab)
{
EphyNotebook *notebook;
gboolean inhibited = FALSE;
notebook = EPHY_NOTEBOOK (gtk_widget_get_parent (tab));
g_signal_emit (G_OBJECT (notebook), signals[TAB_DELETE], 0, tab, &inhibited);
if (inhibited == FALSE)
{
ephy_notebook_remove_tab (notebook, EPHY_TAB (tab));
}
}
static void
tab_label_style_set_cb (GtkWidget *label,
GtkStyle *previous_style,
GtkWidget *hbox)
{
PangoFontMetrics *metrics;
PangoContext *context;
int char_width, h, w;
context = gtk_widget_get_pango_context (label);
metrics = pango_context_get_metrics (context,
label->style->font_desc,
pango_context_get_language (context));
char_width = pango_font_metrics_get_approximate_digit_width (metrics);
pango_font_metrics_unref (metrics);
gtk_icon_size_lookup_for_settings (gtk_widget_get_settings (label),
GTK_ICON_SIZE_MENU, &w, &h);
gtk_widget_set_size_request
(hbox, TAB_WIDTH_N_CHARS * PANGO_PIXELS(char_width) + 2 * w, -1);
}
static GtkWidget *
build_tab_label (EphyNotebook *nb, EphyTab *tab)
{
GtkWidget *hbox, *label_hbox, *label_ebox;
GtkWidget *label, *close_button, *image, *spinner, *icon;
GtkIconSize close_icon_size;
/* set hbox spacing and label padding (see below) so that there's an
* equal amount of space around the label */
hbox = gtk_hbox_new (FALSE, 4);
label_ebox = gtk_event_box_new ();
gtk_event_box_set_visible_window (GTK_EVENT_BOX (label_ebox), FALSE);
gtk_box_pack_start (GTK_BOX (hbox), label_ebox, TRUE, TRUE, 0);
label_hbox = gtk_hbox_new (FALSE, 4);
gtk_container_add (GTK_CONTAINER (label_ebox), label_hbox);
/* setup close button */
close_button = gtk_button_new ();
gtk_button_set_relief (GTK_BUTTON (close_button),
GTK_RELIEF_NONE);
/* don't allow focus on the close button */
gtk_button_set_focus_on_click (GTK_BUTTON (close_button), FALSE);
close_icon_size = gtk_icon_size_from_name (EPHY_ICON_SIZE_TAB_BUTTON);
image = gtk_image_new_from_stock (EPHY_STOCK_CLOSE_TAB, close_icon_size);
gtk_container_add (GTK_CONTAINER (close_button), image);
gtk_box_pack_start (GTK_BOX (hbox), close_button, FALSE, FALSE, 0);
gtk_tooltips_set_tip (nb->priv->title_tips, close_button,
_("Close tab"), NULL);
g_signal_connect (G_OBJECT (close_button), "clicked",
G_CALLBACK (close_button_clicked_cb),
tab);
/* setup load feedback */
spinner = ephy_spinner_new ();
ephy_spinner_set_size (EPHY_SPINNER (spinner), GTK_ICON_SIZE_MENU);
gtk_box_pack_start (GTK_BOX (label_hbox), spinner, FALSE, FALSE, 0);
/* setup site icon, empty by default */
icon = gtk_image_new ();
gtk_box_pack_start (GTK_BOX (label_hbox), icon, FALSE, FALSE, 0);
/* setup label */
label = gtk_label_new ("");
gtk_label_set_ellipsize (GTK_LABEL (label), PANGO_ELLIPSIZE_END);
gtk_label_set_single_line_mode (GTK_LABEL (label), TRUE);
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
gtk_misc_set_padding (GTK_MISC (label), 0, 0);
gtk_box_pack_start (GTK_BOX (label_hbox), label, TRUE, TRUE, 0);
/* Set minimal size */
g_signal_connect (label, "style-set",
G_CALLBACK (tab_label_style_set_cb), hbox);
gtk_widget_show (hbox);
gtk_widget_show (label_ebox);
gtk_widget_show (label_hbox);
gtk_widget_show (label);
gtk_widget_show (image);
gtk_widget_show (close_button);
g_object_set_data (G_OBJECT (hbox), "label", label);
g_object_set_data (G_OBJECT (hbox), "label-ebox", label_ebox);
g_object_set_data (G_OBJECT (hbox), "spinner", spinner);
g_object_set_data (G_OBJECT (hbox), "icon", icon);
g_object_set_data (G_OBJECT (hbox), "close-button", close_button);
g_object_set_data (G_OBJECT (hbox), "tooltips", nb->priv->title_tips);
return hbox;
}
void
ephy_notebook_set_show_tabs (EphyNotebook *nb, gboolean show_tabs)
{
nb->priv->show_tabs = show_tabs;
update_tabs_visibility (nb, FALSE);
}
void
ephy_notebook_add_tab (EphyNotebook *nb,
EphyTab *tab,
int position,
gboolean jump_to)
{
GtkWidget *label;
g_return_if_fail (EPHY_IS_TAB (tab));
label = build_tab_label (nb, tab);
update_tabs_visibility (nb, TRUE);
gtk_notebook_insert_page (GTK_NOTEBOOK (nb), GTK_WIDGET (tab),
label, position);
/* Set up drag-and-drop target */
g_signal_connect (G_OBJECT(label), "drag_data_received",
G_CALLBACK(notebook_drag_data_received_cb), tab);
gtk_drag_dest_set (label, GTK_DEST_DEFAULT_ALL,
url_drag_types, G_N_ELEMENTS (url_drag_types),
GDK_ACTION_MOVE | GDK_ACTION_COPY);
sync_icon (tab, NULL, label);
sync_label (tab, NULL, label);
sync_load_status (tab, NULL, label);
g_signal_connect_object (tab, "notify::icon",
G_CALLBACK (sync_icon), label, 0);
g_signal_connect_object (tab, "notify::title",
G_CALLBACK (sync_label), label, 0);
g_signal_connect_object (tab, "notify::load-status",
G_CALLBACK (sync_load_status), label, 0);
g_signal_emit (G_OBJECT (nb), signals[TAB_ADDED], 0, tab);
/* The signal handler may have reordered the tabs */
position = gtk_notebook_page_num (GTK_NOTEBOOK (nb), GTK_WIDGET (tab));
if (jump_to)
{
gtk_notebook_set_current_page (GTK_NOTEBOOK (nb), position);
g_object_set_data (G_OBJECT (tab), "jump_to",
GINT_TO_POINTER (jump_to));
}
}
static void
smart_tab_switching_on_closure (EphyNotebook *nb,
EphyTab *tab)
{
gboolean jump_to;
jump_to = GPOINTER_TO_INT (g_object_get_data
(G_OBJECT (tab), "jump_to"));
if (!jump_to || !nb->priv->focused_pages)
{
gtk_notebook_next_page (GTK_NOTEBOOK (nb));
}
else
{
GList *l;
GtkWidget *child;
int page_num;
/* activate the last focused tab */
l = g_list_last (nb->priv->focused_pages);
child = GTK_WIDGET (l->data);
page_num = gtk_notebook_page_num (GTK_NOTEBOOK (nb),
child);
gtk_notebook_set_current_page
(GTK_NOTEBOOK (nb), page_num);
}
}
void
ephy_notebook_remove_tab (EphyNotebook *nb,
EphyTab *tab)
{
int position, curr;
GtkWidget *label, *ebox;
g_return_if_fail (EPHY_IS_NOTEBOOK (nb));
g_return_if_fail (EPHY_IS_TAB (tab));
/* Remove the page from the focused pages list */
nb->priv->focused_pages = g_list_remove (nb->priv->focused_pages,
tab);
position = gtk_notebook_page_num (GTK_NOTEBOOK (nb), GTK_WIDGET (tab));
curr = gtk_notebook_get_current_page (GTK_NOTEBOOK (nb));
if (position == curr)
{
smart_tab_switching_on_closure (nb, tab);
}
label = gtk_notebook_get_tab_label (GTK_NOTEBOOK (nb), GTK_WIDGET (tab));
ebox = GTK_WIDGET (g_object_get_data (G_OBJECT (label), "label-ebox"));
gtk_tooltips_set_tip (GTK_TOOLTIPS (nb->priv->title_tips), ebox, NULL, NULL);
g_signal_handlers_disconnect_by_func (tab,
G_CALLBACK (sync_icon), label);
g_signal_handlers_disconnect_by_func (tab,
G_CALLBACK (sync_label), label);
g_signal_handlers_disconnect_by_func (tab,
G_CALLBACK (sync_load_status), label);
/**
* we ref the tab so that it's still alive while the tabs_removed
* signal is processed.
*/
g_object_ref (tab);
gtk_notebook_remove_page (GTK_NOTEBOOK (nb), position);
update_tabs_visibility (nb, FALSE);
g_signal_emit (G_OBJECT (nb), signals[TAB_REMOVED], 0, tab);
g_object_unref (tab);
/* if that was the last tab, destroy the window */
if (gtk_notebook_get_n_pages (GTK_NOTEBOOK (nb)) == 0)
{
gtk_widget_destroy (gtk_widget_get_toplevel (GTK_WIDGET (nb)));
}
}