/*
* Copyright (C) 2000 Marco Pesenti Gritti
* (C) 2001, 2002 Jorn Baayen
*
* 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.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include "toolbar.h"
#include "egg-menu-merge.h"
#include "ephy-file-helpers.h"
#include "ephy-shell.h"
#include "ephy-location-entry.h"
#include "ephy-dnd.h"
#include "ephy-spinner.h"
#include "ephy-spinner-action.h"
#include "ephy-location-action.h"
#include "ephy-favicon-action.h"
#include "ephy-topic-action.h"
#include "ephy-go-action.h"
#include "ephy-navigation-action.h"
#include "ephy-bookmark-action.h"
#include "ephy-zoom-action.h"
#include "window-commands.h"
#include "ephy-string.h"
#include "ephy-debug.h"
#include "ephy-new-bookmark.h"
#include "ephy-stock-icons.h"
#include "eggtoolbar.h"
#include <string.h>
static void toolbar_class_init (ToolbarClass *klass);
static void toolbar_init (Toolbar *t);
static void toolbar_finalize (GObject *object);
static void toolbar_set_window (Toolbar *t, EphyWindow *window);
static void
toolbar_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec);
static void
toolbar_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec);
static GtkTargetEntry drag_targets[] = {
{ EGG_TOOLBAR_ITEM_TYPE, 0, 0 },
{ EPHY_DND_TOPIC_TYPE, 0, 1 },
{ EPHY_DND_BOOKMARK_TYPE, 0, 2 },
};
static int n_drag_targets = G_N_ELEMENTS (drag_targets);
enum
{
PROP_0,
PROP_EPHY_WINDOW
};
static GObjectClass *parent_class = NULL;
struct ToolbarPrivate
{
EphyWindow *window;
EggMenuMerge *ui_merge;
EggActionGroup *action_group;
gboolean visibility;
gboolean can_set_location;
GtkWidget *spinner;
GtkWidget *favicon;
GtkWidget *go;
};
GType
toolbar_get_type (void)
{
static GType toolbar_type = 0;
if (toolbar_type == 0)
{
static const GTypeInfo our_info =
{
sizeof (ToolbarClass),
NULL, /* base_init */
NULL, /* base_finalize */
(GClassInitFunc) toolbar_class_init,
NULL,
NULL, /* class_data */
sizeof (Toolbar),
0, /* n_preallocs */
(GInstanceInitFunc) toolbar_init
};
toolbar_type = g_type_register_static (EGG_EDITABLE_TOOLBAR_TYPE,
"Toolbar",
&our_info, 0);
}
return toolbar_type;
}
static void
go_location_cb (EggAction *action, char *location, EphyWindow *window)
{
ephy_window_load_url (window, location);
}
static void
zoom_to_level_cb (EggAction *action, float zoom, EphyWindow *window)
{
ephy_window_set_zoom (window, zoom);
}
static void
topic_destroy_cb (EphyNode *node,
Toolbar *t)
{
EggAction *action;
char *name;
EphyToolbarsModel *model;
long id;
model = ephy_shell_get_toolbars_model (ephy_shell);
id = ephy_node_get_id (node);
name = g_strdup_printf ("GoTopicId%ld", ephy_node_get_id (node));
action = egg_action_group_get_action (t->priv->action_group, name);
if (action)
{
ephy_toolbars_model_remove_bookmark (model, TRUE, id);
egg_action_group_remove_action (t->priv->action_group, action);
}
g_free (name);
}
static void
bookmark_destroy_cb (EphyNode *node,
Toolbar *t)
{
EggAction *action;
char *name;
EphyToolbarsModel *model;
long id;
model = ephy_shell_get_toolbars_model (ephy_shell);
id = ephy_node_get_id (node);
name = g_strdup_printf ("GoBookmarkId%ld", id);
action = egg_action_group_get_action (t->priv->action_group, name);
if (action)
{
ephy_toolbars_model_remove_bookmark (model, FALSE, id);
egg_action_group_remove_action (t->priv->action_group, action);
}
g_free (name);
}
static void
toolbar_ensure_action (Toolbar *t,
const char *name)
{
EggAction *action = NULL;
long id = 0;
EphyBookmarks *bookmarks;
LOG ("Ensure action %s", name)
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
if (g_str_has_prefix (name, "GoBookmarkId"))
{
EphyNode *node;
if (!ephy_string_to_int (name + strlen ("GoBookmarkId"), &id))
{
return;
}
LOG ("Create action %s", name)
action = ephy_bookmark_action_new (name, id);
g_signal_connect (action, "go_location",
G_CALLBACK (go_location_cb), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
node = ephy_bookmarks_get_from_id (bookmarks, id);
ephy_node_signal_connect_object (node,
EPHY_NODE_DESTROY,
(EphyNodeCallback) bookmark_destroy_cb,
G_OBJECT (t));
}
else if (g_str_has_prefix (name, "GoTopicId"))
{
EphyNode *node;
if (!ephy_string_to_int (name + strlen ("GoTopicId"), &id))
{
return;
}
LOG ("Create action %s", name)
action = ephy_topic_action_new (name, id);
g_signal_connect (action, "go_location",
G_CALLBACK (go_location_cb),
t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
node = ephy_bookmarks_get_from_id (bookmarks, id);
ephy_node_signal_connect_object (node,
EPHY_NODE_DESTROY,
(EphyNodeCallback) topic_destroy_cb,
G_OBJECT (t));
}
}
static void
toolbar_class_init (ToolbarClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
EggEditableToolbarClass *eet_class;
parent_class = g_type_class_peek_parent (klass);
eet_class = EGG_EDITABLE_TOOLBAR_CLASS (klass);
object_class->finalize = toolbar_finalize;
object_class->set_property = toolbar_set_property;
object_class->get_property = toolbar_get_property;
g_object_class_install_property (object_class,
PROP_EPHY_WINDOW,
g_param_spec_object ("EphyWindow",
"EphyWindow",
"Parent window",
EPHY_WINDOW_TYPE,
G_PARAM_READWRITE));
}
static void
toolbar_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
Toolbar *t = TOOLBAR (object);
switch (prop_id)
{
case PROP_EPHY_WINDOW:
toolbar_set_window (t, g_value_get_object (value));
break;
}
}
static void
toolbar_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
Toolbar *t = TOOLBAR (object);
switch (prop_id)
{
case PROP_EPHY_WINDOW:
g_value_set_object (value, t->priv->window);
break;
}
}
static void
toolbar_setup_actions (Toolbar *t)
{
EggAction *action;
t->priv->action_group = egg_action_group_new ("SpecialToolbarActions");
action = g_object_new (EPHY_TYPE_NAVIGATION_ACTION,
"name", "NavigationBack",
"label", _("Back"),
"stock_id", GTK_STOCK_GO_BACK,
"window", t->priv->window,
"direction", EPHY_NAVIGATION_DIRECTION_BACK,
NULL);
g_signal_connect (action, "activate",
G_CALLBACK (window_cmd_go_back), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_NAVIGATION_ACTION,
"name", "NavigationForward",
"label", _("Forward"),
"stock_id", GTK_STOCK_GO_FORWARD,
"window", t->priv->window,
"direction", EPHY_NAVIGATION_DIRECTION_FORWARD,
NULL);
g_signal_connect (action, "activate",
G_CALLBACK (window_cmd_go_forward), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_NAVIGATION_ACTION,
"name", "NavigationUp",
"label", _("Up"),
"window", t->priv->window,
"direction", EPHY_NAVIGATION_DIRECTION_UP,
"stock_id", GTK_STOCK_GO_UP,
NULL);
g_signal_connect (action, "activate",
G_CALLBACK (window_cmd_go_up), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_SPINNER_ACTION,
"name", "Spinner",
"label", _("Spinner"),
NULL);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
/* FIXME: I'm still waiting for the exact term to
* user here from the docs team.
*/
action = g_object_new (EPHY_TYPE_LOCATION_ACTION,
"name", "Location",
"label", _("Address Entry"),
"stock_id", EPHY_STOCK_ENTRY,
"tooltip", _("Enter a web address to open, or a phrase to search for on the web."),
NULL);
g_signal_connect (action, "go_location",
G_CALLBACK (go_location_cb), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_ZOOM_ACTION,
"name", "Zoom",
"label", _("Zoom"),
"zoom", 1.0,
NULL);
g_signal_connect (action, "zoom_to_level",
G_CALLBACK (zoom_to_level_cb), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_FAVICON_ACTION,
"name", "Favicon",
"label", _("Favicon"),
"window", t->priv->window,
NULL);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
action = g_object_new (EPHY_TYPE_GO_ACTION,
"name", "ToolbarGo",
"label", _("Go"),
"stock_id", GTK_STOCK_JUMP_TO,
NULL);
g_signal_connect (action, "activate",
G_CALLBACK (window_cmd_load_location), t->priv->window);
egg_action_group_add_action (t->priv->action_group, action);
g_object_unref (action);
}
static void
action_request_cb (EggEditableToolbar *etoolbar,
char *action_name,
gpointer data)
{
toolbar_ensure_action (TOOLBAR (etoolbar), action_name);
}
static void
init_bookmarks_toolbar (Toolbar *t)
{
EphyToolbarsModel *model;
int i, n_toolbars;
model = ephy_shell_get_toolbars_model (ephy_shell);
n_toolbars = egg_toolbars_model_n_toolbars
(EGG_TOOLBARS_MODEL (model));
for (i = 0; i < n_toolbars; i++)
{
const char *t_name;
t_name = egg_toolbars_model_toolbar_nth
(EGG_TOOLBARS_MODEL (model), i);
g_return_if_fail (t_name != NULL);
if (strcmp (t_name, "BookmarksBar") == 0)
{
egg_editable_toolbar_set_drag_dest
(EGG_EDITABLE_TOOLBAR (t),
drag_targets, n_drag_targets,
t_name);
egg_toolbars_model_set_flags
(EGG_TOOLBARS_MODEL (model),
EGG_TB_MODEL_NOT_REMOVABLE, i);
}
}
}
static void
update_toolbar_remove_flag (EphyToolbarsModel *model, gpointer data)
{
int i, n_toolbars;
int not_removable = 0;
n_toolbars = egg_toolbars_model_n_toolbars
(EGG_TOOLBARS_MODEL (model));
/* If there is only one toolbar and the bookmarks bar */
if (n_toolbars <= 2)
{
not_removable = EGG_TB_MODEL_NOT_REMOVABLE;
}
for (i = 0; i < n_toolbars; i++)
{
const char *t_name;
t_name = egg_toolbars_model_toolbar_nth
(EGG_TOOLBARS_MODEL (model), i);
g_return_if_fail (t_name != NULL);
if (!(strcmp (t_name, "BookmarksBar") == 0))
{
egg_toolbars_model_set_flags
(EGG_TOOLBARS_MODEL (model),
not_removable, i);
}
}
}
static GtkWidget *
get_location_entry (Toolbar *t)
{
EggAction *action;
GtkWidget *location;
action = egg_action_group_get_action
(t->priv->action_group, "Location");
location = ephy_location_action_get_widget
(EPHY_LOCATION_ACTION (action));
return location;
}
static void
location_user_changed_cb (GtkWidget *entry, Toolbar *t)
{
EphyTab *tab;
char *address;
tab = ephy_window_get_active_tab (t->priv->window);
g_return_if_fail (IS_EPHY_TAB (tab));
t->priv->can_set_location = FALSE;
address = ephy_location_entry_get_location (EPHY_LOCATION_ENTRY (entry));
ephy_tab_set_location (tab, address, TAB_ADDRESS_EXPIRE_CURRENT);
g_free (address);
t->priv->can_set_location = TRUE;
}
static void
toolbar_set_window (Toolbar *t, EphyWindow *window)
{
EphyToolbarsModel *model;
g_return_if_fail (t->priv->window == NULL);
t->priv->window = window;
t->priv->ui_merge = EGG_MENU_MERGE (window->ui_merge);
toolbar_setup_actions (t);
egg_menu_merge_insert_action_group (t->priv->ui_merge,
t->priv->action_group, 1);
g_signal_connect (t, "action_request",
G_CALLBACK (action_request_cb),
NULL);
model = ephy_shell_get_toolbars_model (ephy_shell);
g_signal_connect (EGG_TOOLBARS_MODEL (model), "toolbar_added",
G_CALLBACK (update_toolbar_remove_flag),
NULL);
g_signal_connect (EGG_TOOLBARS_MODEL (model), "toolbar_removed",
G_CALLBACK (update_toolbar_remove_flag),
NULL);
g_object_set (G_OBJECT (t),
"ToolbarsModel", model,
"MenuMerge", t->priv->ui_merge,
NULL);
init_bookmarks_toolbar (t);
g_signal_connect_object (get_location_entry (t), "user_changed",
G_CALLBACK (location_user_changed_cb),
t, 0);
}
static void
toolbar_init (Toolbar *t)
{
t->priv = g_new0 (ToolbarPrivate, 1);
t->priv->window = NULL;
t->priv->ui_merge = NULL;
t->priv->visibility = TRUE;
t->priv->can_set_location = TRUE;
}
static void
toolbar_finalize (GObject *object)
{
Toolbar *t;
ToolbarPrivate *p;
EggMenuMerge *merge;
g_return_if_fail (object != NULL);
g_return_if_fail (IS_TOOLBAR (object));
t = TOOLBAR (object);
p = t->priv;
merge = EGG_MENU_MERGE (t->priv->window->ui_merge);
g_return_if_fail (p != NULL);
G_OBJECT_CLASS (parent_class)->finalize (object);
g_object_unref (t->priv->action_group);
egg_menu_merge_remove_action_group (merge, t->priv->action_group);
g_free (t->priv);
LOG ("Toolbar finalized")
}
Toolbar *
toolbar_new (EphyWindow *window)
{
Toolbar *t;
t = TOOLBAR (g_object_new (TOOLBAR_TYPE,
"EphyWindow", window,
NULL));
g_return_val_if_fail (t->priv != NULL, NULL);
return t;
}
static void
location_finished_cb (GtkWidget *location, GtkWidget *toolbar)
{
gtk_widget_hide (toolbar);
g_signal_handlers_disconnect_by_func (G_OBJECT (location),
G_CALLBACK (location_finished_cb),
toolbar);
}
void
toolbar_activate_location (Toolbar *t)
{
GtkWidget *location;
GtkWidget *location_tb;
location = get_location_entry (t);
g_return_if_fail (location != NULL);
location_tb = gtk_widget_get_ancestor (location, EGG_TYPE_TOOLBAR);
g_return_if_fail (location_tb != NULL);
if (!GTK_WIDGET_VISIBLE (location_tb))
{
g_signal_connect (location, "finished",
G_CALLBACK (location_finished_cb), location_tb);
gtk_widget_show (location_tb);
}
ephy_location_entry_activate
(EPHY_LOCATION_ENTRY(location));
}
void
toolbar_spinner_start (Toolbar *t)
{
EggActionGroup *action_group;
EggAction *action;
action_group = t->priv->action_group;
action = egg_action_group_get_action (action_group, "Spinner");
g_object_set (action, "throbbing", TRUE, NULL);
}
void
toolbar_spinner_stop (Toolbar *t)
{
EggActionGroup *action_group;
EggAction *action;
action_group = t->priv->action_group;
action = egg_action_group_get_action (action_group, "Spinner");
g_object_set (action, "throbbing", FALSE, NULL);
}
void
toolbar_set_location (Toolbar *t,
const char *alocation)
{
GtkWidget *location;
if (t->priv->can_set_location)
{
location = get_location_entry (t);
g_return_if_fail (location != NULL);
ephy_location_entry_set_location
(EPHY_LOCATION_ENTRY (location),
alocation ? alocation : "");
}
}
void
toolbar_update_favicon (Toolbar *t)
{
EphyTab *tab;
const char *url;
EggActionGroup *action_group;
EggAction *action;
tab = ephy_window_get_active_tab (t->priv->window);
url = ephy_tab_get_icon_address (tab);
action_group = t->priv->action_group;
action = egg_action_group_get_action (action_group, "Favicon");
g_object_set (action, "icon", url, NULL);
}
char *
toolbar_get_location (Toolbar *t)
{
GtkWidget *location;
location = get_location_entry (t);
g_return_val_if_fail (location != NULL, NULL);
return ephy_location_entry_get_location
(EPHY_LOCATION_ENTRY (location));
}
void
toolbar_clear_location_history (Toolbar *t)
{
GtkWidget *location;
location = get_location_entry (t),
g_return_if_fail (location != NULL);
ephy_location_entry_clear_history (EPHY_LOCATION_ENTRY (location));
}
void
toolbar_update_navigation_actions (Toolbar *t, gboolean back, gboolean forward, gboolean up)
{
EggActionGroup *action_group;
EggAction *action;
action_group = t->priv->action_group;
action = egg_action_group_get_action (action_group, "NavigationBack");
g_object_set (action, "sensitive", back, NULL);
action = egg_action_group_get_action (action_group, "NavigationForward");
g_object_set (action, "sensitive", forward, NULL);
action = egg_action_group_get_action (action_group, "NavigationUp");
g_object_set (action, "sensitive", up, NULL);
}
void
toolbar_set_visibility (Toolbar *t,
gboolean normal_toolbars,
gboolean bmk_toolbars)
{
EphyToolbarsModel *model;
int i, n_toolbars;
model = ephy_shell_get_toolbars_model (ephy_shell);
n_toolbars = egg_toolbars_model_n_toolbars
(EGG_TOOLBARS_MODEL (model));
for (i = 0; i < n_toolbars; i++)
{
const char *t_name;
t_name = egg_toolbars_model_toolbar_nth
(EGG_TOOLBARS_MODEL (model), i);
g_return_if_fail (t_name != NULL);
if (strcmp (t_name, "BookmarksBar") == 0)
{
if (bmk_toolbars)
{
egg_editable_toolbar_show
(EGG_EDITABLE_TOOLBAR (t), t_name);
}
else
{
egg_editable_toolbar_hide
(EGG_EDITABLE_TOOLBAR (t), t_name);
}
}
else
{
if (normal_toolbars)
{
egg_editable_toolbar_show
(EGG_EDITABLE_TOOLBAR (t), t_name);
}
else
{
egg_editable_toolbar_hide
(EGG_EDITABLE_TOOLBAR (t), t_name);
}
}
}
}
void
toolbar_update_zoom (Toolbar *t, float zoom)
{
EggActionGroup *action_group;
EggAction *action;
action_group = t->priv->action_group;
action = egg_action_group_get_action (action_group, "Zoom");
g_object_set (action, "zoom", zoom, NULL);
}