Conflicts: a11y/widgets/ea-combo-button.c a11y/widgets/ea-combo-button.h addressbook/gui/component/addressbook-component.c addressbook/gui/component/addressbook-component.h addressbook/gui/component/addressbook-view.c addressbook/gui/component/addressbook-view.h addressbook/gui/component/component-factory.c addressbook/gui/widgets/e-addressbook-view.c addressbook/gui/widgets/eab-contact-display.c addressbook/gui/widgets/eab-gui-util.h addressbook/gui/widgets/eab-menu.c addressbook/gui/widgets/eab-menu.h addressbook/gui/widgets/eab-popup-control.c addressbook/gui/widgets/eab-popup-control.h addressbook/gui/widgets/eab-popup.c addressbook/gui/widgets/eab-popup.h calendar/gui/cal-search-bar.c calendar/gui/calendar-commands.c calendar/gui/calendar-component.c calendar/gui/comp-editor-factory.c calendar/gui/comp-editor-factory.h calendar/gui/control-factory.c calendar/gui/dialogs/comp-editor.c calendar/gui/e-cal-component-memo-preview.c calendar/gui/e-cal-component-memo-preview.h calendar/gui/e-calendar-table.c calendar/gui/e-memo-table.c calendar/gui/e-memos.c calendar/gui/e-tasks.c calendar/gui/gnome-cal.c calendar/gui/gnome-cal.h calendar/gui/itip-bonobo-control.c calendar/gui/itip-bonobo-control.h calendar/gui/main.c calendar/gui/memos-component.c calendar/gui/memos-control.c calendar/gui/memos-control.h calendar/gui/migration.c calendar/gui/migration.h calendar/gui/tasks-component.c calendar/gui/tasks-control.c calendar/importers/main.c composer/Makefile.am composer/e-composer-header-table.c composer/e-composer-header.c composer/e-composer-header.h composer/e-composer-name-header.c composer/e-composer-private.c composer/e-composer-text-header.c composer/e-msg-composer.c composer/e-msg-composer.h e-util/e-corba-utils.h e-util/e-logger.c e-util/e-logger.h e-util/e-util-labels.c e-util/e-util-labels.h em-format/em-format.c mail/Makefile.am mail/e-mail-shell-migrate.c mail/em-account-editor.c mail/em-account-editor.h mail/em-composer-prefs.c mail/em-composer-utils.c mail/em-composer-utils.h mail/em-folder-browser.c mail/em-folder-tree-model.c mail/em-folder-tree.c mail/em-folder-tree.h mail/em-folder-utils.c mail/em-folder-utils.h mail/em-folder-view.c mail/em-format-html-display.c mail/em-format-html.c mail/em-mailer-prefs.c mail/em-mailer-prefs.h mail/em-message-browser.c mail/em-message-browser.h mail/em-network-prefs.h mail/em-popup.c mail/em-utils.c mail/importers/Makefile.am mail/mail-component-factory.c mail/mail-component.c mail/mail-config-factory.c mail/mail-config-factory.h mail/mail-config.c mail/mail-dialogs.glade mail/mail-types.h plugins/calendar-weather/calendar-weather.c plugins/mail-account-disable/mail-account-disable.c plugins/select-one-source/select-one-source.c po/POTFILES.in shell/e-component-registry.c shell/e-component-registry.h shell/e-component-view.c shell/e-component-view.h shell/e-corba-config-page.c shell/e-corba-config-page.h shell/e-shell-constants.h shell/e-shell-settings-dialog.c shell/e-shell-settings-dialog.h shell/e-shell-window-commands.c shell/e-shell-window.c shell/e-shell.h shell/e-sidebar.c shell/e-sidebar.h shell/e-user-creatable-items-handler.c shell/e-user-creatable-items-handler.h shell/es-menu.c shell/es-menu.h shell/evolution-component.h shell/evolution-config-control.c shell/evolution-config-control.h shell/evolution-listener.c shell/evolution-listener.h shell/evolution-shell-component-utils.c shell/evolution-shell-component-utils.h shell/importer/evolution-importer-client.c shell/importer/evolution-importer-client.h shell/importer/evolution-importer-listener.c shell/importer/evolution-importer-listener.h shell/importer/evolution-importer.c shell/importer/evolution-importer.h shell/importer/evolution-intelligent-importer.c shell/importer/evolution-intelligent-importer.h shell/importer/intelligent.c shell/main.c shell/test/evolution-test-component.c shell/test/evolution-test-component.h widgets/menus/gal-view-instance.c widgets/menus/gal-view-menus.c widgets/menus/gal-view-menus.h widgets/misc/Makefile.am widgets/misc/e-activity-handler.c widgets/misc/e-activity-handler.h widgets/misc/e-charset-picker.c widgets/misc/e-combo-button.c widgets/misc/e-combo-button.h widgets/misc/e-config-page.h widgets/misc/e-dropdown-button.c widgets/misc/e-dropdown-button.h widgets/misc/e-filter-bar.c widgets/misc/e-info-label.c widgets/misc/e-info-label.h widgets/misc/e-multi-config-dialog.c widgets/misc/e-multi-config-dialog.h widgets/misc/e-search-bar.c widgets/misc/e-search-bar.h widgets/misc/e-task-bar.c widgets/misc/e-task-bar.h widgets/misc/e-task-widget.c widgets/misc/e-task-widget.h widgets/misc/test-dropdown-button.c widgets/misc/test-error.c widgets/misc/test-info-label.c widgets/table/e-table-example-1.c
459 lines
12 KiB
C
459 lines
12 KiB
C
/*
|
|
* e-shell-backend.c
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) version 3.
|
|
*
|
|
* 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
#include "e-shell-backend.h"
|
|
|
|
#include <errno.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
#include "e-util/e-util.h"
|
|
|
|
#include "e-shell.h"
|
|
#include "e-shell-view.h"
|
|
|
|
#define E_SHELL_BACKEND_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE \
|
|
((obj), E_TYPE_SHELL_BACKEND, EShellBackendPrivate))
|
|
|
|
struct _EShellBackendPrivate {
|
|
|
|
gpointer shell; /* weak pointer */
|
|
|
|
/* We keep a reference to corresponding EShellView subclass
|
|
* since it keeps a reference back to us. This ensures the
|
|
* subclass is not finalized before we are, otherwise it
|
|
* would leak its EShellBackend reference. */
|
|
EShellViewClass *shell_view_class;
|
|
|
|
gchar *config_dir;
|
|
gchar *data_dir;
|
|
|
|
guint started : 1;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_SHELL
|
|
};
|
|
|
|
enum {
|
|
ACTIVITY_ADDED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static gpointer parent_class;
|
|
static guint signals[LAST_SIGNAL];
|
|
|
|
static void
|
|
shell_backend_set_shell (EShellBackend *shell_backend,
|
|
EShell *shell)
|
|
{
|
|
g_return_if_fail (shell_backend->priv->shell == NULL);
|
|
|
|
shell_backend->priv->shell = shell;
|
|
|
|
g_object_add_weak_pointer (
|
|
G_OBJECT (shell_backend),
|
|
&shell_backend->priv->shell);
|
|
}
|
|
|
|
static void
|
|
shell_backend_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_SHELL:
|
|
shell_backend_set_shell (
|
|
E_SHELL_BACKEND (object),
|
|
g_value_get_object (value));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
shell_backend_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_SHELL:
|
|
g_value_set_object (
|
|
value, e_shell_backend_get_shell (
|
|
E_SHELL_BACKEND (object)));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
shell_backend_dispose (GObject *object)
|
|
{
|
|
EShellBackendPrivate *priv;
|
|
|
|
priv = E_SHELL_BACKEND_GET_PRIVATE (object);
|
|
|
|
if (priv->shell_view_class != NULL) {
|
|
g_type_class_unref (priv->shell_view_class);
|
|
priv->shell_view_class = NULL;
|
|
}
|
|
|
|
/* Chain up to parent's dispose() method. */
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
shell_backend_finalize (GObject *object)
|
|
{
|
|
EShellBackendPrivate *priv;
|
|
|
|
priv = E_SHELL_BACKEND_GET_PRIVATE (object);
|
|
|
|
g_free (priv->data_dir);
|
|
|
|
/* Chain up to parent's finalize() method. */
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
shell_backend_class_init (EShellBackendClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
|
|
parent_class = g_type_class_peek_parent (class);
|
|
g_type_class_add_private (class, sizeof (EShellBackendPrivate));
|
|
|
|
object_class = G_OBJECT_CLASS (class);
|
|
object_class->set_property = shell_backend_set_property;
|
|
object_class->get_property = shell_backend_get_property;
|
|
object_class->dispose = shell_backend_dispose;
|
|
object_class->finalize = shell_backend_finalize;
|
|
|
|
/**
|
|
* EShellBackend:shell
|
|
*
|
|
* The #EShell singleton.
|
|
**/
|
|
g_object_class_install_property (
|
|
object_class,
|
|
PROP_SHELL,
|
|
g_param_spec_object (
|
|
"shell",
|
|
_("Shell"),
|
|
_("The EShell singleton"),
|
|
E_TYPE_SHELL,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
/**
|
|
* EShellBackend::activity-added
|
|
* @shell_backend: the #EShellBackend that emitted the signal
|
|
* @activity: an #EActivity
|
|
*
|
|
* Broadcasts a newly added activity.
|
|
**/
|
|
signals[ACTIVITY_ADDED] = g_signal_new (
|
|
"activity-added",
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_LAST,
|
|
0, NULL, NULL,
|
|
g_cclosure_marshal_VOID__OBJECT,
|
|
G_TYPE_NONE, 1,
|
|
E_TYPE_ACTIVITY);
|
|
}
|
|
|
|
static void
|
|
shell_backend_init (EShellBackend *shell_backend,
|
|
EShellBackendClass *class)
|
|
{
|
|
EShellViewClass *shell_view_class;
|
|
gchar *dirname;
|
|
|
|
shell_backend->priv = E_SHELL_BACKEND_GET_PRIVATE (shell_backend);
|
|
|
|
/* Install a reference to ourselves in the corresponding
|
|
* EShellViewClass structure, */
|
|
shell_view_class = g_type_class_ref (class->shell_view_type);
|
|
shell_view_class->shell_backend = g_object_ref (shell_backend);
|
|
shell_backend->priv->shell_view_class = shell_view_class;
|
|
|
|
/* Determine the user data directory for this backend. */
|
|
shell_backend->priv->data_dir = g_build_filename (
|
|
e_get_user_data_dir (), class->name, NULL);
|
|
|
|
/* Determine the user configuration directory for this backend. */
|
|
shell_backend->priv->config_dir = g_build_filename (
|
|
shell_backend->priv->data_dir, "config", NULL);
|
|
|
|
/* Create the user configuration directory for this backend,
|
|
* which should also create the user data directory. */
|
|
dirname = shell_backend->priv->config_dir;
|
|
if (g_mkdir_with_parents (dirname, 0777) != 0)
|
|
g_critical (
|
|
"Cannot create directory %s: %s",
|
|
dirname, g_strerror (errno));
|
|
}
|
|
|
|
GType
|
|
e_shell_backend_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (G_UNLIKELY (type == 0)) {
|
|
const GTypeInfo type_info = {
|
|
sizeof (EShellBackendClass),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) shell_backend_class_init,
|
|
(GClassFinalizeFunc) NULL,
|
|
NULL, /* class_data */
|
|
sizeof (EShellBackend),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) shell_backend_init,
|
|
NULL /* value_table */
|
|
};
|
|
|
|
type = g_type_register_static (
|
|
G_TYPE_OBJECT, "EShellBackend", &type_info, 0);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_compare:
|
|
* @shell_backend_a: an #EShellBackend
|
|
* @shell_backend_b: an #EShellBackend
|
|
*
|
|
* Using the <structfield>sort_order</structfield> field from both backends'
|
|
* #EShellBackendClass, compares @shell_backend_a with @shell_mobule_b and
|
|
* returns -1, 0 or +1 if @shell_backend_a is found to be less than, equal
|
|
* to or greater than @shell_backend_b, respectively.
|
|
*
|
|
* Returns: -1, 0 or +1, for a less than, equal to or greater than result
|
|
**/
|
|
gint
|
|
e_shell_backend_compare (EShellBackend *shell_backend_a,
|
|
EShellBackend *shell_backend_b)
|
|
{
|
|
gint a = E_SHELL_BACKEND_GET_CLASS (shell_backend_a)->sort_order;
|
|
gint b = E_SHELL_BACKEND_GET_CLASS (shell_backend_b)->sort_order;
|
|
|
|
return (a < b) ? -1 : (a > b);
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_get_config_dir:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Returns the absolute path to the configuration directory for
|
|
* @shell_backend. The string is owned by @shell_backend and should
|
|
* not be modified or freed.
|
|
*
|
|
* Returns: the backend's configuration directory
|
|
**/
|
|
const gchar *
|
|
e_shell_backend_get_config_dir (EShellBackend *shell_backend)
|
|
{
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), NULL);
|
|
g_return_val_if_fail (shell_backend->priv->config_dir != NULL, NULL);
|
|
|
|
return shell_backend->priv->config_dir;
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_get_data_dir:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Returns the absolute path to the data directory for @shell_backend.
|
|
* The string is owned by @shell_backend and should not be modified or
|
|
* freed.
|
|
*
|
|
* Returns: the backend's data directory
|
|
**/
|
|
const gchar *
|
|
e_shell_backend_get_data_dir (EShellBackend *shell_backend)
|
|
{
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), NULL);
|
|
g_return_val_if_fail (shell_backend->priv->data_dir != NULL, NULL);
|
|
|
|
return shell_backend->priv->data_dir;
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_get_shell:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Returns the #EShell singleton.
|
|
*
|
|
* Returns: the #EShell
|
|
**/
|
|
EShell *
|
|
e_shell_backend_get_shell (EShellBackend *shell_backend)
|
|
{
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), NULL);
|
|
|
|
return E_SHELL (shell_backend->priv->shell);
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_add_activity:
|
|
* @shell_backend: an #EShellBackend
|
|
* @activity: an #EActivity
|
|
*
|
|
* Emits an #EShellBackend::activity-added signal.
|
|
**/
|
|
void
|
|
e_shell_backend_add_activity (EShellBackend *shell_backend,
|
|
EActivity *activity)
|
|
{
|
|
g_return_if_fail (E_IS_SHELL_BACKEND (shell_backend));
|
|
g_return_if_fail (E_IS_ACTIVITY (activity));
|
|
|
|
g_signal_emit (shell_backend, signals[ACTIVITY_ADDED], 0, activity);
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_start:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Tells the @shell_backend to begin loading data or running background
|
|
* tasks which may consume significant resources. This gets called in
|
|
* reponse to the user switching to the corresponding #EShellView for
|
|
* the first time. The function is idempotent for each @shell_backend.
|
|
**/
|
|
void
|
|
e_shell_backend_start (EShellBackend *shell_backend)
|
|
{
|
|
EShellBackendClass *class;
|
|
|
|
g_return_if_fail (E_IS_SHELL_BACKEND (shell_backend));
|
|
|
|
if (shell_backend->priv->started)
|
|
return;
|
|
|
|
class = E_SHELL_BACKEND_GET_CLASS (shell_backend);
|
|
|
|
if (class->start != NULL)
|
|
class->start (shell_backend);
|
|
|
|
shell_backend->priv->started = TRUE;
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_is_busy:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Returns %TRUE if @shell_backend is busy and cannot be shutdown at
|
|
* present. Each backend must define what "busy" means to them and
|
|
* determine an appropriate response.
|
|
*
|
|
* XXX This function is likely to change or disappear. I'm toying with
|
|
* the idea of just having it check whether there are any unfinished
|
|
* #EActivity<!-- -->'s left, so we have a consistent and easily
|
|
* testable definition of what "busy" means.
|
|
*
|
|
* Returns: %TRUE if the backend is busy
|
|
**/
|
|
gboolean
|
|
e_shell_backend_is_busy (EShellBackend *shell_backend)
|
|
{
|
|
EShellBackendClass *class;
|
|
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), FALSE);
|
|
|
|
class = E_SHELL_BACKEND_GET_CLASS (shell_backend);
|
|
|
|
if (class->is_busy == NULL)
|
|
return FALSE;
|
|
|
|
return class->is_busy (shell_backend);
|
|
}
|
|
|
|
/**
|
|
* e_shell_backend_shutdown:
|
|
* @shell_backend: an #EShellBackend
|
|
*
|
|
* Alerts @shell_backend to begin shutdown procedures. If the backend is
|
|
* busy and cannot immediately shut down, the function returns %FALSE.
|
|
* A %TRUE response implies @shell_backend has successfully shut down.
|
|
*
|
|
* XXX This function is likely to change or disappear. I'm toying with
|
|
* the idea of just having it check whether there are any unfinished
|
|
* #EActivity<!-- -->'s left, so we have a consistent and easily
|
|
* testable definition of what "busy" means.
|
|
*
|
|
* Returns: %TRUE if the backend has shut down, %FALSE if the backend is
|
|
* busy and cannot immediately shut down
|
|
*/
|
|
gboolean
|
|
e_shell_backend_shutdown (EShellBackend *shell_backend)
|
|
{
|
|
EShellBackendClass *class;
|
|
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), TRUE);
|
|
|
|
class = E_SHELL_BACKEND_GET_CLASS (shell_backend);
|
|
|
|
if (class->shutdown == NULL)
|
|
return TRUE;
|
|
|
|
return class->shutdown (shell_backend);
|
|
}
|
|
|
|
/**
|
|
* e_shell_migrate:
|
|
* @shell_backend: an #EShellBackend
|
|
* @major: major part of version to migrate from
|
|
* @minor: minor part of version to migrate from
|
|
* @micro: micro part of version to migrate from
|
|
* @error: return location for a #GError, or %NULL
|
|
*
|
|
* Attempts to migrate data and settings from version %major.%minor.%micro.
|
|
* Returns %TRUE if the migration was successful or if no action was
|
|
* necessary. Returns %FALSE and sets %error if the migration failed.
|
|
*
|
|
* Returns: %TRUE if successful, %FALSE otherwise
|
|
**/
|
|
gboolean
|
|
e_shell_backend_migrate (EShellBackend *shell_backend,
|
|
gint major,
|
|
gint minor,
|
|
gint micro,
|
|
GError **error)
|
|
{
|
|
EShellBackendClass *class;
|
|
|
|
g_return_val_if_fail (E_IS_SHELL_BACKEND (shell_backend), TRUE);
|
|
|
|
class = E_SHELL_BACKEND_GET_CLASS (shell_backend);
|
|
|
|
if (class->migrate == NULL)
|
|
return TRUE;
|
|
|
|
return class->migrate (shell_backend, major, minor, micro, error);
|
|
}
|