/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
/* vim: set sw=2 ts=2 sts=2 et: */
/* Copyright © 2008 Xan Lopez <xan@gnome.org>
*
* 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-prefs.h"
#include "ephy-embed-shell.h"
#include "ephy-embed-utils.h"
#include "ephy-file-helpers.h"
#include "ephy-langs.h"
#include "ephy-prefs.h"
#include "ephy-settings.h"
#include <glib.h>
#include <math.h>
#ifdef HAVE_WEBKIT2
#include <webkit2/webkit2.h>
#else
#include <webkit/webkit.h>
#endif
typedef struct
{
char *schema;
char *key;
char *webkit_pref;
void (*callback) (GSettings *settings, char *key, gpointer data);
} PrefData;
#ifdef HAVE_WEBKIT2
#define ENABLE_SCRIPTS_SETTING "enable-javascript"
#define DEFAULT_ENCODING_SETTING "default-charset"
static WebKitSettings *webkit_settings = NULL;
#else
#define ENABLE_SCRIPTS_SETTING "enable-scripts"
#define DEFAULT_ENCODING_SETTING "default-encoding"
static WebKitWebSettings *webkit_settings = NULL;
#endif
static void
webkit_pref_callback_user_stylesheet (GSettings *settings,
char *key,
gpointer data)
{
gboolean value = FALSE;
char *uri = NULL;
char *webkit_pref = data;
value = g_settings_get_boolean (settings, key);
if (value)
/* We need the leading file://, so use g_strconcat instead
* of g_build_filename */
uri = g_strconcat ("file://",
ephy_dot_dir (),
G_DIR_SEPARATOR_S,
USER_STYLESHEET_FILENAME,
NULL);
#ifdef HAVE_WEBKIT2
/* TODO: user-stylesheet-uri setting */
#else
g_object_set (webkit_settings, webkit_pref, uri, NULL);
#endif
g_free (uri);
}
static char *
webkit_pref_get_vendor_user_agent (void)
{
char *vendor_user_agent = NULL;
GKeyFile *branding_keyfile;
branding_keyfile = g_key_file_new ();
if (g_key_file_load_from_file (branding_keyfile, SHARE_DIR"/branding.conf",
G_KEY_FILE_NONE, NULL)) {
char *vendor;
char *vendor_sub;
char *vendor_comment;
vendor = g_key_file_get_string (branding_keyfile,
"User Agent", "Vendor", NULL);
vendor_sub = g_key_file_get_string (branding_keyfile,
"User Agent", "VendorSub", NULL);
vendor_comment = g_key_file_get_string (branding_keyfile,
"User Agent", "VendorComment", NULL);
if (vendor) {
vendor_user_agent = g_strconcat (vendor,
vendor_sub ? "/" : "",
vendor_sub ? vendor_sub : "",
vendor_comment ? " (" : "",
vendor_comment ? vendor_comment : "",
vendor_comment ? ")" : "",
NULL);
}
g_free (vendor);
g_free (vendor_sub);
g_free (vendor_comment);
}
g_key_file_free (branding_keyfile);
return vendor_user_agent;
}
#ifdef HAVE_WEBKIT2
static const char *
webkit_pref_get_internal_user_agent (void)
{
static char *user_agent = NULL;
static gboolean initialized = FALSE;
char *vendor_user_agent;
const char *webkit_user_agent;
if (initialized)
return user_agent;
initialized = TRUE;
vendor_user_agent = webkit_pref_get_vendor_user_agent ();
if (!vendor_user_agent)
return NULL;
webkit_user_agent = webkit_settings_get_user_agent (webkit_settings);
user_agent = g_strdup_printf ("%s %s Epiphany/%s", webkit_user_agent,
vendor_user_agent, VERSION);
g_free (vendor_user_agent);
return user_agent;
}
#else
static char *
webkit_pref_get_internal_user_agent (void)
{
char *user_agent;
char *webkit_user_agent;
char *vendor_user_agent;
vendor_user_agent = webkit_pref_get_vendor_user_agent ();
g_object_get (webkit_settings, "user-agent", &webkit_user_agent, NULL);
user_agent = g_strconcat (webkit_user_agent, " ",
vendor_user_agent ? vendor_user_agent : "",
vendor_user_agent ? " " : "",
"Epiphany/"VERSION,
NULL);
g_free (vendor_user_agent);
g_free (webkit_user_agent);
return user_agent;
}
#endif
static void
webkit_pref_callback_user_agent (GSettings *settings,
char *key,
gpointer data)
{
#ifdef HAVE_WEBKIT2
char *value;
const char *internal_user_agent;
value = g_settings_get_string (settings, key);
if (value != NULL && value[0] != '\0') {
webkit_settings_set_user_agent (webkit_settings, value);
g_free (value);
return;
}
g_free (value);
internal_user_agent = webkit_pref_get_internal_user_agent ();
if (internal_user_agent)
webkit_settings_set_user_agent (webkit_settings, internal_user_agent);
else
webkit_settings_set_user_agent_with_application_details (webkit_settings,
"Epiphany", VERSION);
#else
char *value = NULL;
static char *internal_user_agent = NULL;
char *webkit_pref = data;
value = g_settings_get_string (settings, key);
if (value == NULL || value[0] == '\0') {
if (internal_user_agent == NULL)
internal_user_agent = webkit_pref_get_internal_user_agent ();
g_object_set (webkit_settings, webkit_pref, internal_user_agent, NULL);
} else
g_object_set (webkit_settings, webkit_pref, value, NULL);
g_free (value);
#endif
}
#ifdef HAVE_WEBKIT2
/* This doesn't contain WebKit2 specific API, but it's only used inside
* HAVE_WEBKIT2 blocks, so it gives a compile warning when building
* with WebKit1.
*/
static gdouble
get_screen_dpi (GdkScreen *screen)
{
gdouble dpi;
gdouble dp, di;
dpi = gdk_screen_get_resolution (screen);
if (dpi != -1)
return dpi;
dp = hypot (gdk_screen_get_width (screen), gdk_screen_get_height (screen));
di = hypot (gdk_screen_get_width_mm (screen), gdk_screen_get_height_mm (screen)) / 25.4;
return dp / di;
}
#endif
static guint
normalize_font_size (gdouble font_size)
{
#ifdef HAVE_WEBKIT2
/* WebKit2 uses font sizes in pixels. */
GdkScreen *screen;
gdouble dpi;
/* FIXME: We should use the view screen instead of the detault one
* but we don't have access to the view here.
*/
screen = gdk_screen_get_default ();
dpi = screen ? get_screen_dpi (screen) : 96;
return font_size / 72.0 * dpi;
#else
return font_size;
#endif
}
static void
webkit_pref_callback_font_size (GSettings *settings,
char *key,
gpointer data)
{
char *webkit_pref = data;
char *value = NULL;
int size = 9; /* FIXME: What to use here? */
char *schema = NULL;
g_object_get (settings, "schema", &schema, NULL);
/* If we are changing a GNOME font value and we are not using GNOME fonts in
* Epiphany, return. */
if (g_strcmp0 (schema, EPHY_PREFS_WEB_SCHEMA) != 0 &&
g_settings_get_boolean (EPHY_SETTINGS_WEB, EPHY_PREFS_WEB_USE_GNOME_FONTS) != TRUE)
{
g_free (schema);
return;
}
g_free (schema);
value = g_settings_get_string (settings, key);
if (value) {
PangoFontDescription* desc;
desc = pango_font_description_from_string (value);
size = pango_font_description_get_size (desc);
if (pango_font_description_get_size_is_absolute (desc) == FALSE)
size /= PANGO_SCALE;
pango_font_description_free (desc);
}
g_object_set (webkit_settings, webkit_pref, normalize_font_size (size), NULL);
g_free (value);
}
static void
webkit_pref_callback_font_family (GSettings *settings,
char *key,
gpointer data)
{
char *webkit_pref = data;
char *value = NULL;
char *schema = NULL;
g_object_get (settings, "schema", &schema, NULL);
/* If we are changing a GNOME font value and we are not using GNOME fonts in
* Epiphany, return. */
if (g_strcmp0 (schema, EPHY_PREFS_WEB_SCHEMA) != 0 &&
g_settings_get_boolean (EPHY_SETTINGS_WEB, EPHY_PREFS_WEB_USE_GNOME_FONTS) != TRUE)
{
g_free (schema);
return;
}
g_free (schema);
value = g_settings_get_string (settings, key);
if (value) {
PangoFontDescription* desc;
const char *family = NULL;
desc = pango_font_description_from_string (value);
family = pango_font_description_get_family (desc);
g_object_set (webkit_settings, webkit_pref, family, NULL);
pango_font_description_free (desc);
}
g_free (value);
}
#ifndef HAVE_WEBKIT2
/* Part of this code taken from libsoup (soup-session.c) */
static gchar *
build_accept_languages_header (GArray *languages)
{
gchar **langs = NULL;
gchar *langs_str = NULL;
gint delta;
gint i;
g_return_val_if_fail (languages != NULL, NULL);
/* Calculate deltas for the quality values. */
if (languages->len < 10)
delta = 10;
else if (languages->len < 20)
delta = 5;
else
delta = 1;
/* Set quality values for each language. */
langs = (gchar **)languages->data;
for (i = 0; langs[i] != NULL; i++) {
gchar *lang = (gchar *)langs[i];
gint quality = 100 - i * delta;
if (quality > 0 && quality < 100) {
gchar buf[8];
g_ascii_formatd (buf, 8, "%.2f", quality / 100.0);
langs[i] = g_strdup_printf ("%s;q=%s", lang, buf);
} else {
/* Just dup the string in this case. */
langs[i] = g_strdup (lang);
}
g_free (lang);
}
/* Get the result string */
if (languages->len > 0)
langs_str = g_strjoinv (",", langs);
return langs_str;
}
#endif
/* Based on Christian Persch's code from gecko backend of epiphany
(old transform_accept_languages_list() function) */
static void
webkit_pref_callback_accept_languages (GSettings *settings,
char *key,
gpointer data)
{
#ifndef HAVE_WEBKIT2
SoupSession *session;
char *webkit_pref = data;
char *langs_str;
#endif
GArray *array;
char **languages;
int i;
languages = g_settings_get_strv (settings, key);
array = g_array_new (TRUE, FALSE, sizeof (char *));
for (i = 0; languages[i]; i++) {
if (!g_strcmp0 (languages[i], "system")) {
ephy_langs_append_languages (array);
} else if (languages[i][0] != '\0') {
char *str = g_ascii_strdown (languages[i], -1);
g_array_append_val (array, str);
}
}
ephy_langs_sanitise (array);
#ifdef HAVE_WEBKIT2
webkit_web_context_set_preferred_languages (webkit_web_context_get_default (),
(const char * const *)array->data);
#else
langs_str = build_accept_languages_header (array);
/* Update Soup session */
session = webkit_get_default_session ();
g_object_set (G_OBJECT (session), webkit_pref, langs_str, NULL);
g_free (langs_str);
#endif
g_strfreev (languages);
g_array_free (array, TRUE);
}
#ifdef HAVE_WEBKIT2
void
ephy_embed_prefs_set_cookie_accept_policy (WebKitCookieManager *cookie_manager,
const char *settings_policy)
{
WebKitCookieAcceptPolicy policy;
if (g_str_equal (settings_policy, "never"))
policy = WEBKIT_COOKIE_POLICY_ACCEPT_NEVER;
else if (g_str_equal (settings_policy, "always"))
policy = WEBKIT_COOKIE_POLICY_ACCEPT_ALWAYS;
else if (g_str_equal (settings_policy, "no-third-party"))
policy = WEBKIT_COOKIE_POLICY_ACCEPT_NO_THIRD_PARTY;
else {
g_warn_if_reached ();
return;
}
webkit_cookie_manager_set_accept_policy (cookie_manager, policy);
}
#else
void
ephy_embed_prefs_set_cookie_jar_policy (SoupCookieJar *jar,
const char *settings_policy)
{
SoupCookieJarAcceptPolicy policy;
g_return_if_fail (SOUP_IS_COOKIE_JAR (jar));
g_return_if_fail (settings_policy != NULL);
if (g_str_equal (settings_policy, "never"))
policy = SOUP_COOKIE_JAR_ACCEPT_NEVER;
else if (g_str_equal (settings_policy, "always"))
policy = SOUP_COOKIE_JAR_ACCEPT_ALWAYS;
else if (g_str_equal (settings_policy, "no-third-party"))
policy = SOUP_COOKIE_JAR_ACCEPT_NO_THIRD_PARTY;
else {
g_warn_if_reached ();
return;
}
g_object_set (G_OBJECT (jar), SOUP_COOKIE_JAR_ACCEPT_POLICY, policy, NULL);
}
#endif
static void
webkit_pref_callback_cookie_accept_policy (GSettings *settings,
char *key,
gpointer data)
{
#ifdef HAVE_WEBKIT2
WebKitCookieManager *cookie_manager;
char *value;
value = g_settings_get_string (settings, key);
if (!value)
return;
cookie_manager = webkit_web_context_get_cookie_manager (webkit_web_context_get_default ());
ephy_embed_prefs_set_cookie_accept_policy (cookie_manager, value);
g_free (value);
#else
SoupSession *session;
char *value = NULL;
value = g_settings_get_string (settings, key);
if (value) {
SoupSessionFeature *jar;
session = webkit_get_default_session ();
jar = soup_session_get_feature (session, SOUP_TYPE_COOKIE_JAR);
if (!jar) {
g_free (value);
return;
}
ephy_embed_prefs_set_cookie_jar_policy (SOUP_COOKIE_JAR (jar), value);
}
g_free (value);
#endif
}
static void
webkit_pref_callback_gnome_fonts (GSettings *ephy_settings,
char *key,
gpointer data)
{
GSettings *settings;
if (g_settings_get_boolean (ephy_settings, key)) {
g_object_set (webkit_settings,
"default-font-family", "serif",
"sans-serif-font-family", "sans-serif",
"monospace-font-family", "monospace",
"default-font-size", normalize_font_size (12),
"default-monospace-font-size", normalize_font_size (10),
NULL);
} else {
/* Sync with Epiphany values */
settings = ephy_settings;
webkit_pref_callback_font_size (settings, EPHY_PREFS_WEB_SERIF_FONT,
"default-font-size");
webkit_pref_callback_font_size (settings, EPHY_PREFS_WEB_MONOSPACE_FONT,
"default-monospace-font-size");
webkit_pref_callback_font_family (settings, EPHY_PREFS_WEB_SERIF_FONT,
"default-font-family");
webkit_pref_callback_font_family (settings, EPHY_PREFS_WEB_SANS_SERIF_FONT,
"sans-serif-font-family");
webkit_pref_callback_font_family (settings, EPHY_PREFS_WEB_MONOSPACE_FONT,
"monospace-font-family");
webkit_pref_callback_font_family (settings, EPHY_PREFS_WEB_SERIF_FONT,
"serif-font-family");
}
}
static char **
normalize_languages (char **languages)
{
int i;
GPtrArray *langs;
langs = g_ptr_array_new ();
for (i = 0; languages && languages[i]; i++) {
if (g_str_equal (languages[i], "system")) {
char **sys_langs = ephy_langs_get_languages ();
int j;
for (j = 0; sys_langs && sys_langs[j]; j++)
g_ptr_array_add (langs, g_strdelimit (g_strdup (sys_langs[i]), "-", '_'));
g_strfreev (sys_langs);
} else
g_ptr_array_add (langs, g_strdelimit (g_strdup (languages[i]), "-", '_'));
}
g_ptr_array_add (langs, NULL);
return (char **)g_ptr_array_free (langs, FALSE);
}
static void
webkit_pref_callback_enable_spell_checking (GSettings *settings,
char *key,
gpointer data)
{
#ifdef HAVE_WEBKIT2
WebKitWebContext *web_context = NULL;
#endif
gboolean value = FALSE;
char **languages = NULL;
char *langs = NULL;
value = g_settings_get_boolean (settings, key);
if (value) {
char **normalized;
languages = g_settings_get_strv (settings, EPHY_PREFS_WEB_LANGUAGE);
normalized = normalize_languages (languages);
g_strfreev (languages);
languages = normalized;
langs = g_strjoinv (",", languages);
}
#ifdef HAVE_WEBKIT2
web_context = webkit_web_context_get_default ();
webkit_web_context_set_spell_checking_enabled (web_context, value);
webkit_web_context_set_spell_checking_languages (web_context, languages);
#else
g_object_set (webkit_settings, "enable-spell-checking", value, NULL);
g_object_set (webkit_settings, "spell-checking-languages", langs, NULL);
#endif
g_free (langs);
g_strfreev (languages);
}
static const PrefData webkit_pref_entries[] =
{
/* Epiphany font settings */
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_SERIF_FONT,
"default-font-size",
webkit_pref_callback_font_size },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_MONOSPACE_FONT,
"default-monospace-font-size",
webkit_pref_callback_font_size },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_SERIF_FONT,
"default-font-family",
webkit_pref_callback_font_family },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_SANS_SERIF_FONT,
"sans-serif-font-family",
webkit_pref_callback_font_family },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_MONOSPACE_FONT,
"monospace-font-family",
webkit_pref_callback_font_family },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_SERIF_FONT,
"serif-font-family",
webkit_pref_callback_font_family },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_USE_GNOME_FONTS,
NULL,
webkit_pref_callback_gnome_fonts },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_ENABLE_SPELL_CHECKING,
NULL,
webkit_pref_callback_enable_spell_checking },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_ENABLE_USER_CSS,
"user-stylesheet-uri",
webkit_pref_callback_user_stylesheet },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_LANGUAGE,
"accept-language",
webkit_pref_callback_accept_languages },
{ EPHY_PREFS_SCHEMA,
EPHY_PREFS_USER_AGENT,
"user-agent",
webkit_pref_callback_user_agent },
{ EPHY_PREFS_WEB_SCHEMA,
EPHY_PREFS_WEB_COOKIES_POLICY,
"accept-policy",
webkit_pref_callback_cookie_accept_policy },
};
#ifdef HAVE_WEBKIT2
static void
ephy_embed_prefs_apply (EphyEmbed *embed, WebKitSettings *settings)
#else
static void
ephy_embed_prefs_apply (EphyEmbed *embed, WebKitWebSettings *settings)
#endif
{
webkit_web_view_set_settings (EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed),
settings);
}
void
ephy_embed_prefs_init (void)
{
int i;
#ifdef HAVE_WEBKIT2
webkit_settings = webkit_settings_new_with_settings ("enable-developer-extras", TRUE,
"enable-fullscreen", TRUE,
"enable-site-specific-quirks", TRUE,
NULL);
#else
webkit_settings = webkit_web_settings_new ();
/* Hardcoded settings */
g_object_set (webkit_settings,
"enable-default-context-menu", FALSE,
"enable-site-specific-quirks", TRUE,
"enable-page-cache", TRUE,
"enable-developer-extras", TRUE,
"enable-fullscreen", TRUE,
NULL);
#endif
for (i = 0; i < G_N_ELEMENTS (webkit_pref_entries); i++) {
GSettings *settings;
char *key;
settings = ephy_settings_get (webkit_pref_entries[i].schema);
key = g_strconcat ("changed::", webkit_pref_entries[i].key, NULL);
webkit_pref_entries[i].callback (settings,
webkit_pref_entries[i].key,
webkit_pref_entries[i].webkit_pref);
g_signal_connect (settings, key,
G_CALLBACK (webkit_pref_entries[i].callback),
webkit_pref_entries[i].webkit_pref);
g_free (key);
}
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_ENABLE_JAVASCRIPT,
webkit_settings, ENABLE_SCRIPTS_SETTING,
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_MAIN,
EPHY_PREFS_ENABLE_CARET_BROWSING,
webkit_settings, "enable-caret-browsing",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_ENABLE_POPUPS,
webkit_settings, "javascript-can-open-windows-automatically",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_ENABLE_PLUGINS,
webkit_settings, "enable-plugins",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_FONT_MIN_SIZE,
webkit_settings, "minimum-font-size",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_DEFAULT_ENCODING,
webkit_settings, DEFAULT_ENCODING_SETTING,
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_ENABLE_WEBGL,
webkit_settings, "enable-webgl",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_WEB,
EPHY_PREFS_WEB_ENABLE_WEBAUDIO,
webkit_settings, "enable-webaudio",
G_SETTINGS_BIND_GET);
g_settings_bind (EPHY_SETTINGS_MAIN,
EPHY_PREFS_ENABLE_SMOOTH_SCROLLING,
webkit_settings, "enable-smooth-scrolling",
G_SETTINGS_BIND_GET);
}
void
ephy_embed_prefs_shutdown (void)
{
g_object_unref (webkit_settings);
}
void
ephy_embed_prefs_add_embed (EphyEmbed *embed)
{
ephy_embed_prefs_apply (embed, webkit_settings);
}