/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Copyright © 2000-2003 Marco Pesenti Gritti
* Copyright © 2011 Igalia S.L.
*
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include <config.h>
#include <glib.h>
#include <glib/gi18n.h>
#include <glib/gstdio.h>
#include <gtk/gtk.h>
#include "ephy-adblock-manager.h"
#include "ephy-debug.h"
#include "ephy-download.h"
#include "ephy-embed-shell.h"
#include "ephy-embed-single.h"
#include "ephy-encodings.h"
#include "ephy-favicon-cache.h"
#include "ephy-file-helpers.h"
#include "ephy-history.h"
#include "ephy-print-utils.h"
#define PAGE_SETUP_FILENAME "page-setup-gtk.ini"
#define PRINT_SETTINGS_FILENAME "print-settings.ini"
#define LEGACY_PAGE_SETUP_FILENAME "page-setup.ini"
#define EPHY_EMBED_SHELL_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_EMBED_SHELL, EphyEmbedShellPrivate))
#define ENABLE_MIGRATION
struct _EphyEmbedShellPrivate
{
EphyHistory *global_history;
GList *downloads;
EphyFaviconCache *favicon_cache;
EphyEmbedSingle *embed_single;
EphyEncodings *encodings;
EphyAdBlockManager *adblock_manager;
GtkPageSetup *page_setup;
GtkPrintSettings *print_settings;
guint object_count;
gboolean private_instance;
guint single_initialised : 1;
};
enum
{
DOWNLOAD_ADDED,
DOWNLOAD_REMOVED,
PREPARE_CLOSE,
QUIT,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL];
enum
{
PROP_0,
PROP_PRIVATE_INSTANCE,
N_PROPERTIES
};
static GParamSpec *object_properties[N_PROPERTIES] = { NULL, };
static void ephy_embed_shell_class_init (EphyEmbedShellClass *klass);
static void ephy_embed_shell_init (EphyEmbedShell *shell);
EphyEmbedShell *embed_shell = NULL;
G_DEFINE_TYPE (EphyEmbedShell, ephy_embed_shell, GTK_TYPE_APPLICATION)
static void
ephy_embed_shell_dispose (GObject *object)
{
EphyEmbedShell *shell = EPHY_EMBED_SHELL (object);
EphyEmbedShellPrivate *priv = shell->priv;
if (priv->favicon_cache != NULL)
{
LOG ("Unref favicon cache");
g_object_unref (priv->favicon_cache);
priv->favicon_cache = NULL;
}
if (priv->encodings != NULL)
{
LOG ("Unref encodings");
g_object_unref (priv->encodings);
priv->encodings = NULL;
}
if (priv->page_setup != NULL)
{
g_object_unref (priv->page_setup);
priv->page_setup = NULL;
}
if (priv->print_settings != NULL)
{
g_object_unref (priv->print_settings);
priv->print_settings = NULL;
}
G_OBJECT_CLASS (ephy_embed_shell_parent_class)->dispose (object);
}
static void
ephy_embed_shell_finalize (GObject *object)
{
EphyEmbedShell *shell = EPHY_EMBED_SHELL (object);
if (shell->priv->downloads != NULL)
{
LOG ("Destroying downloads list");
g_list_foreach (shell->priv->downloads, (GFunc) g_object_unref, NULL);
g_list_free (shell->priv->downloads);
shell->priv->downloads = NULL;
}
if (shell->priv->global_history)
{
LOG ("Unref history");
g_object_unref (shell->priv->global_history);
}
if (shell->priv->embed_single)
{
LOG ("Unref embed single");
g_object_unref (G_OBJECT (shell->priv->embed_single));
}
if (shell->priv->adblock_manager != NULL)
{
LOG ("Unref adblock manager");
g_object_unref (shell->priv->adblock_manager);
shell->priv->adblock_manager = NULL;
}
G_OBJECT_CLASS (ephy_embed_shell_parent_class)->finalize (object);
}
/**
* ephy_embed_shell_get_favicon_cache:
* @shell: the #EphyEmbedShell
*
* Returns the favicons cache.
*
* Return value: (transfer none): the favicons cache
**/
GObject *
ephy_embed_shell_get_favicon_cache (EphyEmbedShell *shell)
{
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
if (shell->priv->favicon_cache == NULL)
{
shell->priv->favicon_cache = ephy_favicon_cache_new ();
}
return G_OBJECT (shell->priv->favicon_cache);
}
/**
* ephy_embed_shell_get_global_history:
* @shell: the #EphyEmbedShell
*
* Return value: (transfer none):
**/
GObject *
ephy_embed_shell_get_global_history (EphyEmbedShell *shell)
{
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
if (shell->priv->global_history == NULL)
{
shell->priv->global_history = ephy_history_new ();
}
return G_OBJECT (shell->priv->global_history);
}
static GObject *
impl_get_embed_single (EphyEmbedShell *shell)
{
EphyEmbedShellPrivate *priv;
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
priv = shell->priv;
if (priv->embed_single != NULL &&
!priv->single_initialised)
{
g_warning ("ephy_embed_shell_get_embed_single called while the single is being initialised!\n");
return G_OBJECT (priv->embed_single);
}
if (priv->embed_single == NULL)
{
priv->embed_single = EPHY_EMBED_SINGLE
(g_object_new (EPHY_TYPE_EMBED_SINGLE, NULL));
g_assert (priv->embed_single != NULL);
if (!ephy_embed_single_initialize (priv->embed_single))
{
GtkWidget *dialog;
dialog = gtk_message_dialog_new
(NULL,
GTK_DIALOG_MODAL,
GTK_MESSAGE_ERROR,
GTK_BUTTONS_CLOSE,
_("Epiphany can't be used now. "
"Initialization failed."));
gtk_dialog_run (GTK_DIALOG (dialog));
exit (0);
}
priv->single_initialised = TRUE;
}
return G_OBJECT (shell->priv->embed_single);
}
/**
* ephy_embed_shell_get_embed_single:
* @shell: the #EphyEmbedShell
*
* Return value: (transfer none):
**/
GObject *
ephy_embed_shell_get_embed_single (EphyEmbedShell *shell)
{
EphyEmbedShellClass *klass = EPHY_EMBED_SHELL_GET_CLASS (shell);
return klass->get_embed_single (shell);
}
/**
* ephy_embed_shell_get_encodings:
* @shell: the #EphyEmbedShell
*
* Return value: (transfer none):
**/
GObject *
ephy_embed_shell_get_encodings (EphyEmbedShell *shell)
{
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
if (shell->priv->encodings == NULL)
{
shell->priv->encodings = ephy_encodings_new ();
}
return G_OBJECT (shell->priv->encodings);
}
void
ephy_embed_shell_prepare_close (EphyEmbedShell *shell)
{
g_signal_emit (shell, signals[PREPARE_CLOSE], 0);
}
static void
ephy_embed_shell_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyEmbedShell *embed_shell = EPHY_EMBED_SHELL (object);
switch (prop_id)
{
case PROP_PRIVATE_INSTANCE:
embed_shell->priv->private_instance = g_value_get_boolean (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ephy_embed_shell_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
EphyEmbedShell *embed_shell = EPHY_EMBED_SHELL (object);
switch (prop_id)
{
case PROP_PRIVATE_INSTANCE:
g_value_set_boolean (value, embed_shell->priv->private_instance);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ephy_embed_shell_init (EphyEmbedShell *shell)
{
shell->priv = EPHY_EMBED_SHELL_GET_PRIVATE (shell);
/* globally accessible singleton */
g_assert (embed_shell == NULL);
embed_shell = shell;
shell->priv->downloads = NULL;
}
static void
ephy_embed_shell_class_init (EphyEmbedShellClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->dispose = ephy_embed_shell_dispose;
object_class->finalize = ephy_embed_shell_finalize;
object_class->set_property = ephy_embed_shell_set_property;
object_class->get_property = ephy_embed_shell_get_property;
klass->get_embed_single = impl_get_embed_single;
object_properties[PROP_PRIVATE_INSTANCE] =
g_param_spec_boolean ("private-instance",
"Private instance",
"Whether this Epiphany instance is private.",
FALSE,
G_PARAM_READWRITE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK |
G_PARAM_STATIC_BLURB | G_PARAM_CONSTRUCT_ONLY);
g_object_class_install_properties (object_class,
N_PROPERTIES,
object_properties);
/**
* EphyEmbed::download-added:
* @shell: the #EphyEmbedShell
* @download: the #EphyDownload added
*
* Emitted when a #EphyDownload has been added to the global watch list of
* @shell, via ephy_embed_shell_add_download.
**/
signals[DOWNLOAD_ADDED] =
g_signal_new ("download-added",
EPHY_TYPE_EMBED_SHELL,
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyEmbedShellClass, download_added),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE, 1, EPHY_TYPE_DOWNLOAD);
/**
* EphyEmbed::download-removed:
* @shell: the #EphyEmbedShell
* @download: the #EphyDownload being removed
*
* Emitted when a #EphyDownload has been removed from the global watch list of
* @shell, via ephy_embed_shell_remove_download.
**/
signals[DOWNLOAD_REMOVED] =
g_signal_new ("download-removed",
EPHY_TYPE_EMBED_SHELL,
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyEmbedShellClass, download_removed),
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE, 1, EPHY_TYPE_DOWNLOAD);
/**
* EphyEmbed::prepare-close:
* @shell: the #EphyEmbedShell
*
* The ::prepare-close signal is emitted when epiphany is preparing to
* quit on command from the session manager. You can use it when you need
* to do something special (shut down a service, for example).
**/
signals[PREPARE_CLOSE] =
g_signal_new ("prepare-close",
EPHY_TYPE_EMBED_SHELL,
G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyEmbedShellClass, prepare_close),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
/**
* EphyEmbedShell::quit:
* @shell: an #EphyEmbedShell
*
* The ::quit is emitted when all windows (browser windows, popups,
* download windows, etc) are closed and the @shell is ready to be
* closed.
*
* Since: 2.30
**/
signals[QUIT] =
g_signal_new ("quit",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
0,
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_type_class_add_private (object_class, sizeof (EphyEmbedShellPrivate));
}
/**
* ephy_embed_shell_get_default:
*
* Retrieves the default #EphyEmbedShell object
*
* Return value: (transfer none): the default #EphyEmbedShell
**/
EphyEmbedShell *
ephy_embed_shell_get_default (void)
{
return embed_shell;
}
/**
* ephy_embed_shell_get_adblock_manager:
* @shell: the #EphyEmbedShell
*
* Returns the adblock manager.
*
* Return value: (transfer none): the adblock manager
**/
GObject *
ephy_embed_shell_get_adblock_manager (EphyEmbedShell *shell)
{
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
if (shell->priv->adblock_manager == NULL)
{
shell->priv->adblock_manager = g_object_new (EPHY_TYPE_ADBLOCK_MANAGER, NULL);
}
return G_OBJECT (shell->priv->adblock_manager);
}
void
ephy_embed_shell_set_page_setup (EphyEmbedShell *shell,
GtkPageSetup *page_setup)
{
EphyEmbedShellPrivate *priv;
char *path;
g_return_if_fail (EPHY_IS_EMBED_SHELL (shell));
priv = shell->priv;
if (page_setup != NULL)
{
g_object_ref (page_setup);
}
else
{
page_setup = gtk_page_setup_new ();
}
if (priv->page_setup != NULL)
{
g_object_unref (priv->page_setup);
}
priv->page_setup = page_setup;
path = g_build_filename (ephy_dot_dir (), PAGE_SETUP_FILENAME, NULL);
gtk_page_setup_to_file (page_setup, path, NULL);
g_free (path);
}
/**
* ephy_embed_shell_get_page_setup:
*
* Return value: (transfer none):
**/
GtkPageSetup *
ephy_embed_shell_get_page_setup (EphyEmbedShell *shell)
{
EphyEmbedShellPrivate *priv;
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
priv = shell->priv;
if (priv->page_setup == NULL)
{
GError *error = NULL;
char *path;
path = g_build_filename (ephy_dot_dir (), PAGE_SETUP_FILENAME, NULL);
priv->page_setup = gtk_page_setup_new_from_file (path, &error);
g_free (path);
#ifdef ENABLE_MIGRATION
/* If the file doesn't exist, try to fall back to the old format */
if (error != NULL &&
error->domain == G_FILE_ERROR &&
error->code == G_FILE_ERROR_NOENT)
{
path = g_build_filename (ephy_dot_dir (), LEGACY_PAGE_SETUP_FILENAME, NULL);
priv->page_setup = ephy_print_utils_page_setup_new_from_file (path, NULL);
if (priv->page_setup != NULL)
{
/* Delete the old file, so we don't migrate again */
g_unlink (path);
}
g_free (path);
} else if (error != NULL)
g_warning ("error: %s\n", error->message);
#endif /* ENABLE_MIGRATION */
if (error)
{
g_error_free (error);
}
/* If that still didn't work, create a new, empty one */
if (priv->page_setup == NULL)
{
priv->page_setup = gtk_page_setup_new ();
}
}
return priv->page_setup;
}
/**
* ephy_embed_shell_set_print_gettings:
* @shell: the #EphyEmbedShell
* @settings: the new #GtkPrintSettings object
*
* Sets the global #GtkPrintSettings object.
*
**/
void
ephy_embed_shell_set_print_settings (EphyEmbedShell *shell,
GtkPrintSettings *settings)
{
EphyEmbedShellPrivate *priv;
char *path;
g_return_if_fail (EPHY_IS_EMBED_SHELL (shell));
priv = shell->priv;
if (settings != NULL)
{
g_object_ref (settings);
}
if (priv->print_settings != NULL)
{
g_object_unref (priv->print_settings);
}
priv->print_settings = settings ? settings : gtk_print_settings_new ();
path = g_build_filename (ephy_dot_dir (), PRINT_SETTINGS_FILENAME, NULL);
gtk_print_settings_to_file (settings, path, NULL);
g_free (path);
}
/**
* ephy_embed_shell_get_print_settings:
* @shell: the #EphyEmbedShell
*
* Gets the global #GtkPrintSettings object.
*
* Returns: (transfer none): a #GtkPrintSettings object
**/
GtkPrintSettings *
ephy_embed_shell_get_print_settings (EphyEmbedShell *shell)
{
EphyEmbedShellPrivate *priv;
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
priv = shell->priv;
if (priv->print_settings == NULL)
{
GError *error = NULL;
char *path;
path = g_build_filename (ephy_dot_dir (), PRINT_SETTINGS_FILENAME, NULL);
priv->print_settings = gtk_print_settings_new_from_file (path, &error);
g_free (path);
/* Note: the gtk print settings file format is the same as our
* legacy one, so no need to migrate here.
*/
if (priv->print_settings == NULL)
{
priv->print_settings = gtk_print_settings_new ();
}
}
return priv->print_settings;
}
/**
* ephy_embed_shell_get_downloads:
* @shell: the #EphyEmbedShell
*
* Gets the global #GList object listing active downloads.
*
* Returns: (transfer none): a #GList object
**/
GList *
ephy_embed_shell_get_downloads (EphyEmbedShell *shell)
{
EphyEmbedShellPrivate *priv;
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), NULL);
priv = shell->priv;
return priv->downloads;
}
void
ephy_embed_shell_add_download (EphyEmbedShell *shell, EphyDownload *download)
{
EphyEmbedShellPrivate *priv;
g_return_if_fail (EPHY_IS_EMBED_SHELL (shell));
priv = shell->priv;
priv->downloads = g_list_prepend (priv->downloads, download);
g_signal_emit_by_name (shell, "download-added", download, NULL);
}
void
ephy_embed_shell_remove_download (EphyEmbedShell *shell, EphyDownload *download)
{
EphyEmbedShellPrivate *priv;
g_return_if_fail (EPHY_IS_EMBED_SHELL (shell));
priv = shell->priv;
priv->downloads = g_list_remove (priv->downloads, download);
g_signal_emit_by_name (shell, "download-removed", download, NULL);
}
static void
object_notify_cb (EphyEmbedShell *shell, GObject *object)
{
shell->priv->object_count--;
if (shell->priv->object_count == 0)
g_signal_emit (shell, signals[QUIT], 0);
}
void
_ephy_embed_shell_track_object (EphyEmbedShell *shell, GObject *object)
{
g_return_if_fail (EPHY_IS_EMBED_SHELL (shell));
g_return_if_fail (G_IS_OBJECT (object));
g_object_weak_ref (object, (GWeakNotify)object_notify_cb, shell);
shell->priv->object_count++;
}
/**
* ephy_embed_shell_is_private_instance:
* @shell: an #EphyEmbedShell
*
* Returns: whether @shell is a private instance
**/
gboolean
ephy_embed_shell_is_private_instance (EphyEmbedShell *shell)
{
g_return_val_if_fail (EPHY_IS_EMBED_SHELL (shell), FALSE);
return shell->priv->private_instance;
}