1238 lines
31 KiB
C
1238 lines
31 KiB
C
/*
|
|
* This file is part of the Main Menu.
|
|
*
|
|
* Copyright (c) 2007 Novell, Inc.
|
|
*
|
|
* The Main Menu 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 of the License, or (at your option)
|
|
* any later version.
|
|
*
|
|
* The Main Menu 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
|
|
* the Main Menu; if not, write to the Free Software Foundation, Inc., 51
|
|
* Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "bookmark-agent.h"
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#else
|
|
# define PACKAGE "gnome-main-menu"
|
|
#endif
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <glib/gi18n-lib.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gio/gio.h>
|
|
|
|
#include "libslab-utils.h"
|
|
|
|
#define MODIFIABLE_APPS_GCONF_KEY "/desktop/gnome/applications/main-menu/lock-down/user_modifiable_apps"
|
|
#define MODIFIABLE_DOCS_GCONF_KEY "/desktop/gnome/applications/main-menu/lock-down/user_modifiable_docs"
|
|
#define MODIFIABLE_DIRS_GCONF_KEY "/desktop/gnome/applications/main-menu/lock-down/user_modifiable_dirs"
|
|
#define MODIFIABLE_SYS_GCONF_KEY "/desktop/gnome/applications/main-menu/lock-down/user_modifiable_system_area"
|
|
|
|
#define USER_APPS_STORE_FILE_NAME "applications.xbel"
|
|
#define USER_DOCS_STORE_FILE_NAME "documents.xbel"
|
|
#define USER_DIRS_STORE_FILE_NAME "places.xbel"
|
|
#define SYSTEM_STORE_FILE_NAME "system-items.xbel"
|
|
#define CALC_TEMPLATE_FILE_NAME "empty.ots"
|
|
#define WRITER_TEMPLATE_FILE_NAME "empty.ott"
|
|
|
|
#define GTK_BOOKMARKS_FILE ".gtk-bookmarks"
|
|
|
|
#define TYPE_IS_RECENT(type) ((type) == BOOKMARK_STORE_RECENT_APPS || (type) == BOOKMARK_STORE_RECENT_DOCS)
|
|
|
|
typedef struct {
|
|
BookmarkStoreType type;
|
|
|
|
BookmarkItem **items;
|
|
gint n_items;
|
|
BookmarkStoreStatus status;
|
|
|
|
GBookmarkFile *store;
|
|
gboolean needs_sync;
|
|
|
|
gchar *store_path;
|
|
gchar *user_store_path;
|
|
gboolean user_modifiable;
|
|
gboolean reorderable;
|
|
const gchar *store_filename;
|
|
const gchar *lockdown_key;
|
|
|
|
GFileMonitor *store_monitor;
|
|
GFileMonitor *user_store_monitor;
|
|
guint gconf_monitor;
|
|
|
|
void (* update_path) (BookmarkAgent *);
|
|
void (* load_store) (BookmarkAgent *);
|
|
void (* save_store) (BookmarkAgent *);
|
|
void (* create_item) (BookmarkAgent *, const gchar *);
|
|
|
|
gchar *gtk_store_path;
|
|
GFileMonitor *gtk_store_monitor;
|
|
} BookmarkAgentPrivate;
|
|
|
|
#define PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), BOOKMARK_AGENT_TYPE, BookmarkAgentPrivate))
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_ITEMS,
|
|
PROP_STATUS
|
|
};
|
|
|
|
static BookmarkAgent *instances [BOOKMARK_STORE_N_TYPES];
|
|
|
|
static BookmarkAgentClass *bookmark_agent_parent_class = NULL;
|
|
|
|
static void bookmark_agent_base_init (BookmarkAgentClass *);
|
|
static void bookmark_agent_class_init (BookmarkAgentClass *);
|
|
static void bookmark_agent_init (BookmarkAgent *);
|
|
static BookmarkAgent *bookmark_agent_new (BookmarkStoreType );
|
|
|
|
static void get_property (GObject *, guint, GValue *, GParamSpec *);
|
|
static void set_property (GObject *, guint, const GValue *, GParamSpec *);
|
|
static void finalize (GObject *);
|
|
|
|
static void update_agent (BookmarkAgent *);
|
|
static void update_items (BookmarkAgent *);
|
|
static void save_store (BookmarkAgent *);
|
|
static gint get_rank (BookmarkAgent *, const gchar *);
|
|
static void set_rank (BookmarkAgent *, const gchar *, gint);
|
|
|
|
static void load_xbel_store (BookmarkAgent *);
|
|
static void load_places_store (BookmarkAgent *);
|
|
static void update_user_spec_path (BookmarkAgent *);
|
|
static void save_xbel_store (BookmarkAgent *);
|
|
static void create_app_item (BookmarkAgent *, const gchar *);
|
|
static void create_doc_item (BookmarkAgent *, const gchar *);
|
|
static void create_dir_item (BookmarkAgent *, const gchar *);
|
|
|
|
static void store_monitor_cb (GFileMonitor *, GFile *, GFile *,
|
|
GFileMonitorEvent, gpointer);
|
|
static void gconf_notify_cb (GConfClient *, guint, GConfEntry *, gpointer);
|
|
static void weak_destroy_cb (gpointer, GObject *);
|
|
|
|
static gint recent_item_mru_comp_func (gconstpointer a, gconstpointer b);
|
|
|
|
static gchar *find_package_data_file (const gchar *filename);
|
|
|
|
|
|
GType
|
|
bookmark_agent_get_type ()
|
|
{
|
|
static GType g_define_type_id = 0;
|
|
|
|
if (G_UNLIKELY (g_define_type_id == 0)) {
|
|
static const GTypeInfo info = {
|
|
sizeof (BookmarkAgentClass),
|
|
(GBaseInitFunc) bookmark_agent_base_init,
|
|
NULL,
|
|
(GClassInitFunc) bookmark_agent_class_init,
|
|
NULL, NULL,
|
|
sizeof (BookmarkAgent), 0,
|
|
(GInstanceInitFunc) bookmark_agent_init,
|
|
NULL
|
|
};
|
|
|
|
g_define_type_id = g_type_register_static (
|
|
G_TYPE_OBJECT, "BookmarkAgent", & info, 0);
|
|
}
|
|
|
|
return g_define_type_id;
|
|
}
|
|
|
|
BookmarkAgent *
|
|
bookmark_agent_get_instance (BookmarkStoreType type)
|
|
{
|
|
g_return_val_if_fail (0 <= type, NULL);
|
|
g_return_val_if_fail (type < BOOKMARK_STORE_N_TYPES, NULL);
|
|
|
|
if (! instances [type]) {
|
|
instances [type] = bookmark_agent_new (type);
|
|
g_object_weak_ref (G_OBJECT (instances [type]), weak_destroy_cb, GINT_TO_POINTER (type));
|
|
}
|
|
else
|
|
g_object_ref (G_OBJECT (instances [type]));
|
|
|
|
return instances [type];
|
|
}
|
|
|
|
gboolean
|
|
bookmark_agent_has_item (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
return g_bookmark_file_has_item (PRIVATE (this)->store, uri);
|
|
}
|
|
|
|
void
|
|
bookmark_agent_add_item (BookmarkAgent *this, const BookmarkItem *item)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
if (! item)
|
|
return;
|
|
|
|
g_return_if_fail (priv->user_modifiable);
|
|
g_return_if_fail (item->uri);
|
|
g_return_if_fail (item->mime_type);
|
|
|
|
g_bookmark_file_set_mime_type (priv->store, item->uri, item->mime_type);
|
|
|
|
if (item->mtime)
|
|
g_bookmark_file_set_modified (priv->store, item->uri, item->mtime);
|
|
|
|
if (item->title)
|
|
g_bookmark_file_set_title (priv->store, item->uri, item->title);
|
|
|
|
g_bookmark_file_add_application (priv->store, item->uri, item->app_name, item->app_exec);
|
|
|
|
set_rank (this, item->uri, g_bookmark_file_get_size (priv->store) - 1);
|
|
|
|
save_store (this);
|
|
}
|
|
|
|
void
|
|
bookmark_agent_move_item (BookmarkAgent *this, const gchar *uri, const gchar *uri_new)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
GError *error = NULL;
|
|
|
|
if (! TYPE_IS_RECENT (priv->type))
|
|
return;
|
|
|
|
gtk_recent_manager_move_item (
|
|
gtk_recent_manager_get_default (), uri, uri_new, & error);
|
|
|
|
if (error)
|
|
libslab_handle_g_error (
|
|
& error, "%s: unable to update %s with renamed file, [%s] -> [%s].",
|
|
G_STRFUNC, priv->store_path, uri, uri_new);
|
|
}
|
|
|
|
void
|
|
bookmark_agent_remove_item (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gint rank;
|
|
|
|
GError *error = NULL;
|
|
|
|
gchar **uris = NULL;
|
|
gint rank_i;
|
|
gint i;
|
|
|
|
|
|
g_return_if_fail (priv->user_modifiable);
|
|
|
|
if (! bookmark_agent_has_item (this, uri))
|
|
return;
|
|
|
|
if (TYPE_IS_RECENT (priv->type)) {
|
|
gtk_recent_manager_remove_item (
|
|
gtk_recent_manager_get_default (), uri, & error);
|
|
|
|
if (error)
|
|
libslab_handle_g_error (
|
|
& error, "%s: unable to remove [%s] from %s.",
|
|
G_STRFUNC, priv->store_path, uri);
|
|
}
|
|
else {
|
|
rank = get_rank (this, uri);
|
|
|
|
g_bookmark_file_remove_item (priv->store, uri, NULL);
|
|
|
|
if (rank >= 0) {
|
|
uris = g_bookmark_file_get_uris (priv->store, NULL);
|
|
|
|
for (i = 0; uris && uris [i]; ++i) {
|
|
rank_i = get_rank (this, uris [i]);
|
|
|
|
if (rank_i > rank)
|
|
set_rank (this, uris [i], rank_i - 1);
|
|
}
|
|
|
|
g_strfreev (uris);
|
|
}
|
|
|
|
save_store (this);
|
|
}
|
|
}
|
|
|
|
void
|
|
bookmark_agent_reorder_items (BookmarkAgent *this, const gchar **uris)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gint i;
|
|
|
|
|
|
g_return_if_fail (priv->reorderable);
|
|
|
|
for (i = 0; uris && uris [i]; ++i)
|
|
set_rank (this, uris [i], i);
|
|
|
|
save_store (this);
|
|
}
|
|
|
|
static GList *
|
|
make_items_from_bookmark_file (BookmarkAgent *this, GBookmarkFile *store)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
gchar **uris;
|
|
gint i;
|
|
GList *items_ordered;
|
|
|
|
if (!store)
|
|
return NULL;
|
|
|
|
uris = g_bookmark_file_get_uris (store, NULL);
|
|
items_ordered = NULL;
|
|
|
|
for (i = 0; uris && uris [i]; ++i) {
|
|
gboolean include;
|
|
|
|
if (priv->type == BOOKMARK_STORE_RECENT_APPS)
|
|
include = g_bookmark_file_has_group (store, uris [i], "recently-used-apps", NULL);
|
|
else
|
|
include = ! g_bookmark_file_get_is_private (store, uris [i], NULL);
|
|
|
|
if (include) {
|
|
BookmarkItem *item;
|
|
|
|
item = g_new0 (BookmarkItem, 1);
|
|
|
|
item->uri = g_strdup (uris [i]);
|
|
item->mime_type = g_bookmark_file_get_mime_type (store, uris [i], NULL);
|
|
item->mtime = g_bookmark_file_get_modified (store, uris [i], NULL);
|
|
|
|
items_ordered = g_list_prepend (items_ordered, item);
|
|
}
|
|
}
|
|
|
|
items_ordered = g_list_sort (items_ordered, recent_item_mru_comp_func);
|
|
|
|
g_strfreev (uris);
|
|
|
|
return items_ordered;
|
|
}
|
|
|
|
void
|
|
bookmark_agent_update_from_bookmark_file (BookmarkAgent *this, GBookmarkFile *store)
|
|
{
|
|
BookmarkAgentPrivate *priv;
|
|
GList *items_ordered;
|
|
GList *node;
|
|
|
|
g_return_if_fail (IS_BOOKMARK_AGENT (this));
|
|
|
|
priv = PRIVATE (this);
|
|
|
|
libslab_checkpoint ("bookmark_agent_update_from_bookmark_file(): start updating");
|
|
|
|
items_ordered = make_items_from_bookmark_file (this, store);
|
|
|
|
g_bookmark_file_free (priv->store);
|
|
priv->store = g_bookmark_file_new ();
|
|
|
|
for (node = items_ordered; node; node = node->next) {
|
|
BookmarkItem *item;
|
|
|
|
item = (BookmarkItem *) node->data;
|
|
|
|
g_bookmark_file_set_mime_type (priv->store, item->uri, item->mime_type);
|
|
g_bookmark_file_set_modified (priv->store, item->uri, item->mtime);
|
|
|
|
bookmark_item_free (item);
|
|
}
|
|
|
|
g_list_free (items_ordered);
|
|
|
|
libslab_checkpoint ("bookmark_agent_update_from_bookmark_file(): updating internal items");
|
|
update_items (this);
|
|
|
|
libslab_checkpoint ("bookmark_agent_update_from_bookmark_file(): end updating");
|
|
}
|
|
|
|
void
|
|
bookmark_item_free (BookmarkItem *item)
|
|
{
|
|
if (! item)
|
|
return;
|
|
|
|
g_free (item->uri);
|
|
g_free (item->title);
|
|
g_free (item->mime_type);
|
|
g_free (item->icon);
|
|
g_free (item->app_name);
|
|
g_free (item->app_exec);
|
|
g_free (item);
|
|
}
|
|
|
|
static void
|
|
bookmark_agent_base_init (BookmarkAgentClass *this_class)
|
|
{
|
|
gint i;
|
|
|
|
for (i = 0; i < BOOKMARK_STORE_N_TYPES; ++i)
|
|
instances [i] = NULL;
|
|
}
|
|
|
|
static void
|
|
bookmark_agent_class_init (BookmarkAgentClass *this_class)
|
|
{
|
|
GObjectClass *g_obj_class = G_OBJECT_CLASS (this_class);
|
|
|
|
GParamSpec *items_pspec;
|
|
GParamSpec *status_pspec;
|
|
|
|
|
|
g_obj_class->get_property = get_property;
|
|
g_obj_class->set_property = set_property;
|
|
g_obj_class->finalize = finalize;
|
|
|
|
items_pspec = g_param_spec_pointer (
|
|
BOOKMARK_AGENT_ITEMS_PROP, BOOKMARK_AGENT_ITEMS_PROP,
|
|
"the null-terminated list which contains the bookmark items in this store",
|
|
G_PARAM_READABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB);
|
|
|
|
status_pspec = g_param_spec_int (
|
|
BOOKMARK_AGENT_STORE_STATUS_PROP, BOOKMARK_AGENT_STORE_STATUS_PROP, "the status of the store",
|
|
BOOKMARK_STORE_DEFAULT_ONLY, BOOKMARK_STORE_USER, BOOKMARK_STORE_DEFAULT,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB);
|
|
|
|
g_object_class_install_property (g_obj_class, PROP_ITEMS, items_pspec);
|
|
g_object_class_install_property (g_obj_class, PROP_STATUS, status_pspec);
|
|
|
|
g_type_class_add_private (this_class, sizeof (BookmarkAgentPrivate));
|
|
|
|
bookmark_agent_parent_class = g_type_class_peek_parent (this_class);
|
|
}
|
|
|
|
static void
|
|
bookmark_agent_init (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
priv->type = -1;
|
|
|
|
priv->items = NULL;
|
|
priv->n_items = 0;
|
|
priv->status = BOOKMARK_STORE_ABSENT;
|
|
|
|
priv->store = NULL;
|
|
priv->needs_sync = FALSE;
|
|
|
|
priv->store_path = NULL;
|
|
priv->user_store_path = NULL;
|
|
priv->user_modifiable = FALSE;
|
|
priv->reorderable = FALSE;
|
|
priv->store_filename = NULL;
|
|
priv->lockdown_key = NULL;
|
|
|
|
priv->store_monitor = NULL;
|
|
priv->user_store_monitor = NULL;
|
|
priv->gconf_monitor = 0;
|
|
|
|
priv->update_path = NULL;
|
|
priv->load_store = NULL;
|
|
priv->save_store = NULL;
|
|
priv->create_item = NULL;
|
|
|
|
priv->gtk_store_path = NULL;
|
|
priv->gtk_store_monitor = NULL;
|
|
}
|
|
|
|
static BookmarkAgent *
|
|
bookmark_agent_new (BookmarkStoreType type)
|
|
{
|
|
BookmarkAgent *this;
|
|
BookmarkAgentPrivate *priv;
|
|
GFile *gtk_store_file;
|
|
|
|
this = g_object_new (BOOKMARK_AGENT_TYPE, NULL);
|
|
priv = PRIVATE (this);
|
|
|
|
priv->type = type;
|
|
priv->store = g_bookmark_file_new ();
|
|
|
|
switch (type) {
|
|
case BOOKMARK_STORE_USER_APPS:
|
|
priv->lockdown_key = MODIFIABLE_APPS_GCONF_KEY;
|
|
priv->store_filename = USER_APPS_STORE_FILE_NAME;
|
|
priv->create_item = create_app_item;
|
|
|
|
break;
|
|
|
|
case BOOKMARK_STORE_USER_DOCS:
|
|
priv->lockdown_key = MODIFIABLE_DOCS_GCONF_KEY;
|
|
priv->store_filename = USER_DOCS_STORE_FILE_NAME;
|
|
priv->create_item = create_doc_item;
|
|
|
|
break;
|
|
|
|
case BOOKMARK_STORE_USER_DIRS:
|
|
priv->lockdown_key = MODIFIABLE_DIRS_GCONF_KEY;
|
|
priv->store_filename = USER_DIRS_STORE_FILE_NAME;
|
|
priv->create_item = create_dir_item;
|
|
|
|
priv->user_modifiable = GPOINTER_TO_INT (libslab_get_gconf_value (priv->lockdown_key));
|
|
priv->reorderable = FALSE;
|
|
|
|
priv->load_store = load_places_store;
|
|
|
|
priv->gtk_store_path = g_build_filename (g_get_home_dir (), GTK_BOOKMARKS_FILE, NULL);
|
|
gtk_store_file = g_file_new_for_path (priv->gtk_store_path);
|
|
priv->gtk_store_monitor = g_file_monitor_file (gtk_store_file,
|
|
0, NULL, NULL);
|
|
if (priv->gtk_store_monitor) {
|
|
g_signal_connect (priv->gtk_store_monitor, "changed",
|
|
G_CALLBACK (store_monitor_cb), this);
|
|
}
|
|
|
|
g_object_unref (gtk_store_file);
|
|
|
|
break;
|
|
|
|
case BOOKMARK_STORE_RECENT_APPS:
|
|
case BOOKMARK_STORE_RECENT_DOCS:
|
|
priv->user_modifiable = TRUE;
|
|
priv->reorderable = FALSE;
|
|
|
|
priv->store_path = g_build_filename (g_get_home_dir (), ".recently-used.xbel", NULL);
|
|
|
|
break;
|
|
|
|
case BOOKMARK_STORE_SYSTEM:
|
|
priv->lockdown_key = MODIFIABLE_SYS_GCONF_KEY;
|
|
priv->store_filename = SYSTEM_STORE_FILE_NAME;
|
|
priv->create_item = create_app_item;
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (
|
|
type == BOOKMARK_STORE_USER_APPS || type == BOOKMARK_STORE_USER_DOCS ||
|
|
type == BOOKMARK_STORE_USER_DIRS || type == BOOKMARK_STORE_SYSTEM)
|
|
{
|
|
priv->user_modifiable = GPOINTER_TO_INT (libslab_get_gconf_value (priv->lockdown_key));
|
|
|
|
priv->user_store_path = g_build_filename (
|
|
g_get_user_data_dir (), PACKAGE, priv->store_filename, NULL);
|
|
|
|
priv->update_path = update_user_spec_path;
|
|
|
|
priv->gconf_monitor = libslab_gconf_notify_add (
|
|
priv->lockdown_key, gconf_notify_cb, this);
|
|
}
|
|
|
|
if (type == BOOKMARK_STORE_USER_APPS || type == BOOKMARK_STORE_USER_DOCS || type == BOOKMARK_STORE_SYSTEM) {
|
|
priv->reorderable = TRUE;
|
|
priv->load_store = load_xbel_store;
|
|
priv->save_store = save_xbel_store;
|
|
}
|
|
|
|
update_agent (this);
|
|
|
|
return this;
|
|
}
|
|
|
|
static void
|
|
get_property (GObject *g_obj, guint prop_id, GValue *value, GParamSpec *pspec)
|
|
{
|
|
BookmarkAgent *this = BOOKMARK_AGENT (g_obj);
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
|
|
switch (prop_id) {
|
|
case PROP_ITEMS:
|
|
g_value_set_pointer (value, priv->items);
|
|
break;
|
|
|
|
case PROP_STATUS:
|
|
g_value_set_int (value, priv->status);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_property (GObject *g_obj, guint prop_id, const GValue *value, GParamSpec *pspec)
|
|
{
|
|
/* no writeable properties */
|
|
}
|
|
|
|
static void
|
|
finalize (GObject *g_obj)
|
|
{
|
|
BookmarkAgent *this = BOOKMARK_AGENT (g_obj);
|
|
BookmarkAgentPrivate *priv = PRIVATE (g_obj);
|
|
|
|
gint i;
|
|
|
|
|
|
for (i = 0; priv->items && priv->items [i]; ++i)
|
|
bookmark_item_free (priv->items [i]);
|
|
|
|
g_free (priv->items);
|
|
g_free (priv->store_path);
|
|
g_free (priv->user_store_path);
|
|
g_free (priv->gtk_store_path);
|
|
|
|
if (priv->store_monitor) {
|
|
g_signal_handlers_disconnect_by_func (priv->store_monitor, store_monitor_cb, this);
|
|
g_file_monitor_cancel (priv->store_monitor);
|
|
g_object_unref (priv->store_monitor);
|
|
}
|
|
|
|
if (priv->user_store_monitor) {
|
|
g_signal_handlers_disconnect_by_func (priv->user_store_monitor, store_monitor_cb, this);
|
|
g_file_monitor_cancel (priv->user_store_monitor);
|
|
g_object_unref (priv->user_store_monitor);
|
|
}
|
|
|
|
if (priv->gtk_store_monitor) {
|
|
g_signal_handlers_disconnect_by_func (priv->gtk_store_monitor, store_monitor_cb, this);
|
|
g_file_monitor_cancel (priv->gtk_store_monitor);
|
|
g_object_unref (priv->gtk_store_monitor);
|
|
}
|
|
|
|
libslab_gconf_notify_remove (priv->gconf_monitor);
|
|
|
|
g_bookmark_file_free (priv->store);
|
|
|
|
G_OBJECT_CLASS (bookmark_agent_parent_class)->finalize (g_obj);
|
|
}
|
|
|
|
static void
|
|
update_agent (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
if (priv->update_path)
|
|
priv->update_path (this);
|
|
|
|
if (priv->load_store)
|
|
priv->load_store (this);
|
|
|
|
update_items (this);
|
|
}
|
|
|
|
static void
|
|
update_items (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar **uris = NULL;
|
|
gchar **uris_ordered = NULL;
|
|
gsize n_uris = 0;
|
|
gint rank = -1;
|
|
gint rank_corr = -1;
|
|
gboolean needs_update = FALSE;
|
|
gboolean store_corrupted = FALSE;
|
|
gchar *new_title, *old_title;
|
|
|
|
gint i;
|
|
|
|
|
|
uris = g_bookmark_file_get_uris (priv->store, & n_uris);
|
|
uris_ordered = g_new0 (gchar *, n_uris + 1);
|
|
uris_ordered [n_uris] = NULL;
|
|
|
|
for (i = 0; uris && uris [i]; ++i) {
|
|
rank = get_rank (this, uris [i]);
|
|
|
|
if (rank < 0 || rank >= n_uris)
|
|
rank = i;
|
|
|
|
if (uris_ordered [rank]) {
|
|
store_corrupted = TRUE;
|
|
rank_corr = rank;
|
|
|
|
for (rank = 0; rank < n_uris; ++rank)
|
|
if (! uris_ordered [rank])
|
|
break;
|
|
|
|
g_warning (
|
|
"store corruption [%s] - multiple uris with same rank (%d): [%s] [%s], moving latter to %d",
|
|
priv->store_path, rank_corr, uris_ordered [rank_corr], uris [i], rank);
|
|
}
|
|
|
|
set_rank (this, uris [i], rank);
|
|
|
|
uris_ordered [rank] = uris [i];
|
|
}
|
|
|
|
if (priv->n_items != n_uris)
|
|
needs_update = TRUE;
|
|
|
|
for (i = 0; ! needs_update && uris_ordered && uris_ordered [i]; ++i) {
|
|
if (priv->type == BOOKMARK_STORE_USER_DIRS) {
|
|
new_title = g_bookmark_file_get_title (priv->store, uris_ordered [i], NULL);
|
|
old_title = priv->items [i]->title;
|
|
if (!new_title && !old_title) {
|
|
if (strcmp (priv->items [i]->uri, uris_ordered [i]))
|
|
needs_update = TRUE;
|
|
}
|
|
else if ((new_title && !old_title) || (!new_title && old_title))
|
|
needs_update = TRUE;
|
|
else if (strcmp (old_title, new_title))
|
|
needs_update = TRUE;
|
|
g_free (new_title);
|
|
}
|
|
else if (strcmp (priv->items [i]->uri, uris_ordered [i]))
|
|
needs_update = TRUE;
|
|
}
|
|
|
|
if (needs_update) {
|
|
for (i = 0; priv->items && priv->items [i]; ++i)
|
|
bookmark_item_free (priv->items [i]);
|
|
|
|
g_free (priv->items);
|
|
|
|
priv->n_items = n_uris;
|
|
priv->items = g_new0 (BookmarkItem *, priv->n_items + 1);
|
|
|
|
for (i = 0; uris_ordered && uris_ordered [i]; ++i) {
|
|
priv->items [i] = g_new0 (BookmarkItem, 1);
|
|
priv->items [i]->uri = g_strdup (uris_ordered [i]);
|
|
priv->items [i]->title = g_bookmark_file_get_title (priv->store, uris_ordered [i], NULL);
|
|
priv->items [i]->mime_type = g_bookmark_file_get_mime_type (priv->store, uris_ordered [i], NULL);
|
|
priv->items [i]->mtime = g_bookmark_file_get_modified (priv->store, uris_ordered [i], NULL);
|
|
priv->items [i]->app_name = NULL;
|
|
priv->items [i]->app_exec = NULL;
|
|
|
|
g_bookmark_file_get_icon (priv->store, uris_ordered [i], & priv->items [i]->icon, NULL, NULL);
|
|
}
|
|
|
|
/* Since the bookmark store for recently-used items is updated by the caller of BookmarkAgent,
|
|
* we don't emit notifications in that case. The caller will know when to update itself.
|
|
*/
|
|
if (!TYPE_IS_RECENT (priv->type))
|
|
g_object_notify (G_OBJECT (this), BOOKMARK_AGENT_ITEMS_PROP);
|
|
}
|
|
|
|
if (store_corrupted)
|
|
save_store (this);
|
|
|
|
g_strfreev (uris);
|
|
g_free (uris_ordered);
|
|
}
|
|
|
|
static void
|
|
save_store (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar *dir;
|
|
|
|
|
|
g_return_if_fail (priv->user_modifiable);
|
|
|
|
priv->needs_sync = TRUE;
|
|
priv->update_path (this);
|
|
|
|
dir = g_path_get_dirname (priv->store_path);
|
|
g_mkdir_with_parents (dir, 0700);
|
|
g_free (dir);
|
|
|
|
priv->save_store (this);
|
|
update_items (this);
|
|
}
|
|
|
|
static gint
|
|
get_rank (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar **groups;
|
|
gint rank;
|
|
|
|
gint i;
|
|
|
|
|
|
if (! priv->reorderable)
|
|
return -1;
|
|
|
|
groups = g_bookmark_file_get_groups (priv->store, uri, NULL, NULL);
|
|
rank = -1;
|
|
|
|
for (i = 0; groups && groups [i]; ++i) {
|
|
if (g_str_has_prefix (groups [i], "rank-")) {
|
|
if (rank >= 0)
|
|
g_warning (
|
|
"store corruption - multiple ranks for same uri: [%s] [%s]",
|
|
priv->store_path, uri);
|
|
|
|
rank = atoi (& groups [i] [5]);
|
|
}
|
|
}
|
|
|
|
g_strfreev (groups);
|
|
|
|
return rank;
|
|
}
|
|
|
|
static void
|
|
set_rank (BookmarkAgent *this, const gchar *uri, gint rank)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar **groups;
|
|
gchar *group;
|
|
|
|
gint i;
|
|
|
|
|
|
if (! (priv->reorderable && bookmark_agent_has_item (this, uri)))
|
|
return;
|
|
|
|
groups = g_bookmark_file_get_groups (priv->store, uri, NULL, NULL);
|
|
|
|
for (i = 0; groups && groups [i]; ++i)
|
|
if (g_str_has_prefix (groups [i], "rank-"))
|
|
g_bookmark_file_remove_group (priv->store, uri, groups [i], NULL);
|
|
|
|
g_strfreev (groups);
|
|
|
|
group = g_strdup_printf ("rank-%d", rank);
|
|
g_bookmark_file_add_group (priv->store, uri, group);
|
|
g_free (group);
|
|
}
|
|
|
|
static void
|
|
load_xbel_store (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar **uris = NULL;
|
|
|
|
GError *error = NULL;
|
|
|
|
gint i;
|
|
gboolean success;
|
|
|
|
if (!priv->store_path)
|
|
success = FALSE;
|
|
else {
|
|
libslab_checkpoint ("load_xbel_store(): start loading %s", priv->store_path);
|
|
success = g_bookmark_file_load_from_file (priv->store, priv->store_path, & error);
|
|
libslab_checkpoint ("load_xbel_store(): end loading %s", priv->store_path);
|
|
}
|
|
|
|
if (!success) {
|
|
g_bookmark_file_free (priv->store);
|
|
priv->store = g_bookmark_file_new ();
|
|
|
|
libslab_handle_g_error (
|
|
& error, "%s: couldn't load bookmark file [%s]\n",
|
|
G_STRFUNC, priv->store_path ? priv->store_path : "NULL");
|
|
|
|
return;
|
|
}
|
|
|
|
libslab_checkpoint ("load_xbel_store(): start creating items from %s", priv->store_path);
|
|
|
|
uris = g_bookmark_file_get_uris (priv->store, NULL);
|
|
|
|
for (i = 0; uris && uris [i]; ++i)
|
|
priv->create_item (this, uris [i]);
|
|
|
|
g_strfreev (uris);
|
|
|
|
libslab_checkpoint ("load_xbel_store(): end creating items from %s", priv->store_path);
|
|
}
|
|
|
|
static void
|
|
load_places_store (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar **uris;
|
|
gchar **groups;
|
|
gchar **bookmarks = NULL;
|
|
|
|
gchar *buf, *label, *uri;
|
|
|
|
gint i, j, bookmark_len;
|
|
|
|
load_xbel_store (this);
|
|
|
|
uris = g_bookmark_file_get_uris (priv->store, NULL);
|
|
|
|
for (i = 0; uris && uris [i]; ++i) {
|
|
groups = g_bookmark_file_get_groups (priv->store, uris [i], NULL, NULL);
|
|
|
|
for (j = 0; groups && groups [j]; ++j) {
|
|
if (! strcmp (groups [j], "gtk-bookmarks")) {
|
|
g_bookmark_file_remove_item (priv->store, uris [i], NULL);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_strfreev (groups);
|
|
}
|
|
|
|
g_strfreev (uris);
|
|
|
|
g_file_get_contents (priv->gtk_store_path, & buf, NULL, NULL);
|
|
|
|
if (buf) {
|
|
bookmarks = g_strsplit (buf, "\n", -1);
|
|
g_free (buf);
|
|
}
|
|
|
|
for (i = 0; bookmarks && bookmarks [i]; ++i) {
|
|
bookmark_len = strlen (bookmarks [i]);
|
|
if (bookmark_len > 0) {
|
|
label = strstr (bookmarks[i], " ");
|
|
if (label != NULL)
|
|
uri = g_strndup (bookmarks [i], bookmark_len - strlen (label));
|
|
else
|
|
uri = bookmarks [i];
|
|
g_bookmark_file_add_group (priv->store, uri, "gtk-bookmarks");
|
|
priv->create_item (this, uri);
|
|
if (label != NULL) {
|
|
label++;
|
|
if (strlen (label) > 0)
|
|
g_bookmark_file_set_title (priv->store, uri, label);
|
|
g_free (uri);
|
|
}
|
|
}
|
|
}
|
|
|
|
g_strfreev (bookmarks);
|
|
}
|
|
|
|
static gchar *
|
|
find_package_data_file (const gchar *filename)
|
|
{
|
|
const gchar * const *dirs = NULL;
|
|
gchar *path = NULL;
|
|
gint i;
|
|
|
|
|
|
dirs = g_get_system_data_dirs ();
|
|
|
|
for (i = 0; ! path && dirs && dirs [i]; ++i) {
|
|
path = g_build_filename (dirs [i], PACKAGE, filename, NULL);
|
|
|
|
if (! g_file_test (path, G_FILE_TEST_EXISTS)) {
|
|
g_free (path);
|
|
path = NULL;
|
|
}
|
|
}
|
|
|
|
return path;
|
|
}
|
|
|
|
static void
|
|
update_user_spec_path (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gboolean use_user_path;
|
|
gchar *path = NULL;
|
|
|
|
BookmarkStoreStatus status;
|
|
|
|
use_user_path = priv->user_modifiable &&
|
|
(priv->needs_sync || g_file_test (priv->user_store_path, G_FILE_TEST_EXISTS));
|
|
|
|
if (use_user_path)
|
|
path = g_strdup (priv->user_store_path);
|
|
else
|
|
path = find_package_data_file (priv->store_filename);
|
|
|
|
if (use_user_path)
|
|
status = BOOKMARK_STORE_USER;
|
|
else if (path && priv->user_modifiable)
|
|
status = BOOKMARK_STORE_DEFAULT;
|
|
else if (path)
|
|
status = BOOKMARK_STORE_DEFAULT_ONLY;
|
|
else
|
|
status = BOOKMARK_STORE_ABSENT;
|
|
|
|
if (priv->status != status) {
|
|
priv->status = status;
|
|
g_object_notify (G_OBJECT (this), BOOKMARK_AGENT_STORE_STATUS_PROP);
|
|
|
|
if (priv->user_store_monitor) {
|
|
g_file_monitor_cancel (priv->user_store_monitor);
|
|
g_object_unref (priv->user_store_monitor);
|
|
priv->user_store_monitor = NULL;
|
|
}
|
|
|
|
if (priv->status == BOOKMARK_STORE_DEFAULT) {
|
|
GFile *user_store_file;
|
|
|
|
user_store_file = g_file_new_for_path (priv->user_store_path);
|
|
priv->user_store_monitor = g_file_monitor_file (user_store_file,
|
|
0, NULL, NULL);
|
|
if (priv->user_store_monitor) {
|
|
g_signal_connect (priv->user_store_monitor, "changed",
|
|
G_CALLBACK (store_monitor_cb), this);
|
|
}
|
|
|
|
g_object_unref (user_store_file);
|
|
}
|
|
}
|
|
|
|
if (libslab_strcmp (priv->store_path, path)) {
|
|
g_free (priv->store_path);
|
|
priv->store_path = path;
|
|
|
|
if (priv->store_monitor) {
|
|
g_file_monitor_cancel (priv->store_monitor);
|
|
g_object_unref (priv->store_monitor);
|
|
}
|
|
|
|
if (priv->store_path) {
|
|
GFile *store_file;
|
|
|
|
store_file = g_file_new_for_path (priv->store_path);
|
|
priv->store_monitor = g_file_monitor_file (store_file,
|
|
0, NULL, NULL);
|
|
if (priv->store_monitor) {
|
|
g_signal_connect (priv->store_monitor, "changed",
|
|
G_CALLBACK (store_monitor_cb), this);
|
|
}
|
|
|
|
g_object_unref (store_file);
|
|
}
|
|
}
|
|
else
|
|
g_free (path);
|
|
}
|
|
|
|
static void
|
|
save_xbel_store (BookmarkAgent *this)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
if (! g_bookmark_file_to_file (priv->store, priv->store_path, & error))
|
|
libslab_handle_g_error (
|
|
& error, "%s: couldn't save bookmark file [%s]\n", G_STRFUNC, priv->store_path);
|
|
}
|
|
|
|
static void
|
|
create_app_item (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
GnomeDesktopItem *ditem;
|
|
gchar *uri_new = NULL;
|
|
|
|
ditem = libslab_gnome_desktop_item_new_from_unknown_id (uri);
|
|
|
|
if (ditem) {
|
|
uri_new = g_strdup (gnome_desktop_item_get_location (ditem));
|
|
gnome_desktop_item_unref (ditem);
|
|
}
|
|
|
|
if (! uri_new)
|
|
return;
|
|
|
|
if (libslab_strcmp (uri, uri_new))
|
|
g_bookmark_file_move_item (priv->store, uri, uri_new, NULL);
|
|
|
|
g_free (uri_new);
|
|
}
|
|
|
|
static void
|
|
create_doc_item (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar *uri_new = NULL;
|
|
gchar *path;
|
|
gchar *dir;
|
|
gchar *file;
|
|
gchar *template = NULL;
|
|
gsize length;
|
|
gchar *contents;
|
|
|
|
|
|
if (! (strcmp (uri, "BLANK_SPREADSHEET") && strcmp (uri, "BLANK_DOCUMENT"))) {
|
|
dir = g_build_filename (g_get_home_dir (), "Documents", NULL);
|
|
|
|
if (! strcmp (uri, "BLANK_SPREADSHEET")) {
|
|
g_bookmark_file_set_title (priv->store, uri, "BLANK_SPREADSHEET");
|
|
file = g_strconcat (_("New Spreadsheet"), ".ots", NULL);
|
|
template = find_package_data_file (CALC_TEMPLATE_FILE_NAME);
|
|
}
|
|
else {
|
|
g_bookmark_file_set_title (priv->store, uri, "BLANK_DOCUMENT");
|
|
file = g_strconcat (_("New Document"), ".ott", NULL);
|
|
template = find_package_data_file (WRITER_TEMPLATE_FILE_NAME);
|
|
}
|
|
|
|
path = g_build_filename (dir, file, NULL);
|
|
|
|
if (! g_file_test (path, G_FILE_TEST_EXISTS)) {
|
|
g_mkdir_with_parents (dir, 0700);
|
|
|
|
if (template != NULL) {
|
|
if (g_file_get_contents (template, & contents, & length, NULL))
|
|
g_file_set_contents (path, contents, length, NULL);
|
|
|
|
g_free (contents);
|
|
}
|
|
else
|
|
fclose (g_fopen (path, "w"));
|
|
}
|
|
|
|
uri_new = g_filename_to_uri (path, NULL, NULL);
|
|
|
|
g_free (dir);
|
|
g_free (file);
|
|
g_free (path);
|
|
g_free (template);
|
|
}
|
|
|
|
if (! uri_new)
|
|
return;
|
|
|
|
if (libslab_strcmp (uri, uri_new))
|
|
g_bookmark_file_move_item (priv->store, uri, uri_new, NULL);
|
|
|
|
g_free (uri_new);
|
|
}
|
|
|
|
static void
|
|
create_dir_item (BookmarkAgent *this, const gchar *uri)
|
|
{
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gchar *uri_new = NULL;
|
|
gchar *path;
|
|
gchar *name = NULL;
|
|
gchar *icon = NULL;
|
|
|
|
gchar *buf;
|
|
gchar *tag_open_ptr = NULL;
|
|
gchar *tag_close_ptr = NULL;
|
|
gchar *search_string = NULL;
|
|
|
|
if (! strcmp (uri, "HOME")) {
|
|
uri_new = g_filename_to_uri (g_get_home_dir (), NULL, NULL);
|
|
name = _("Home");
|
|
icon = "gnome-fs-home";
|
|
}
|
|
else if (! strcmp (uri, "DOCUMENTS")) {
|
|
path = g_build_filename (g_get_home_dir (), "Documents", NULL);
|
|
name = _("Documents");
|
|
uri_new = g_filename_to_uri (path, NULL, NULL);
|
|
g_free (path);
|
|
}
|
|
else if (! strcmp (uri, "DESKTOP")) {
|
|
path = g_build_filename (g_get_home_dir (), "Desktop", NULL);
|
|
name = _("Desktop");
|
|
uri_new = g_filename_to_uri (path, NULL, NULL);
|
|
icon = "gnome-fs-desktop";
|
|
g_free (path);
|
|
}
|
|
else if (! strcmp (uri, "file:///")) {
|
|
icon = "drive-harddisk";
|
|
name = _("File System");
|
|
}
|
|
else if (! strcmp (uri, "network:")) {
|
|
icon = "network-workgroup";
|
|
name = _("Network Servers");
|
|
}
|
|
else if (g_str_has_prefix (uri, "x-nautilus-search")) {
|
|
icon = "system-search";
|
|
|
|
path = g_build_filename (g_get_home_dir (), ".nautilus", "searches", & uri [21], NULL);
|
|
|
|
if (g_file_test (path, G_FILE_TEST_EXISTS)) {
|
|
g_file_get_contents (path, & buf, NULL, NULL);
|
|
|
|
if (buf) {
|
|
tag_open_ptr = strstr (buf, "<text>");
|
|
tag_close_ptr = strstr (buf, "</text>");
|
|
}
|
|
|
|
if (tag_open_ptr && tag_close_ptr) {
|
|
tag_close_ptr [0] = '\0';
|
|
|
|
search_string = g_strdup_printf ("\"%s\"", & tag_open_ptr [6]);
|
|
|
|
tag_close_ptr [0] = 'a';
|
|
}
|
|
|
|
g_free (buf);
|
|
}
|
|
|
|
if (search_string)
|
|
name = search_string;
|
|
else
|
|
name = _("Search");
|
|
|
|
g_free (path);
|
|
}
|
|
|
|
if (icon)
|
|
g_bookmark_file_set_icon (priv->store, uri, icon, "image/png");
|
|
|
|
if (name)
|
|
g_bookmark_file_set_title (priv->store, uri, name);
|
|
|
|
if (uri_new && libslab_strcmp (uri, uri_new))
|
|
g_bookmark_file_move_item (priv->store, uri, uri_new, NULL);
|
|
|
|
g_free (uri_new);
|
|
}
|
|
|
|
static void
|
|
store_monitor_cb (GFileMonitor *mon, GFile *f1, GFile *f2,
|
|
GFileMonitorEvent event_type, gpointer user_data)
|
|
{
|
|
update_agent (BOOKMARK_AGENT (user_data));
|
|
}
|
|
|
|
static void
|
|
gconf_notify_cb (GConfClient *client, guint conn_id,
|
|
GConfEntry *entry, gpointer user_data)
|
|
{
|
|
BookmarkAgent *this = BOOKMARK_AGENT (user_data);
|
|
BookmarkAgentPrivate *priv = PRIVATE (this);
|
|
|
|
gboolean user_modifiable;
|
|
|
|
|
|
user_modifiable = GPOINTER_TO_INT (libslab_get_gconf_value (priv->lockdown_key));
|
|
|
|
if (priv->user_modifiable != user_modifiable) {
|
|
priv->user_modifiable = user_modifiable;
|
|
update_agent (this);
|
|
}
|
|
}
|
|
|
|
static void
|
|
weak_destroy_cb (gpointer data, GObject *g_obj)
|
|
{
|
|
instances [GPOINTER_TO_INT (data)] = NULL;
|
|
}
|
|
|
|
static gint
|
|
recent_item_mru_comp_func (gconstpointer a, gconstpointer b)
|
|
{
|
|
return ((BookmarkItem *) b)->mtime - ((BookmarkItem *) a)->mtime;
|
|
}
|