/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
/* vim: set sw=2 ts=2 sts=2 et: */
/*
* Copyright © 2007 Xan Lopez
* Copyright © 2008 Jan Alonzo
* Copyright © 2009 Gustavo Noronha Silva
* Copyright © 2009 Igalia S.L.
* Copyright © 2009 Collabora Ltd.
*
* 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 "ephy-embed.h"
#ifndef HAVE_WEBKIT2
#include "ephy-adblock-manager.h"
#endif
#include "ephy-debug.h"
#include "ephy-download.h"
#include "ephy-embed-prefs.h"
#include "ephy-embed-shell.h"
#include "ephy-prefs.h"
#include "ephy-settings.h"
#include "ephy-web-view.h"
#include "nautilus-floating-bar.h"
#include <glib/gi18n.h>
#ifdef HAVE_WEBKIT2
#include <webkit2/webkit2.h>
#else
#include <webkit/webkit.h>
#endif
static void ephy_embed_constructed (GObject *object);
#ifndef HAVE_WEBKIT2
static gboolean ephy_embed_inspect_show_cb (WebKitWebInspector *inspector,
EphyEmbed *embed);
static gboolean ephy_embed_inspect_close_cb (WebKitWebInspector *inspector,
EphyEmbed *embed);
#endif
static void ephy_embed_restored_window_cb (EphyEmbedShell *shell,
EphyEmbed *embed);
#define EPHY_EMBED_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_EMBED, EphyEmbedPrivate))
#define EPHY_EMBED_STATUSBAR_TAB_MESSAGE_CONTEXT_DESCRIPTION "tab_message"
typedef struct {
gchar *text;
guint context_id;
guint message_id;
} EphyEmbedStatusbarMsg;
struct _EphyEmbedPrivate
{
GtkBox *top_widgets_vbox;
#ifndef HAVE_WEBKIT2
GtkScrolledWindow *scrolled_window;
GtkWidget *inspector_window;
GtkWidget *inspector_web_view;
GtkWidget *inspector_scrolled_window;
gboolean inspector_attached;
#endif
GtkPaned *paned;
WebKitWebView *web_view;
GSList *destroy_on_transition_list;
GtkWidget *floating_bar;
GtkWidget *progress;
GtkWidget *fullscreen_message_label;
char *fullscreen_string;
#ifdef HAVE_WEBKIT2
WebKitURIRequest *delayed_request;
#else
WebKitNetworkRequest *delayed_request;
#endif
GtkWidget *overview;
guint overview_mode : 1;
GSList *messages;
GSList *keys;
guint seq_context_id;
guint seq_message_id;
guint tab_message_id;
guint pop_statusbar_later_source_id;
guint fullscreen_message_id;
guint clear_progress_source_id;
gulong status_handler_id;
gulong progress_update_handler_id;
#ifndef HAVE_WEBKIT2
gulong adblock_handler_id;
#endif
};
enum
{
PROP_0,
PROP_OVERVIEW_MODE,
};
G_DEFINE_TYPE (EphyEmbed, ephy_embed, GTK_TYPE_BOX)
/* Portions of the following code based on GTK+.
* License block as follows:
*
* GTK - The GIMP Toolkit
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
* GtkStatusbar Copyright (C) 1998 Shawn T. Amundson
*
* This library 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) any later version.
*
* This library 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 this library; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
* file for a list of people on the GTK+ Team. See the ChangeLog
* files for a list of changes. These files are distributed with
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
*
*/
static guint
ephy_embed_statusbar_get_context_id (EphyEmbed *embed, const char *context_description)
{
char *string;
guint id;
g_return_val_if_fail (EPHY_IS_EMBED (embed), 0);
g_return_val_if_fail (context_description != NULL, 0);
/* we need to preserve namespaces on object datas */
string = g_strconcat ("ephy-embed-status-bar-context:", context_description, NULL);
id = GPOINTER_TO_UINT (g_object_get_data (G_OBJECT (embed), string));
if (id == 0) {
EphyEmbedPrivate *priv = embed->priv;
id = priv->seq_context_id++;
g_object_set_data_full (G_OBJECT (embed), string, GUINT_TO_POINTER (id), NULL);
priv->keys = g_slist_prepend (priv->keys, string);
} else
g_free (string);
return id;
}
static void
ephy_embed_set_statusbar_label (EphyEmbed *embed, const char *label)
{
EphyEmbedPrivate *priv = embed->priv;
nautilus_floating_bar_set_label (NAUTILUS_FLOATING_BAR (priv->floating_bar), label);
if (label == NULL || label[0] == '\0') {
gtk_widget_hide (priv->floating_bar);
gtk_widget_set_halign (priv->floating_bar, GTK_ALIGN_START);
} else
gtk_widget_show (priv->floating_bar);
}
static void
ephy_embed_statusbar_update (EphyEmbed *embed, const char *text)
{
g_return_if_fail (EPHY_IS_EMBED (embed));
ephy_embed_set_statusbar_label (embed, text);
}
static guint
ephy_embed_statusbar_push (EphyEmbed *embed, guint context_id, const char *text)
{
EphyEmbedPrivate *priv;
EphyEmbedStatusbarMsg *msg;
g_return_val_if_fail (EPHY_IS_EMBED (embed), 0);
g_return_val_if_fail (context_id != 0, 0);
g_return_val_if_fail (text != NULL, 0);
priv = embed->priv;
msg = g_slice_new (EphyEmbedStatusbarMsg);
msg->text = g_strdup (text);
msg->context_id = context_id;
msg->message_id = priv->seq_message_id++;
priv->messages = g_slist_prepend (priv->messages, msg);
ephy_embed_statusbar_update (embed, text);
return msg->message_id;
}
/* End of code based on GTK+ GtkStatusbar. */
static void
ephy_embed_statusbar_pop (EphyEmbed *embed, guint context_id)
{
EphyEmbedPrivate *priv;
EphyEmbedStatusbarMsg *msg;
GSList *list;
g_return_if_fail (EPHY_IS_EMBED (embed));
g_return_if_fail (context_id != 0);
priv = embed->priv;
for (list = priv->messages; list; list = list->next) {
EphyEmbedStatusbarMsg *msg = list->data;
if (msg->context_id == context_id) {
priv->messages = g_slist_remove_link (priv->messages, list);
g_free (msg->text);
g_slice_free (EphyEmbedStatusbarMsg, msg);
g_slist_free_1 (list);
break;
}
}
msg = priv->messages ? priv->messages->data : NULL;
ephy_embed_statusbar_update (embed, msg ? msg->text : NULL);
}
#ifndef HAVE_WEBKIT2
static void
resource_request_starting_cb (WebKitWebView *web_view,
WebKitWebFrame *web_frame,
WebKitWebResource *web_resource,
WebKitNetworkRequest *request,
WebKitNetworkResponse *response,
EphyEmbed *embed)
{
EphyAdBlockManager *adblock_manager = EPHY_ADBLOCK_MANAGER (ephy_embed_shell_get_adblock_manager (ephy_embed_shell_get_default ()));
const char *uri = webkit_network_request_get_uri (request);
/* FIXME: How do we implement the other CHECK_TYPEs? Perhaps we
* should figure out a way of adding more information about what the
* resource is for to WebResource? */
if (!ephy_adblock_manager_should_load (adblock_manager, embed, uri,
AD_URI_CHECK_TYPE_OTHER)) {
g_signal_emit_by_name (EPHY_WEB_VIEW (web_view),
"content-blocked", uri);
webkit_network_request_set_uri (request, "about:blank");
}
}
#endif
static void
ephy_embed_destroy_top_widgets (EphyEmbed *embed)
{
GSList *iter;
for (iter = embed->priv->destroy_on_transition_list; iter; iter = iter->next)
gtk_widget_destroy (GTK_WIDGET (iter->data));
}
static void
remove_from_destroy_list_cb (GtkWidget *widget, EphyEmbed *embed)
{
GSList *list;
list = embed->priv->destroy_on_transition_list;
list = g_slist_remove (list, widget);
embed->priv->destroy_on_transition_list = list;
}
#ifdef HAVE_WEBKIT2
static void
load_changed_cb (WebKitWebView *web_view,
WebKitLoadEvent load_event,
EphyEmbed *embed)
{
switch (load_event) {
case WEBKIT_LOAD_STARTED: {
const char *uri;
uri = webkit_web_view_get_uri (web_view);
ephy_embed_set_overview_mode (embed, g_strcmp0 (uri, "ephy-about:overview") == 0);
break;
}
case WEBKIT_LOAD_COMMITTED:
ephy_embed_destroy_top_widgets (embed);
break;
default:
break;
}
}
#else
static void
load_status_changed_cb (WebKitWebView *web_view,
GParamSpec *spec,
EphyEmbed *embed)
{
WebKitLoadStatus status = webkit_web_view_get_load_status (web_view);
const char *address;
if (status == WEBKIT_LOAD_COMMITTED) {
ephy_embed_destroy_top_widgets (embed);
address = ephy_web_view_get_address (EPHY_WEB_VIEW (web_view));
ephy_embed_set_overview_mode (embed, strcmp (address, "ephy-about:overview") == 0);
}
}
#endif
static void
ephy_embed_grab_focus (GtkWidget *widget)
{
GtkWidget *child;
child = GTK_WIDGET (ephy_embed_get_web_view (EPHY_EMBED (widget)));
if (child)
gtk_widget_grab_focus (child);
}
static gboolean
fullscreen_message_label_hide (EphyEmbed *embed)
{
if (embed->priv->fullscreen_message_id) {
gtk_widget_hide (embed->priv->fullscreen_message_label);
g_source_remove (embed->priv->fullscreen_message_id);
embed->priv->fullscreen_message_id = 0;
}
return FALSE;
}
void
ephy_embed_entering_fullscreen (EphyEmbed *embed)
{
gtk_widget_show (embed->priv->fullscreen_message_label);
if (embed->priv->fullscreen_message_id)
g_source_remove (embed->priv->fullscreen_message_id);
embed->priv->fullscreen_message_id = g_timeout_add_seconds (5,
(GSourceFunc)fullscreen_message_label_hide,
embed);
}
void
ephy_embed_leaving_fullscreen (EphyEmbed *embed)
{
fullscreen_message_label_hide (embed);
}
static void
ephy_embed_dispose (GObject *object)
{
EphyEmbed *embed = EPHY_EMBED (object);
EphyEmbedPrivate *priv = embed->priv;
#ifndef HAVE_WEBKIT2
if (priv->inspector_window) {
WebKitWebInspector *inspector;
inspector = webkit_web_view_get_inspector (priv->web_view);
g_signal_handlers_disconnect_by_func (inspector,
ephy_embed_inspect_show_cb,
priv->inspector_window);
g_signal_handlers_disconnect_by_func (inspector,
ephy_embed_inspect_close_cb,
priv->inspector_window);
gtk_widget_destroy (GTK_WIDGET (priv->inspector_window));
priv->inspector_window = NULL;
}
#endif
if (priv->pop_statusbar_later_source_id) {
g_source_remove (priv->pop_statusbar_later_source_id);
priv->pop_statusbar_later_source_id = 0;
}
if (priv->clear_progress_source_id) {
g_source_remove (priv->clear_progress_source_id);
priv->clear_progress_source_id = 0;
}
/* Do not listen to status message notifications anymore, if we try
* to update the statusbar after dispose we might crash. */
if (priv->status_handler_id) {
g_signal_handler_disconnect (priv->web_view, priv->status_handler_id);
priv->status_handler_id = 0;
}
if (priv->progress_update_handler_id) {
g_signal_handler_disconnect (priv->web_view, priv->progress_update_handler_id);
priv->progress_update_handler_id = 0;
}
if (priv->fullscreen_message_id) {
g_source_remove (priv->fullscreen_message_id);
priv->fullscreen_message_id = 0;
}
#ifndef HAVE_WEBKIT2
if (priv->adblock_handler_id) {
g_signal_handler_disconnect (priv->web_view, priv->adblock_handler_id);
priv->adblock_handler_id = 0;
}
#endif
g_clear_object (&priv->delayed_request);
G_OBJECT_CLASS (ephy_embed_parent_class)->dispose (object);
}
static void
ephy_embed_finalize (GObject *object)
{
EphyEmbed *embed = EPHY_EMBED (object);
EphyEmbedPrivate *priv = embed->priv;
EphyEmbedShell *shell = ephy_embed_shell_get_default ();
GSList *list;
g_signal_handlers_disconnect_by_func(shell, ephy_embed_restored_window_cb, embed);
list = priv->destroy_on_transition_list;
for (; list; list = list->next) {
GtkWidget *widget = GTK_WIDGET (list->data);
g_signal_handlers_disconnect_by_func (widget, remove_from_destroy_list_cb, embed);
}
g_slist_free (priv->destroy_on_transition_list);
for (list = priv->messages; list; list = list->next) {
EphyEmbedStatusbarMsg *msg;
msg = list->data;
g_free (msg->text);
g_slice_free (EphyEmbedStatusbarMsg, msg);
}
g_slist_free (priv->messages);
priv->messages = NULL;
for (list = priv->keys; list; list = list->next)
g_free (list->data);
g_slist_free (priv->keys);
priv->keys = NULL;
g_free (embed->priv->fullscreen_string);
G_OBJECT_CLASS (ephy_embed_parent_class)->finalize (object);
}
static void
ephy_embed_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyEmbed *embed = EPHY_EMBED (object);
switch (prop_id)
{
case PROP_OVERVIEW_MODE:
ephy_embed_set_overview_mode (embed, g_value_get_boolean (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
ephy_embed_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
EphyEmbed *embed = EPHY_EMBED (object);
switch (prop_id)
{
case PROP_OVERVIEW_MODE:
g_value_set_boolean (value, ephy_embed_get_overview_mode (embed));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
ephy_embed_class_init (EphyEmbedClass *klass)
{
GObjectClass *object_class = (GObjectClass *)klass;
GtkWidgetClass *widget_class = (GtkWidgetClass *)klass;
object_class->constructed = ephy_embed_constructed;
object_class->finalize = ephy_embed_finalize;
object_class->dispose = ephy_embed_dispose;
object_class->set_property = ephy_embed_set_property;
object_class->get_property = ephy_embed_get_property;
widget_class->grab_focus = ephy_embed_grab_focus;
/**
* EphyEmbed:overview-mode:
*
* If %TRUE activates the overview mode in this #EphyEmbed.
**/
g_object_class_install_property (object_class,
PROP_OVERVIEW_MODE,
g_param_spec_boolean ("overview-mode",
"Overview mode",
"Whether the embed is showing the overview",
FALSE,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
g_type_class_add_private (G_OBJECT_CLASS (klass), sizeof(EphyEmbedPrivate));
}
#ifdef HAVE_WEBKIT2
static gboolean
ephy_embed_attach_inspector_cb (WebKitWebInspector *inspector,
EphyEmbed *embed)
{
GtkWidget *inspector_view = GTK_WIDGET (webkit_web_inspector_get_web_view (inspector));
int inspected_view_height;
guint attached_height;
inspected_view_height = gtk_widget_get_allocated_height (GTK_WIDGET (embed->priv->web_view));
attached_height = webkit_web_inspector_get_attached_height (inspector);
gtk_paned_set_position (embed->priv->paned, inspected_view_height - attached_height);
gtk_paned_add2 (embed->priv->paned, inspector_view);
gtk_widget_show (inspector_view);
return TRUE;
}
#else
static WebKitWebView *
ephy_embed_inspect_web_view_cb (WebKitWebInspector *inspector,
WebKitWebView *web_view,
EphyEmbed *embed)
{
EphyEmbedPrivate *priv = embed->priv;
priv->inspector_web_view = ephy_web_view_new();
gtk_container_add (GTK_CONTAINER (priv->inspector_scrolled_window),
priv->inspector_web_view);
return WEBKIT_WEB_VIEW (priv->inspector_web_view);
}
static gboolean
ephy_embed_attach_inspector_cb (WebKitWebInspector *inspector,
EphyEmbed *embed)
{
embed->priv->inspector_attached = TRUE;
gtk_widget_hide (embed->priv->inspector_window);
gtk_widget_reparent (GTK_WIDGET (embed->priv->inspector_scrolled_window),
GTK_WIDGET (embed->priv->paned));
return TRUE;
}
static void
ephy_embed_detach_inspector (EphyEmbed *embed, WebKitWebInspector *inspector)
{
embed->priv->inspector_attached = FALSE;
gtk_widget_reparent (GTK_WIDGET (embed->priv->inspector_scrolled_window),
GTK_WIDGET (embed->priv->inspector_window));
}
static gboolean
ephy_embed_detach_inspector_cb (WebKitWebInspector *inspector,
EphyEmbed *embed)
{
ephy_embed_detach_inspector (embed, inspector);
gtk_widget_show_all (embed->priv->inspector_window);
return TRUE;
}
static gboolean
ephy_embed_inspect_show_cb (WebKitWebInspector *inspector,
EphyEmbed *embed)
{
if (!embed->priv->inspector_attached) {
gtk_widget_show_all (embed->priv->inspector_window);
gtk_window_present (GTK_WINDOW (embed->priv->inspector_window));
} else {
GtkAllocation allocation;
gtk_widget_get_allocation (GTK_WIDGET (embed->priv->scrolled_window), &allocation);
/* Set a sane position for the mover */
gtk_paned_set_position (embed->priv->paned, allocation.height * 0.5);
gtk_widget_show (embed->priv->inspector_scrolled_window);
}
return TRUE;
}
static gboolean
ephy_embed_inspect_close_cb (WebKitWebInspector *inspector,
EphyEmbed *embed)
{
EphyEmbedPrivate *priv = embed->priv;
if (!priv->inspector_window)
return TRUE;
gtk_widget_destroy (priv->inspector_web_view);
priv->inspector_web_view = NULL;
if (!priv->inspector_attached)
gtk_widget_hide (priv->inspector_window);
else
gtk_widget_hide (priv->inspector_scrolled_window);
ephy_embed_detach_inspector (embed, inspector);
return TRUE;
}
#endif
void
ephy_embed_auto_download_url (EphyEmbed *embed, const char *url)
{
EphyDownload *download;
download = ephy_download_new_for_uri (url, NULL);
ephy_download_set_auto_destination (download);
ephy_download_set_action (download, EPHY_DOWNLOAD_ACTION_OPEN);
}
#ifndef HAVE_WEBKIT2
static gboolean
download_requested_cb (WebKitWebView *web_view,
WebKitDownload *download,
EphyEmbed *embed)
{
EphyDownload *ed;
GtkWidget *toplevel;
GtkWindow *window = NULL;
/* Is download locked down? */
if (g_settings_get_boolean (EPHY_SETTINGS_LOCKDOWN,
EPHY_PREFS_LOCKDOWN_SAVE_TO_DISK))
return FALSE;
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (embed));
if (GTK_IS_WINDOW (toplevel))
window = GTK_WINDOW (toplevel);
ed = ephy_download_new_for_download (download, window);
ephy_download_set_auto_destination (ed);
return TRUE;
}
#endif
static void
ephy_embed_set_fullscreen_message (EphyEmbed *embed,
gboolean is_html5_fullscreen)
{
char *message;
if (G_UNLIKELY (embed->priv->fullscreen_string == NULL))
embed->priv->fullscreen_string = g_strdup (_("Press %s to exit fullscreen"));
/* Translators: 'ESC' and 'F11' are keyboard keys. */
message = g_strdup_printf (embed->priv->fullscreen_string, is_html5_fullscreen ? _("ESC") : _("F11"));
gtk_label_set_text (GTK_LABEL (embed->priv->fullscreen_message_label),
message);
g_free (message);
}
static gboolean
entering_fullscreen_cb (WebKitWebView *web_view,
#ifndef HAVE_WEBKIT2
GObject *element,
#endif
EphyEmbed *embed)
{
ephy_embed_set_fullscreen_message (embed, TRUE);
return FALSE;
}
static gboolean
leaving_fullscreen_cb (WebKitWebView *web_view,
#ifndef HAVE_WEBKIT2
GObject *element,
#endif
EphyEmbed *embed)
{
ephy_embed_set_fullscreen_message (embed, FALSE);
return FALSE;
}
static gboolean
pop_statusbar_later_cb (gpointer data)
{
EphyEmbed *embed = EPHY_EMBED (data);
EphyEmbedPrivate *priv = embed->priv;
ephy_embed_statusbar_pop (embed, priv->tab_message_id);
priv->pop_statusbar_later_source_id = 0;
return FALSE;
}
static void
status_message_notify_cb (EphyWebView *view, GParamSpec *pspec, EphyEmbed *embed)
{
const char *message;
EphyEmbedPrivate *priv;
message = ephy_web_view_get_status_message (view);
priv = embed->priv;
if (message) {
if (priv->pop_statusbar_later_source_id) {
g_source_remove (priv->pop_statusbar_later_source_id);
priv->pop_statusbar_later_source_id = 0;
}
ephy_embed_statusbar_pop (embed, priv->tab_message_id);
ephy_embed_statusbar_push (embed, priv->tab_message_id, message);
} else {
/* A short timeout before hiding the statusbar ensures that while moving
over a series of links, the overlay widget doesn't flicker on and off. */
if (priv->pop_statusbar_later_source_id == 0) {
priv->pop_statusbar_later_source_id = g_timeout_add (250, pop_statusbar_later_cb, embed);
}
}
}
#ifdef HAVE_WEBKIT2
static void
window_geometry_changed (WebKitWindowProperties *properties, GParamSpec *pspec, EphyEmbed *embed)
{
GtkWidget *window;
gboolean is_popup;
GdkRectangle geometry;
window = gtk_widget_get_toplevel (GTK_WIDGET (embed));
if (!window || !gtk_widget_is_toplevel (window))
return;
g_object_get (window, "is-popup", &is_popup, NULL);
if (!is_popup)
return;
webkit_window_properties_get_geometry (properties, &geometry);
if (geometry.x >= 0 && geometry.y >= 0)
gtk_window_move (GTK_WINDOW (window), geometry.x, geometry.y);
if (geometry.width > 0 && geometry.height > 0)
gtk_window_resize (GTK_WINDOW (window), geometry.width, geometry.height);
}
#else
static void
window_resize_requested (WebKitWebWindowFeatures *features, GParamSpec *pspec, EphyEmbed *embed)
{
GtkWidget *window;
gboolean is_popup;
const char *property_name;
int width, height;
window = gtk_widget_get_toplevel (GTK_WIDGET (embed));
if (!window || !gtk_widget_is_toplevel (window))
return;
g_object_get (window, "is-popup", &is_popup, NULL);
if (!is_popup)
return;
property_name = g_param_spec_get_name (pspec);
if (g_str_equal (property_name, "x") || g_str_equal (property_name, "y")) {
int x, y;
g_object_get (features, "x", &x, "y", &y, NULL);
gtk_window_move (GTK_WINDOW (window), x, y);
return;
}
g_object_get (features, "width", &width, "height", &height, NULL);
gtk_window_resize (GTK_WINDOW (window), width, height);
}
#endif
static gboolean
clear_progress_cb (EphyEmbed *embed)
{
gtk_widget_hide (embed->priv->progress);
embed->priv->clear_progress_source_id = 0;
return FALSE;
}
static void
progress_update (EphyWebView *view, GParamSpec *pspec, EphyEmbed *embed)
{
gdouble progress;
gboolean loading;
const char *uri;
EphyEmbedPrivate *priv = embed->priv;
if (priv->clear_progress_source_id) {
g_source_remove (priv->clear_progress_source_id);
priv->clear_progress_source_id = 0;
}
uri = webkit_web_view_get_uri (priv->web_view);
if (!uri || g_str_has_prefix (uri, "ephy-about:") ||
g_str_has_prefix (uri, "about:"))
return;
#ifdef HAVE_WEBKIT2
progress = webkit_web_view_get_estimated_load_progress (priv->web_view);
#else
progress = webkit_web_view_get_progress (priv->web_view);
#endif
loading = ephy_web_view_is_loading (EPHY_WEB_VIEW (priv->web_view));
if (progress == 1.0 || !loading)
priv->clear_progress_source_id = g_timeout_add (500,
(GSourceFunc)clear_progress_cb,
embed);
else
gtk_widget_show (priv->progress);
gtk_progress_bar_set_fraction (GTK_PROGRESS_BAR (priv->progress),
(loading || progress == 1.0) ? progress : 0.0);
}
#ifndef HAVE_WEBKIT2
static void
setup_adblock (GSettings *settings,
char *key,
EphyEmbed *embed)
{
EphyEmbedPrivate *priv = embed->priv;
EphyWebView *web_view = ephy_embed_get_web_view (embed);
if (g_settings_get_boolean (EPHY_SETTINGS_WEB, EPHY_PREFS_WEB_ENABLE_ADBLOCK) &&
priv->adblock_handler_id == 0) {
priv->adblock_handler_id = g_signal_connect (web_view, "resource_request_starting",
G_CALLBACK (resource_request_starting_cb), embed);
} else if (priv->adblock_handler_id) {
g_signal_handler_disconnect (web_view, priv->adblock_handler_id);
priv->adblock_handler_id = 0;
}
}
#endif
static void
ephy_embed_maybe_load_delayed_request (EphyEmbed *embed)
{
EphyEmbedPrivate *priv = embed->priv;
EphyWebView *web_view;
if (!priv->delayed_request)
return;
web_view = ephy_embed_get_web_view (embed);
ephy_web_view_load_request (web_view, priv->delayed_request);
g_clear_object (&priv->delayed_request);
/* This is to allow UI elements watching load status to show that the page is
* loading as soon as possible.
*/
#ifdef HAVE_WEBKIT2
g_signal_emit_by_name (web_view, "load-changed", WEBKIT_LOAD_STARTED);
#else
g_object_notify (G_OBJECT (web_view), "load-status");
#endif
}
static void
ephy_embed_restored_window_cb (EphyEmbedShell *shell, EphyEmbed *embed)
{
if (!gtk_widget_get_mapped (GTK_WIDGET (embed)))
return;
ephy_embed_maybe_load_delayed_request (embed);
}
static void
ephy_embed_mapped_cb (GtkWidget *widget, gpointer data)
{
ephy_embed_maybe_load_delayed_request ((EphyEmbed*)widget);
}
static void
ephy_embed_constructed (GObject *object)
{
EphyEmbed *embed = (EphyEmbed*)object;
EphyEmbedPrivate *priv = embed->priv;
EphyEmbedShell *shell = ephy_embed_shell_get_default ();
#ifndef HAVE_WEBKIT2
GtkWidget *scrolled_window;
#endif
GtkWidget *paned;
WebKitWebView *web_view;
#ifdef HAVE_WEBKIT2
WebKitWindowProperties *window_properties;
#else
WebKitWebWindowFeatures *window_features;
#endif
WebKitWebInspector *inspector;
GtkWidget *overlay;
g_signal_connect (shell, "window-restored",
G_CALLBACK (ephy_embed_restored_window_cb), embed);
g_signal_connect (embed, "map",
G_CALLBACK (ephy_embed_mapped_cb), NULL);
/* Skeleton */
web_view = WEBKIT_WEB_VIEW (ephy_web_view_new ());
#ifndef HAVE_WEBKIT2
scrolled_window = GTK_WIDGET (priv->scrolled_window);
#endif
overlay = gtk_overlay_new ();
gtk_widget_add_events (overlay,
GDK_ENTER_NOTIFY_MASK |
GDK_LEAVE_NOTIFY_MASK);
#ifdef HAVE_WEBKIT2
gtk_container_add (GTK_CONTAINER (overlay), GTK_WIDGET (web_view));
#else
gtk_container_add (GTK_CONTAINER (overlay), scrolled_window);
#endif
/* The overview. In incognito mode we don't use it. */
if (ephy_embed_shell_get_mode (ephy_embed_shell_get_default ()) !=
EPHY_EMBED_SHELL_MODE_INCOGNITO) {
priv->overview = ephy_overview_new ();
gtk_widget_set_halign (priv->overview, GTK_ALIGN_FILL);
gtk_widget_set_valign (priv->overview, GTK_ALIGN_FILL);
gtk_overlay_add_overlay (GTK_OVERLAY (overlay), priv->overview);
g_object_bind_property (embed, "overview-mode",
priv->overview, "visible",
G_BINDING_SYNC_CREATE);
}
/* Floating message popup for fullscreen mode. */
priv->fullscreen_message_label = gtk_label_new (NULL);
gtk_widget_set_name (priv->fullscreen_message_label, "fullscreen-popup");
gtk_widget_set_halign (priv->fullscreen_message_label, GTK_ALIGN_CENTER);
gtk_widget_set_valign (priv->fullscreen_message_label, GTK_ALIGN_CENTER);
gtk_widget_set_no_show_all (priv->fullscreen_message_label, TRUE);
gtk_overlay_add_overlay (GTK_OVERLAY (overlay), priv->fullscreen_message_label);
ephy_embed_set_fullscreen_message (embed, FALSE);
/* statusbar is hidden by default */
priv->floating_bar = nautilus_floating_bar_new (NULL, FALSE);
gtk_widget_set_halign (priv->floating_bar, GTK_ALIGN_START);
gtk_widget_set_valign (priv->floating_bar, GTK_ALIGN_END);
gtk_widget_set_no_show_all (priv->floating_bar, TRUE);
gtk_overlay_add_overlay (GTK_OVERLAY (overlay), priv->floating_bar);
priv->progress = gtk_progress_bar_new ();
gtk_style_context_add_class (gtk_widget_get_style_context (priv->progress),
GTK_STYLE_CLASS_OSD);
gtk_widget_set_halign (priv->progress, GTK_ALIGN_FILL);
gtk_widget_set_valign (priv->progress, GTK_ALIGN_START);
gtk_overlay_add_overlay (GTK_OVERLAY (overlay), priv->progress);
paned = GTK_WIDGET (priv->paned);
priv->web_view = web_view;
#ifdef HAVE_WEBKIT2
priv->progress_update_handler_id = g_signal_connect (web_view, "notify::estimated-load-progress",
G_CALLBACK (progress_update), object);
#else
priv->progress_update_handler_id = g_signal_connect (web_view, "notify::progress",
G_CALLBACK (progress_update), object);
#endif
#ifndef HAVE_WEBKIT2
gtk_container_add (GTK_CONTAINER (scrolled_window),
GTK_WIDGET (web_view));
#endif
gtk_paned_pack1 (GTK_PANED (paned), GTK_WIDGET (overlay),
TRUE, FALSE);
gtk_box_pack_start (GTK_BOX (embed),
GTK_WIDGET (priv->top_widgets_vbox),
FALSE, FALSE, 0);
gtk_box_pack_start (GTK_BOX (embed), paned, TRUE, TRUE, 0);
gtk_widget_show (GTK_WIDGET (priv->top_widgets_vbox));
gtk_widget_show (GTK_WIDGET (web_view));
gtk_widget_show_all (paned);
#ifdef HAVE_WEBKIT2
g_object_connect (web_view,
"signal::load-changed", G_CALLBACK (load_changed_cb), embed,
"signal::enter-fullscreen", G_CALLBACK (entering_fullscreen_cb), embed,
"signal::leave-fullscreen", G_CALLBACK (leaving_fullscreen_cb), embed,
NULL);
#else
g_object_connect (web_view,
"signal::notify::load-status", G_CALLBACK (load_status_changed_cb), embed,
"signal::download-requested", G_CALLBACK (download_requested_cb), embed,
"signal::entering-fullscreen", G_CALLBACK (entering_fullscreen_cb), embed,
"signal::leaving-fullscreen", G_CALLBACK (leaving_fullscreen_cb), embed,
NULL);
g_signal_connect (EPHY_SETTINGS_WEB,
"changed::" EPHY_PREFS_WEB_ENABLE_ADBLOCK,
G_CALLBACK (setup_adblock), embed);
setup_adblock (EPHY_SETTINGS_WEB, EPHY_PREFS_WEB_ENABLE_ADBLOCK, embed);
#endif
priv->status_handler_id = g_signal_connect (web_view, "notify::status-message",
G_CALLBACK (status_message_notify_cb),
embed);
#ifdef HAVE_WEBKIT2
/* Window properties */
window_properties = webkit_web_view_get_window_properties (web_view);
g_signal_connect (window_properties, "notify::geometry",
G_CALLBACK (window_geometry_changed),
embed);
#else
/* Window features */
window_features = webkit_web_view_get_window_features (web_view);
g_object_connect (window_features,
"signal::notify::x", G_CALLBACK (window_resize_requested), embed,
"signal::notify::y", G_CALLBACK (window_resize_requested), embed,
"signal::notify::width", G_CALLBACK (window_resize_requested), embed,
"signal::notify::height", G_CALLBACK (window_resize_requested), embed,
NULL);
#endif
/* The inspector */
inspector = webkit_web_view_get_inspector (web_view);
#ifdef HAVE_WEBKIT2
g_signal_connect (inspector, "attach",
G_CALLBACK (ephy_embed_attach_inspector_cb),
embed);
#else
priv->inspector_attached = TRUE;
priv->inspector_scrolled_window = gtk_scrolled_window_new (NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (priv->inspector_scrolled_window),
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtk_container_add (GTK_CONTAINER (priv->paned),
priv->inspector_scrolled_window);
priv->inspector_window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_title (GTK_WINDOW (priv->inspector_window),
_("Web Inspector"));
gtk_window_set_default_size (GTK_WINDOW (priv->inspector_window),
800, 600);
g_signal_connect (priv->inspector_window,
"delete-event", G_CALLBACK (gtk_widget_hide_on_delete),
NULL);
g_object_connect (inspector,
"signal::inspect-web-view", G_CALLBACK (ephy_embed_inspect_web_view_cb),
embed,
"signal::show-window", G_CALLBACK (ephy_embed_inspect_show_cb),
embed,
"signal::close-window", G_CALLBACK (ephy_embed_inspect_close_cb),
embed,
"signal::attach-window", G_CALLBACK (ephy_embed_attach_inspector_cb),
embed,
"signal::detach-window", G_CALLBACK (ephy_embed_detach_inspector_cb),
embed,
NULL);
#endif
ephy_embed_prefs_add_embed (embed);
}
static void
ephy_embed_init (EphyEmbed *embed)
{
EphyEmbedPrivate *priv;
priv = embed->priv = EPHY_EMBED_GET_PRIVATE (embed);
gtk_orientable_set_orientation (GTK_ORIENTABLE (embed),
GTK_ORIENTATION_VERTICAL);
#ifndef HAVE_WEBKIT2
priv->scrolled_window = GTK_SCROLLED_WINDOW (gtk_scrolled_window_new (NULL, NULL));
#endif
priv->paned = GTK_PANED (gtk_paned_new (GTK_ORIENTATION_VERTICAL));
priv->top_widgets_vbox = GTK_BOX (gtk_box_new (GTK_ORIENTATION_VERTICAL, 0));
priv->seq_context_id = 1;
priv->seq_message_id = 1;
priv->tab_message_id = ephy_embed_statusbar_get_context_id (embed, EPHY_EMBED_STATUSBAR_TAB_MESSAGE_CONTEXT_DESCRIPTION);
#ifndef HAVE_WEBKIT2
gtk_scrolled_window_set_policy (priv->scrolled_window,
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtk_scrolled_window_set_shadow_type (priv->scrolled_window,
GTK_SHADOW_IN);
#endif
}
/**
* ephy_embed_get_web_view:
* @embed: and #EphyEmbed
*
* Returns the #EphyWebView wrapped by @embed.
*
* Returns: (transfer none): an #EphyWebView
**/
EphyWebView*
ephy_embed_get_web_view (EphyEmbed *embed)
{
g_return_val_if_fail (EPHY_IS_EMBED (embed), NULL);
return EPHY_WEB_VIEW (embed->priv->web_view);
}
/**
* ephy_embed_add_top_widget:
* @embed: an #EphyEmbed
* @widget: a #GtkWidget
* @destroy_on_transition: whether the widget be automatically
* destroyed on page transitions
*
* Adds a #GtkWidget to the top of the embed.
*/
void
ephy_embed_add_top_widget (EphyEmbed *embed, GtkWidget *widget, gboolean destroy_on_transition)
{
GSList *list;
if (destroy_on_transition) {
list = embed->priv->destroy_on_transition_list;
list = g_slist_prepend (list, widget);
embed->priv->destroy_on_transition_list = list;
g_signal_connect (widget, "destroy", G_CALLBACK (remove_from_destroy_list_cb), embed);
}
gtk_box_pack_end (embed->priv->top_widgets_vbox,
GTK_WIDGET (widget), TRUE, TRUE, 0);
}
/**
* ephy_embed_remove_top_widget:
* @embed: an #EphyEmbed
* @widget: a #GtkWidget
*
* Removes an #GtkWidget from the top of the embed. The #GtkWidget
* must have been added using ephy_embed_add_top_widget(), and not
* have been removed by other means. See gtk_container_remove() for
* details.
*/
void
ephy_embed_remove_top_widget (EphyEmbed *embed, GtkWidget *widget)
{
if (g_slist_find (embed->priv->destroy_on_transition_list, widget)) {
GSList *list;
g_signal_handlers_disconnect_by_func (widget, remove_from_destroy_list_cb, embed);
list = embed->priv->destroy_on_transition_list;
list = g_slist_remove (list, widget);
embed->priv->destroy_on_transition_list = list;
}
gtk_container_remove (GTK_CONTAINER (embed->priv->top_widgets_vbox),
GTK_WIDGET (widget));
}
void
ephy_embed_set_overview_mode (EphyEmbed *embed, gboolean overview_mode)
{
EphyEmbedPrivate *priv;
g_return_if_fail (EPHY_IS_EMBED (embed));
priv = embed->priv;
if (priv->overview_mode == overview_mode)
return;
priv->overview_mode = overview_mode;
g_object_notify (G_OBJECT (embed), "overview-mode");
}
gboolean
ephy_embed_get_overview_mode (EphyEmbed *embed)
{
g_return_val_if_fail (EPHY_IS_EMBED (embed), FALSE);
return embed->priv->overview_mode;
}
/**
* ephy_embed_set_delayed_load_request:
* @embed: a #EphyEmbed
* @request: a #WebKitNetworkRequest
*
* Sets the #WebKitNetworkRequest that should be loaded when the tab this embed
* is on is switched to.
*/
#ifdef HAVE_WEBKIT2
void
ephy_embed_set_delayed_load_request (EphyEmbed *embed, WebKitURIRequest *request)
#else
void
ephy_embed_set_delayed_load_request (EphyEmbed *embed, WebKitNetworkRequest *request)
#endif
{
g_return_if_fail (EPHY_IS_EMBED (embed));
#ifdef HAVE_WEBKIT2
g_return_if_fail (WEBKIT_IS_URI_REQUEST (request));
#else
g_return_if_fail (WEBKIT_IS_NETWORK_REQUEST (request));
#endif
g_clear_object (&embed->priv->delayed_request);
g_object_ref (request);
embed->priv->delayed_request = request;
}
/**
* ephy_embed_has_load_pending:
* @embed: a #EphyEmbed
*
* Checks whether a load has been delayed for this #EphyEmbed.
*
* Returns: %TRUE or %FALSE
*/
gboolean
ephy_embed_has_load_pending (EphyEmbed *embed)
{
g_return_val_if_fail (EPHY_IS_EMBED (embed), FALSE);
return !!embed->priv->delayed_request;
}
/**
* ephy_embed_get_overview:
* @embed: a #EphyEmbed
*
* Gets the #EphyOverview in this @embed
*
* Returns: (transfer none): the overview widget
**/
EphyOverview *
ephy_embed_get_overview (EphyEmbed *embed)
{
g_return_val_if_fail (EPHY_IS_EMBED (embed), NULL);
return EPHY_OVERVIEW (embed->priv->overview);
}