/*
* Copyright (C) 2002 Marco Pesenti Gritti
*
* 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 "ephy-toolbars-model.h"
#include "ephy-dnd.h"
#include "ephy-bookmarks.h"
#include "ephy-node-common.h"
#include "ephy-file-helpers.h"
#include "ephy-shell.h"
#include "ephy-debug.h"
#include "ephy-string.h"
#include <string.h>
static void ephy_toolbars_model_class_init (EphyToolbarsModelClass *klass);
static void ephy_toolbars_model_init (EphyToolbarsModel *t);
static void ephy_toolbars_model_finalize (GObject *object);
enum
{
ACTION_ADDED,
LAST_SIGNAL
};
enum
{
PROP_0,
PROP_BOOKMARKS
};
enum
{
URL,
NAME
};
static GObjectClass *parent_class = NULL;
struct EphyToolbarsModelPrivate
{
EphyBookmarks *bookmarks;
char *xml_file;
};
GType
ephy_toolbars_model_get_type (void)
{
static GType ephy_toolbars_model_type = 0;
if (ephy_toolbars_model_type == 0)
{
static const GTypeInfo our_info = {
sizeof (EphyToolbarsModelClass),
NULL, /* base_init */
NULL, /* base_finalize */
(GClassInitFunc) ephy_toolbars_model_class_init,
NULL,
NULL, /* class_data */
sizeof (EphyToolbarsModel),
0, /* n_preallocs */
(GInstanceInitFunc) ephy_toolbars_model_init
};
ephy_toolbars_model_type = g_type_register_static (EGG_TOOLBARS_MODEL_TYPE,
"EphyToolbarsModel",
&our_info, 0);
}
return ephy_toolbars_model_type;
}
char *
ephy_toolbars_model_get_action_name (EphyToolbarsModel *model,
gboolean topic, long id)
{
char *action_name;
const char *name;
EphyNode *node;
EphyNodePriority priority;
node = ephy_bookmarks_get_from_id (model->priv->bookmarks, id);
g_return_val_if_fail (node != NULL, NULL);
priority = ephy_node_get_property_int
(node, EPHY_NODE_KEYWORD_PROP_PRIORITY);
if (topic)
{
if (priority != EPHY_NODE_NORMAL_PRIORITY)
{
action_name = g_strdup_printf ("GoSpecialTopic-%ld", id);
}
else
{
name = ephy_node_get_property_string
(node, EPHY_NODE_KEYWORD_PROP_NAME);
action_name = g_strdup_printf ("GoTopic-%s", name);
}
}
else
{
name = ephy_node_get_property_string
(node, EPHY_NODE_BMK_PROP_LOCATION);
action_name = g_strdup_printf ("GoBookmark-%s", name);
}
return action_name;
}
static void
topic_destroy_cb (EphyNode *node,
EphyToolbarsModel *model)
{
long id;
id = ephy_node_get_id (node);
ephy_toolbars_model_remove_bookmark (model, TRUE, id);
}
static void
bookmark_destroy_cb (EphyNode *node,
EphyToolbarsModel *model)
{
long id;
id = ephy_node_get_id (node);
ephy_toolbars_model_remove_bookmark (model, FALSE, id);
}
static gboolean
setup_item (EphyToolbarsModel *model,
const char *name,
gboolean *is_bookmark,
gboolean *is_topic,
long *id)
{
EphyBookmarks *bookmarks = model->priv->bookmarks;
EphyNode *node = NULL;
*is_topic = FALSE;
if (g_str_has_prefix (name, "GoBookmark-"))
{
node = ephy_bookmarks_find_bookmark
(bookmarks, name + strlen ("GoBookmark-"));
if (node == NULL) return FALSE;
ephy_node_signal_connect_object (node,
EPHY_NODE_DESTROY,
(EphyNodeCallback) bookmark_destroy_cb,
G_OBJECT (model));
}
else if (g_str_has_prefix (name, "GoTopic-"))
{
EphyNode *node;
node = ephy_bookmarks_find_keyword
(bookmarks, name + strlen ("GoTopic-"), FALSE);
if (node == NULL) return FALSE;
ephy_node_signal_connect_object (node,
EPHY_NODE_DESTROY,
(EphyNodeCallback) topic_destroy_cb,
G_OBJECT (model));
*is_topic = TRUE;
}
else if (g_str_has_prefix (name, "GoSpecialTopic-"))
{
EphyNode *node;
long id;
if (!ephy_string_to_int (name + strlen ("GoSpecialTopic-"), &id))
{
return FALSE;
}
node = ephy_bookmarks_get_from_id (bookmarks, id);
if (node == NULL) return FALSE;
ephy_node_signal_connect_object (node,
EPHY_NODE_DESTROY,
(EphyNodeCallback) topic_destroy_cb,
G_OBJECT (model));
*is_topic = TRUE;
}
if (node)
{
*is_bookmark = TRUE;
*id = ephy_node_get_id (node);
}
return TRUE;
}
static const char *
impl_add_item (EggToolbarsModel *t,
int toolbar_position,
int position,
GdkAtom type,
const char *name)
{
char *action_name = NULL;
const char *res;
gboolean valid, duped_bookmark = FALSE, is_bookmark, is_topic;
long id = -1;
LOG ("Add item %s", name)
if (gdk_atom_intern (EPHY_DND_TOPIC_TYPE, FALSE) == type)
{
GList *nodes;
nodes = ephy_dnd_node_list_extract_nodes (name);
id = ephy_node_get_id (nodes->data);
action_name = ephy_toolbars_model_get_action_name
(EPHY_TOOLBARS_MODEL (t), TRUE, id);
g_list_free (nodes);
}
else if (gdk_atom_intern (EPHY_DND_BOOKMARK_TYPE, FALSE) == type)
{
GList *nodes;
nodes = ephy_dnd_node_list_extract_nodes (name);
id = ephy_node_get_id (nodes->data);
action_name = ephy_toolbars_model_get_action_name
(EPHY_TOOLBARS_MODEL (t), FALSE, id);
g_list_free (nodes);
}
else if (gdk_atom_intern (EPHY_DND_URL_TYPE, FALSE) == type)
{
EphyNode *node = NULL;
EphyBookmarks *bookmarks;
gchar **netscape_url;
netscape_url = g_strsplit (name, "\n", 2);
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
node = ephy_bookmarks_find_bookmark (bookmarks, netscape_url[URL]);
if (!node)
{
/* Create the bookmark, it does not exist */
EphyHistory *gh;
const char *icon;
node = ephy_bookmarks_add (bookmarks, netscape_url[NAME], netscape_url[URL]);
g_return_val_if_fail (node != NULL, NULL);
gh = ephy_embed_shell_get_global_history (EPHY_EMBED_SHELL (ephy_shell));
icon = ephy_history_get_icon (gh, netscape_url[URL]);
if (icon)
{
ephy_bookmarks_set_icon (bookmarks, netscape_url[URL], icon);
}
}
id = ephy_node_get_id (node);
action_name = ephy_toolbars_model_get_action_name
(EPHY_TOOLBARS_MODEL (t), FALSE, id);
g_strfreev (netscape_url);
}
res = action_name ? action_name : name;
valid = setup_item (EPHY_TOOLBARS_MODEL (t), res, &is_bookmark, &is_topic, &id);
if (!valid)
{
g_warning ("Invalid bookmark in the toolbar. Removed.");
return NULL;
}
if (is_bookmark)
{
duped_bookmark = ephy_toolbars_model_has_bookmark
(EPHY_TOOLBARS_MODEL (t), is_topic, id);
}
if (!duped_bookmark)
{
EGG_TOOLBARS_MODEL_CLASS (parent_class)->add_item
(t, toolbar_position, position, type, res);
}
return res;
}
static void
ephy_toolbars_model_set_bookmarks (EphyToolbarsModel *model, EphyBookmarks *bookmarks)
{
EggToolbarsModel *egg_model = EGG_TOOLBARS_MODEL (model);
model->priv->bookmarks = bookmarks;
g_object_ref (model->priv->bookmarks);
if (g_file_test (model->priv->xml_file, G_FILE_TEST_EXISTS))
{
egg_toolbars_model_load (egg_model,
model->priv->xml_file);
}
else
{
const char *default_xml;
default_xml = ephy_file ("epiphany-toolbar.xml");
egg_toolbars_model_load (egg_model, default_xml);
}
}
static void
ephy_toolbars_model_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyToolbarsModel *model;
model = EPHY_TOOLBARS_MODEL (object);
switch (prop_id)
{
case PROP_BOOKMARKS:
ephy_toolbars_model_set_bookmarks (model, g_value_get_object (value));
break;
}
}
static void
ephy_toolbars_model_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
EphyToolbarsModel *model;
model = EPHY_TOOLBARS_MODEL (object);
switch (prop_id)
{
case PROP_BOOKMARKS:
g_value_set_object (value, model->priv->bookmarks);
break;
}
}
static void
ephy_toolbars_model_class_init (EphyToolbarsModelClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
EggToolbarsModelClass *etm_class;
etm_class = EGG_TOOLBARS_MODEL_CLASS (klass);
parent_class = g_type_class_peek_parent (klass);
object_class->finalize = ephy_toolbars_model_finalize;
object_class->set_property = ephy_toolbars_model_set_property;
object_class->get_property = ephy_toolbars_model_get_property;
etm_class->add_item = impl_add_item;
g_object_class_install_property (object_class,
PROP_BOOKMARKS,
g_param_spec_object ("bookmarks",
"Bookmarks",
"Bookmarks",
EPHY_BOOKMARKS_TYPE,
G_PARAM_READWRITE));
}
static void
item_added (EphyToolbarsModel *model, int toolbar_position, int position)
{
egg_toolbars_model_save (EGG_TOOLBARS_MODEL (model),
model->priv->xml_file);
}
static void
item_removed (EphyToolbarsModel *model, int toolbar_position, int position)
{
egg_toolbars_model_save (EGG_TOOLBARS_MODEL (model),
model->priv->xml_file);
}
static void
toolbar_added (EphyToolbarsModel *model, int position)
{
egg_toolbars_model_save (EGG_TOOLBARS_MODEL (model),
model->priv->xml_file);
}
static void
toolbar_removed (EphyToolbarsModel *model, int position)
{
egg_toolbars_model_save (EGG_TOOLBARS_MODEL (model),
model->priv->xml_file);
}
static void
ephy_toolbars_model_init (EphyToolbarsModel *t)
{
t->priv = g_new0 (EphyToolbarsModelPrivate, 1);
t->priv->bookmarks = NULL;
t->priv->xml_file = g_build_filename (ephy_dot_dir (),
"ephy-toolbars.xml",
NULL);
g_signal_connect (t, "item_added", G_CALLBACK (item_added), NULL);
g_signal_connect (t, "item_removed", G_CALLBACK (item_removed), NULL);
g_signal_connect (t, "toolbar_added", G_CALLBACK (toolbar_added), NULL);
g_signal_connect (t, "toolbar_removed", G_CALLBACK (toolbar_removed), NULL);
}
static void
ephy_toolbars_model_finalize (GObject *object)
{
EphyToolbarsModel *t = EPHY_TOOLBARS_MODEL (object);
g_return_if_fail (object != NULL);
g_return_if_fail (IS_EPHY_TOOLBARS_MODEL (object));
g_object_unref (t->priv->bookmarks);
g_free (t->priv->xml_file);
g_free (t->priv);
G_OBJECT_CLASS (parent_class)->finalize (object);
}
EphyToolbarsModel *
ephy_toolbars_model_new (EphyBookmarks *bookmarks)
{
EphyToolbarsModel *t;
t = EPHY_TOOLBARS_MODEL (g_object_new (EPHY_TOOLBARS_MODEL_TYPE,
"bookmarks", bookmarks,
NULL));
g_return_val_if_fail (t->priv != NULL, NULL);
return t;
}
static int
get_toolbar_pos (EphyToolbarsModel *model,
const char *name)
{
int i, n_toolbars;
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);
if (strcmp (name, t_name) == 0)
{
return i;
}
}
return -1;
}
static gboolean
get_toolbar_and_item_pos (EphyToolbarsModel *model,
const char *action_name,
int *toolbar,
int *position)
{
int n_toolbars, n_items;
int t,i;
n_toolbars = egg_toolbars_model_n_toolbars (EGG_TOOLBARS_MODEL (model));
for (t = 0; t < n_toolbars; t++)
{
n_items = egg_toolbars_model_n_items
(EGG_TOOLBARS_MODEL (model), t);
for (i = 0; i < n_items; i++)
{
const char *i_name;
gboolean is_separator;
i_name = egg_toolbars_model_item_nth
(EGG_TOOLBARS_MODEL (model), t, i,
&is_separator);
g_return_val_if_fail (i_name != NULL, FALSE);
if (strcmp (i_name, action_name) == 0)
{
*toolbar = t;
*position = i;
return TRUE;
}
}
}
return FALSE;
}
void
ephy_toolbars_model_remove_bookmark (EphyToolbarsModel *model,
gboolean topic,
long id)
{
char *action_name;
int toolbar, position;
action_name = ephy_toolbars_model_get_action_name (model, topic, id);
g_return_if_fail (action_name != NULL);
if (get_toolbar_and_item_pos (model, action_name, &toolbar, &position))
{
egg_toolbars_model_remove_item (EGG_TOOLBARS_MODEL (model),
toolbar, position);
}
g_free (action_name);
}
void
ephy_toolbars_model_add_bookmark (EphyToolbarsModel *model,
gboolean topic,
long id)
{
char *action_name;
int toolbar_position;
action_name = ephy_toolbars_model_get_action_name (model, topic, id);
g_return_if_fail (action_name != NULL);
toolbar_position = get_toolbar_pos (model, "BookmarksBar");
g_return_if_fail (toolbar_position != -1);
egg_toolbars_model_add_item (EGG_TOOLBARS_MODEL (model),
toolbar_position, -1,
0, action_name);
g_free (action_name);
}
gboolean
ephy_toolbars_model_has_bookmark (EphyToolbarsModel *model,
gboolean topic,
long id)
{
char *action_name;
gboolean found;
int toolbar, pos;
action_name = ephy_toolbars_model_get_action_name (model, topic, id);
g_return_val_if_fail (action_name != NULL, FALSE);
found = get_toolbar_and_item_pos (model, action_name, &toolbar, &pos);
g_free (action_name);
return found;
}
void
ephy_toolbars_model_set_flag (EphyToolbarsModel *model,
EggTbModelFlags flags)
{
EggToolbarsModel *t = EGG_TOOLBARS_MODEL (model);
int i, n_toolbars;
n_toolbars = egg_toolbars_model_n_toolbars
(EGG_TOOLBARS_MODEL (model));
for (i = 0; i < n_toolbars; i++)
{
EggTbModelFlags old_flags;
old_flags = egg_toolbars_model_get_flags (t, i);
egg_toolbars_model_set_flags (t, old_flags | flags, i);
}
}
void
ephy_toolbars_model_unset_flag (EphyToolbarsModel *model,
EggTbModelFlags flags)
{
EggToolbarsModel *t = EGG_TOOLBARS_MODEL (model);
int i, n_toolbars;
n_toolbars = egg_toolbars_model_n_toolbars
(EGG_TOOLBARS_MODEL (model));
for (i = 0; i < n_toolbars; i++)
{
EggTbModelFlags old_flags;
old_flags = egg_toolbars_model_get_flags (t, i);
egg_toolbars_model_set_flags (t, old_flags ^ flags, i);
}
}