1675 lines
59 KiB
C
1675 lines
59 KiB
C
/*
|
|
Copyright (C) 2008-2013 Christian Dywan <christian@twotoasts.de>
|
|
Copyright (C) 2011 Peter Hatina <phatina@redhat.com>
|
|
|
|
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.1 of the License, or (at your option) any later version.
|
|
|
|
See the file COPYING for the full license text.
|
|
*/
|
|
|
|
#include "midori-websettings.h"
|
|
|
|
#include "midori-app.h"
|
|
#include "midori-extension.h"
|
|
#include "sokoke.h"
|
|
#include <midori/midori-core.h> /* Vala API */
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <glib/gstdio.h>
|
|
#include <string.h>
|
|
|
|
#include <config.h>
|
|
#if HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#if defined (G_OS_UNIX)
|
|
#include <sys/utsname.h>
|
|
#endif
|
|
#if defined(__FreeBSD__)
|
|
#include <sys/types.h>
|
|
#include <sys/sysctl.h>
|
|
#endif
|
|
|
|
#if defined (G_OS_WIN32)
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_WEBKIT2
|
|
#define WEB_SETTINGS_STRING(x) "WebKitSettings::"x""
|
|
#else
|
|
#define WEB_SETTINGS_STRING(x) "WebKitWebSettings::"x""
|
|
#endif
|
|
|
|
struct _MidoriWebSettings
|
|
{
|
|
MidoriSettings parent_instance;
|
|
|
|
MidoriToolbarStyle toolbar_style : 3;
|
|
MidoriStartup load_on_startup : 2;
|
|
MidoriNewTabType new_tab_type: 3;
|
|
MidoriPreferredEncoding preferred_encoding : 3;
|
|
gint close_buttons_left;
|
|
MidoriNewPage open_new_pages_in : 2;
|
|
gboolean first_party_cookies_only : 1;
|
|
MidoriProxy proxy_type : 2;
|
|
MidoriIdentity identify_as : 3;
|
|
|
|
gchar* http_accept_language;
|
|
gchar* accept;
|
|
gchar* ident_string;
|
|
|
|
gint clear_private_data;
|
|
gchar* clear_data;
|
|
gchar* site_data_rules;
|
|
gboolean enforce_font_family;
|
|
gchar* user_stylesheet_uri;
|
|
gchar* user_stylesheet_uri_cached;
|
|
GHashTable* user_stylesheets;
|
|
gboolean print_without_dialog;
|
|
};
|
|
|
|
struct _MidoriWebSettingsClass
|
|
{
|
|
MidoriSettingsClass parent_class;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MidoriWebSettings, midori_web_settings, MIDORI_TYPE_SETTINGS);
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
|
|
PROP_TOOLBAR_STYLE,
|
|
|
|
PROP_LOAD_ON_STARTUP,
|
|
PROP_NEW_TAB,
|
|
PROP_PREFERRED_ENCODING,
|
|
|
|
PROP_CLOSE_BUTTONS_LEFT,
|
|
PROP_OPEN_NEW_PAGES_IN,
|
|
PROP_ENABLE_FULLSCREEN,
|
|
|
|
PROP_ENABLE_PLUGINS,
|
|
PROP_ENABLE_PAGE_CACHE,
|
|
|
|
PROP_PROXY_TYPE,
|
|
PROP_IDENTIFY_AS,
|
|
PROP_USER_AGENT,
|
|
PROP_PREFERRED_LANGUAGES,
|
|
|
|
PROP_SITE_DATA_RULES,
|
|
PROP_ENABLE_DNS_PREFETCHING,
|
|
PROP_ENFORCE_FONT_FAMILY,
|
|
PROP_USER_STYLESHEET_URI,
|
|
PROP_PRINT_WITHOUT_DIALOG,
|
|
};
|
|
|
|
GType
|
|
midori_startup_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_STARTUP_BLANK_PAGE, "MIDORI_STARTUP_BLANK_PAGE", N_("Show Speed Dial") },
|
|
{ MIDORI_STARTUP_HOMEPAGE, "MIDORI_STARTUP_HOMEPAGE", N_("Show Homepage") },
|
|
{ MIDORI_STARTUP_LAST_OPEN_PAGES, "MIDORI_STARTUP_LAST_OPEN_PAGES", N_("Show last open tabs") },
|
|
{ MIDORI_STARTUP_DELAYED_PAGES, "MIDORI_STARTUP_DELAYED_PAGES", N_("Show last tabs without loading") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriStartup", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_newtab_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_NEWTAB_BLANK_PAGE, "MIDORI_NEWTAB_BLANK_PAGE", N_("Show Blank Page") },
|
|
{ MIDORI_NEWTAB_HOMEPAGE, "MIDORI_NEWTAB_HOMEPAGE", N_("Show Homepage") },
|
|
{ MIDORI_NEWTAB_SEARCH, "MIDORI_NEWTAB_SEARCH", N_("Show default Search Engine") },
|
|
{ MIDORI_NEWTAB_SPEED_DIAL, "MIDORI_NEWTAB_SPEED_DIAL", N_("Show Speed Dial") },
|
|
{ MIDORI_NEWTAB_CUSTOM, "MIDORI_NEWTAB_CUSTOM", N_("Show custom page") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriNewTabType", values);
|
|
};
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_preferred_encoding_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_ENCODING_CHINESE, "MIDORI_ENCODING_CHINESE", N_("Chinese Traditional (BIG5)") },
|
|
{ MIDORI_ENCODING_CHINESE_SIMPLIFIED, "MIDORI_ENCODING_CHINESE_SIMPLIFIED", N_("Chinese Simplified (GB18030)") },
|
|
{ MIDORI_ENCODING_JAPANESE, "MIDORI_ENCODING_JAPANESE", N_("Japanese (SHIFT_JIS)") },
|
|
{ MIDORI_ENCODING_KOREAN, "MIDORI_ENCODING_KOREAN", N_("Korean (EUC-KR)") },
|
|
{ MIDORI_ENCODING_RUSSIAN, "MIDORI_ENCODING_RUSSIAN", N_("Russian (KOI8-R)") },
|
|
{ MIDORI_ENCODING_UNICODE, "MIDORI_ENCODING_UNICODE", N_("Unicode (UTF-8)") },
|
|
{ MIDORI_ENCODING_WESTERN, "MIDORI_ENCODING_WESTERN", N_("Western (ISO-8859-1)") },
|
|
{ MIDORI_ENCODING_CUSTOM, "MIDORI_ENCODING_CUSTOM", N_("Custom…") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriPreferredEncoding", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_new_page_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_NEW_PAGE_TAB, "MIDORI_NEW_PAGE_TAB", N_("New tab") },
|
|
{ MIDORI_NEW_PAGE_WINDOW, "MIDORI_NEW_PAGE_WINDOW", N_("New window") },
|
|
{ MIDORI_NEW_PAGE_CURRENT, "MIDORI_NEW_PAGE_CURRENT", N_("Current tab") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriNewPage", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_toolbar_style_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_TOOLBAR_DEFAULT, "MIDORI_TOOLBAR_DEFAULT", N_("Default") },
|
|
{ MIDORI_TOOLBAR_ICONS, "MIDORI_TOOLBAR_ICONS", N_("Icons") },
|
|
{ MIDORI_TOOLBAR_SMALL_ICONS, "MIDORI_TOOLBAR_SMALL_ICONS", N_("Small icons") },
|
|
{ MIDORI_TOOLBAR_TEXT, "MIDORI_TOOLBAR_TEXT", N_("Text") },
|
|
{ MIDORI_TOOLBAR_BOTH, "MIDORI_TOOLBAR_BOTH", N_("Icons and text") },
|
|
{ MIDORI_TOOLBAR_BOTH_HORIZ, "MIDORI_TOOLBAR_BOTH_HORIZ", N_("Text beside icons") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriToolbarStyle", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_proxy_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_PROXY_AUTOMATIC, "MIDORI_PROXY_AUTOMATIC", N_("Automatic (GNOME or environment)") },
|
|
{ MIDORI_PROXY_HTTP, "MIDORI_PROXY_HTTP", N_("HTTP proxy server") },
|
|
{ MIDORI_PROXY_NONE, "MIDORI_PROXY_NONE", N_("No proxy server") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriProxy", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
GType
|
|
midori_identity_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
if (!type)
|
|
{
|
|
static const GEnumValue values[] = {
|
|
{ MIDORI_IDENT_MIDORI, "MIDORI_IDENT_MIDORI", N_("_Automatic") },
|
|
{ MIDORI_IDENT_GENUINE, "MIDORI_IDENT_GENUINE", N_("Midori") },
|
|
{ MIDORI_IDENT_CHROME, "MIDORI_IDENT_CHROME", N_("Chrome") },
|
|
{ MIDORI_IDENT_SAFARI, "MIDORI_IDENT_SAFARI", N_("Safari") },
|
|
{ MIDORI_IDENT_IPHONE, "MIDORI_IDENT_IPHONE", N_("iPhone") },
|
|
{ MIDORI_IDENT_FIREFOX, "MIDORI_IDENT_FIREFOX", N_("Firefox") },
|
|
{ MIDORI_IDENT_EXPLORER, "MIDORI_IDENT_EXPLORER", N_("Internet Explorer") },
|
|
{ MIDORI_IDENT_CUSTOM, "MIDORI_IDENT_CUSTOM", N_("Custom…") },
|
|
{ 0, NULL, NULL }
|
|
};
|
|
type = g_enum_register_static ("MidoriIdentity", values);
|
|
}
|
|
return type;
|
|
}
|
|
|
|
static void
|
|
midori_web_settings_finalize (GObject* object);
|
|
|
|
static void
|
|
midori_web_settings_set_property (GObject* object,
|
|
guint prop_id,
|
|
const GValue* value,
|
|
GParamSpec* pspec);
|
|
|
|
static void
|
|
midori_web_settings_get_property (GObject* object,
|
|
guint prop_id,
|
|
GValue* value,
|
|
GParamSpec* pspec);
|
|
|
|
/**
|
|
* midori_web_settings_low_memory_profile:
|
|
*
|
|
* Determines if the system has a relatively small amount of memory.
|
|
*
|
|
* Returns: %TRUE if there is relatively little memory available
|
|
**/
|
|
static gboolean
|
|
midori_web_settings_low_memory_profile ()
|
|
{
|
|
#ifdef _WIN32
|
|
/* See http://msdn.microsoft.com/en-us/library/windows/desktop/aa366589(v=vs.85).aspx */
|
|
MEMORYSTATUSEX mem;
|
|
mem.dwLength = sizeof (mem);
|
|
if (GlobalMemoryStatusEx (&mem))
|
|
return mem.ullTotalPhys / 1024 / 1024 < 352;
|
|
#elif defined(__FreeBSD__)
|
|
size_t size;
|
|
int mem_total;
|
|
size = sizeof mem_total;
|
|
|
|
sysctlbyname("hw.realmem", &mem_total, &size, NULL, 0);
|
|
|
|
return mem_total / 1048576 < 352;
|
|
#else
|
|
gchar* contents;
|
|
const gchar* total;
|
|
if (g_file_get_contents ("/proc/meminfo", &contents, NULL, NULL)
|
|
&& contents && (total = strstr (contents, "MemTotal:")) && *total)
|
|
{
|
|
const gchar* value = katze_skip_whitespace (total + 9);
|
|
gdouble mem_total = g_ascii_strtoll (value, NULL, 0);
|
|
g_free (contents);
|
|
return mem_total / 1024.0 < 352 + 1;
|
|
}
|
|
g_free (contents);
|
|
#endif
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
midori_web_settings_class_init (MidoriWebSettingsClass* class)
|
|
{
|
|
GObjectClass* gobject_class;
|
|
GParamFlags flags;
|
|
|
|
gobject_class = G_OBJECT_CLASS (class);
|
|
gobject_class->finalize = midori_web_settings_finalize;
|
|
gobject_class->set_property = midori_web_settings_set_property;
|
|
gobject_class->get_property = midori_web_settings_get_property;
|
|
|
|
flags = G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_TOOLBAR_STYLE,
|
|
g_param_spec_enum (
|
|
"toolbar-style",
|
|
"Toolbar Style:",
|
|
_("The style of the toolbar"),
|
|
MIDORI_TYPE_TOOLBAR_STYLE,
|
|
MIDORI_TOOLBAR_DEFAULT,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_LOAD_ON_STARTUP,
|
|
g_param_spec_enum (
|
|
"load-on-startup",
|
|
"When Midori starts:",
|
|
"What to do when Midori starts",
|
|
MIDORI_TYPE_STARTUP,
|
|
MIDORI_STARTUP_LAST_OPEN_PAGES,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_NEW_TAB,
|
|
g_param_spec_enum (
|
|
"new-tab-type",
|
|
"New tab behavior:",
|
|
"What to show in newly opened tabs",
|
|
MIDORI_TYPE_NEWTAB,
|
|
MIDORI_NEWTAB_SPEED_DIAL,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PREFERRED_ENCODING,
|
|
g_param_spec_enum (
|
|
"preferred-encoding",
|
|
"Preferred Encoding",
|
|
"The preferred character encoding",
|
|
MIDORI_TYPE_PREFERRED_ENCODING,
|
|
MIDORI_ENCODING_WESTERN,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:close-buttons-left:
|
|
*
|
|
* Whether to show close buttons on the left side.
|
|
*
|
|
* Since: 0.3.1
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_CLOSE_BUTTONS_LEFT,
|
|
g_param_spec_boolean (
|
|
"close-buttons-left",
|
|
"Close buttons on the left",
|
|
"Whether to show close buttons on the left side",
|
|
FALSE,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_OPEN_NEW_PAGES_IN,
|
|
g_param_spec_enum (
|
|
"open-new-pages-in",
|
|
"Open new pages in:",
|
|
"Where to open new pages",
|
|
MIDORI_TYPE_NEW_PAGE,
|
|
MIDORI_NEW_PAGE_TAB,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_ENABLE_PLUGINS,
|
|
g_param_spec_boolean (
|
|
"enable-plugins",
|
|
"Enable Netscape plugins",
|
|
"Enable embedded Netscape plugin objects",
|
|
TRUE,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_ENABLE_PAGE_CACHE,
|
|
g_param_spec_boolean ("enable-page-cache",
|
|
"Enable page cache",
|
|
"Whether the page cache should be used",
|
|
!midori_web_settings_low_memory_profile (),
|
|
flags));
|
|
|
|
if (g_object_class_find_property (gobject_class, "enable-fullscreen"))
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_ENABLE_FULLSCREEN,
|
|
g_param_spec_boolean (
|
|
"enable-fullscreen",
|
|
"Enable Fullscreen",
|
|
"Allow experimental fullscreen API",
|
|
TRUE,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:proxy-type:
|
|
*
|
|
* The type of proxy server to use.
|
|
*
|
|
* Since: 0.2.5
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PROXY_TYPE,
|
|
g_param_spec_enum (
|
|
"proxy-type",
|
|
"Proxy server",
|
|
"The type of proxy server to use",
|
|
MIDORI_TYPE_PROXY,
|
|
MIDORI_PROXY_AUTOMATIC,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:identify-as:
|
|
*
|
|
* What to identify as to web pages.
|
|
*
|
|
* Since: 0.1.2
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_IDENTIFY_AS,
|
|
g_param_spec_enum (
|
|
"identify-as",
|
|
"Identify as",
|
|
"What to identify as to web pages",
|
|
MIDORI_TYPE_IDENTITY,
|
|
MIDORI_IDENT_MIDORI,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:user-agent:
|
|
*
|
|
* The browser identification string.
|
|
*
|
|
* Since: 0.2.3
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_USER_AGENT,
|
|
g_param_spec_string (
|
|
"user-agent",
|
|
"Identification string",
|
|
"The application identification string",
|
|
NULL,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:preferred-languages:
|
|
*
|
|
* A comma separated list of languages preferred for rendering multilingual
|
|
* webpages and spell checking.
|
|
*
|
|
* Since: 0.2.3
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PREFERRED_LANGUAGES,
|
|
g_param_spec_string (
|
|
"preferred-languages",
|
|
"Preferred languages",
|
|
"A comma separated list of languages",
|
|
NULL,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:site-data-rules:
|
|
*
|
|
* Rules for accepting, denying and preserving cookies and other data.
|
|
* See midori_web_settings_get_site_data_policy() for details.
|
|
*
|
|
* Since: 0.4.4
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_SITE_DATA_RULES,
|
|
g_param_spec_string (
|
|
"site-data-rules",
|
|
"Rules for accepting, denying and preserving cookies and other data",
|
|
"Cookies, HTML5 databases, local storage and application cache blocking",
|
|
NULL,
|
|
flags));
|
|
|
|
/**
|
|
* MidoriWebSettings:enforce-font-family:
|
|
*
|
|
* Whether to enforce user font preferences with an internal stylesheet.
|
|
*
|
|
* Since: 0.4.2
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_ENFORCE_FONT_FAMILY,
|
|
g_param_spec_boolean (
|
|
"enforce-font-family",
|
|
_("Always use my font choices"),
|
|
_("Override fonts picked by websites with user preferences"),
|
|
FALSE,
|
|
flags));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PRINT_WITHOUT_DIALOG,
|
|
g_param_spec_boolean (
|
|
"print-without-dialog",
|
|
"Print without dialog",
|
|
"Print without showing a dialog box",
|
|
FALSE,
|
|
flags));
|
|
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_USER_STYLESHEET_URI,
|
|
g_param_spec_string (
|
|
"user-stylesheet-uri",
|
|
"User stylesheet URI",
|
|
"Load stylesheets from a local URI",
|
|
NULL,
|
|
flags));
|
|
}
|
|
|
|
static void
|
|
notify_default_encoding_cb (GObject* object,
|
|
GParamSpec* pspec)
|
|
{
|
|
MidoriWebSettings* web_settings;
|
|
gchar* string;
|
|
const gchar* encoding;
|
|
|
|
web_settings = MIDORI_WEB_SETTINGS (object);
|
|
|
|
g_object_get (object, pspec->name, &string, NULL);
|
|
encoding = string ? string : "";
|
|
if (!strcmp (encoding, "BIG5"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_CHINESE;
|
|
else if (!strcmp (encoding, "GB18030"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_CHINESE_SIMPLIFIED;
|
|
else if (!strcmp (encoding, "SHIFT_JIS"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_JAPANESE;
|
|
else if (!strcmp (encoding, "EUC-KR"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_KOREAN;
|
|
else if (!strcmp (encoding, "KOI8-R"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_RUSSIAN;
|
|
else if (!strcmp (encoding, "UTF-8"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_UNICODE;
|
|
else if (!strcmp (encoding, "ISO-8859-1"))
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_WESTERN;
|
|
else
|
|
web_settings->preferred_encoding = MIDORI_ENCODING_CUSTOM;
|
|
g_free (string);
|
|
g_object_notify (object, "preferred-encoding");
|
|
}
|
|
|
|
static void
|
|
notify_default_font_family_cb (GObject* object,
|
|
GParamSpec* pspec)
|
|
{
|
|
if (katze_object_get_boolean (object, "enforce-font-family"))
|
|
g_object_set (object, "enforce-font-family", TRUE, NULL);
|
|
}
|
|
static void
|
|
midori_web_settings_init (MidoriWebSettings* web_settings)
|
|
{
|
|
web_settings->user_stylesheet_uri = web_settings->user_stylesheet_uri_cached = NULL;
|
|
web_settings->user_stylesheets = NULL;
|
|
|
|
#if defined (_WIN32) && !GTK_CHECK_VERSION (3, 0, 0)
|
|
/* Try to work-around black borders on native widgets and GTK+2 on Win32 */
|
|
midori_web_settings_add_style (web_settings, "black-widgets-workaround",
|
|
"input[type='checkbox'] { -webkit-appearance: checkbox !important }"
|
|
" input[type='radio'] { -webkit-appearance: radio !important }"
|
|
" * { -webkit-appearance: none !important }");
|
|
#endif
|
|
|
|
g_signal_connect (web_settings, "notify::default-charset",
|
|
G_CALLBACK (notify_default_encoding_cb), NULL);
|
|
g_signal_connect (web_settings, "notify::default-font-family",
|
|
G_CALLBACK (notify_default_font_family_cb), NULL);
|
|
}
|
|
|
|
static void
|
|
midori_web_settings_finalize (GObject* object)
|
|
{
|
|
MidoriWebSettings* web_settings;
|
|
|
|
web_settings = MIDORI_WEB_SETTINGS (object);
|
|
|
|
katze_assign (web_settings->http_accept_language, NULL);
|
|
katze_assign (web_settings->accept, NULL);
|
|
katze_assign (web_settings->ident_string, NULL);
|
|
katze_assign (web_settings->user_stylesheet_uri, NULL);
|
|
katze_assign (web_settings->user_stylesheet_uri_cached, NULL);
|
|
if (web_settings->user_stylesheets != NULL)
|
|
g_hash_table_destroy (web_settings->user_stylesheets);
|
|
|
|
G_OBJECT_CLASS (midori_web_settings_parent_class)->finalize (object);
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_has_plugin_support:
|
|
*
|
|
* Determines if Netscape plugins are supported.
|
|
*
|
|
* Returns: %TRUE if Netscape plugins can be used
|
|
*
|
|
* Since: 0.4.4
|
|
**/
|
|
gboolean
|
|
midori_web_settings_has_plugin_support (void)
|
|
{
|
|
return !midori_debug ("unarmed") && g_strcmp0 (g_getenv ("MOZ_PLUGIN_PATH"), "/");
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_skip_plugin:
|
|
* @path: the path to the plugin file
|
|
*
|
|
* Tests if a plugin is redundant. WebKit sometimes provides
|
|
* duplicate listings of plugins due to library deployment
|
|
* miscellanea.
|
|
*
|
|
* Returns: %TRUE if the passed plugin shouldn't be shown in UI listings.
|
|
*
|
|
* Since: 0.5.1
|
|
**/
|
|
gboolean
|
|
midori_web_settings_skip_plugin (const gchar* path)
|
|
{
|
|
static GHashTable* plugins = NULL;
|
|
gchar* basename = NULL;
|
|
gchar* plugin_path = NULL;
|
|
|
|
if (!path)
|
|
return TRUE;
|
|
|
|
if (!plugins)
|
|
plugins = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
basename = g_path_get_basename (path);
|
|
|
|
plugin_path = g_hash_table_lookup (plugins, basename);
|
|
if (g_strcmp0 (path, plugin_path) == 0)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (plugin_path != NULL)
|
|
{
|
|
g_free (basename);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
g_hash_table_insert (plugins, basename, g_strdup (path));
|
|
|
|
/* Note: do not free basename */
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_get_site_data_policy:
|
|
* @settings: the MidoriWebSettings instance
|
|
* @uri: the URI for which to make the policy decision
|
|
*
|
|
* Tests if @uri may store site data.
|
|
*
|
|
* Returns: a #MidoriSiteDataPolicy
|
|
*
|
|
* Since: 0.4.4
|
|
**/
|
|
MidoriSiteDataPolicy
|
|
midori_web_settings_get_site_data_policy (MidoriWebSettings* settings,
|
|
const gchar* uri)
|
|
{
|
|
MidoriSiteDataPolicy policy = MIDORI_SITE_DATA_UNDETERMINED;
|
|
gchar* hostname;
|
|
const gchar* match;
|
|
|
|
g_return_val_if_fail (MIDORI_IS_WEB_SETTINGS (settings), policy);
|
|
|
|
if (!(settings->site_data_rules && *settings->site_data_rules))
|
|
return policy;
|
|
|
|
/*
|
|
* Values prefixed with "-" are always blocked
|
|
* Values prefixed with "+" are always accepted
|
|
* Values prefixed with "!" are not cleared in Clear Private Data
|
|
* FIXME: "*" is a wildcard
|
|
* FIXME: indicate type of storage the rule applies to
|
|
* FIXME: support matching of the whole URI
|
|
**/
|
|
hostname = midori_uri_parse_hostname (uri, NULL);
|
|
match = strstr (settings->site_data_rules, hostname ? hostname : uri);
|
|
if (match != NULL && match != settings->site_data_rules)
|
|
{
|
|
const gchar* prefix = match - 1;
|
|
if (*prefix == '-')
|
|
policy = MIDORI_SITE_DATA_BLOCK;
|
|
else if (*prefix == '+')
|
|
policy = MIDORI_SITE_DATA_ACCEPT;
|
|
else if (*prefix == '!')
|
|
policy = MIDORI_SITE_DATA_PRESERVE;
|
|
else
|
|
g_warning ("%s: Matched with no prefix '%s'", G_STRFUNC, match);
|
|
}
|
|
g_free (hostname);
|
|
return policy;
|
|
}
|
|
|
|
#if (!HAVE_OSX && defined (G_OS_UNIX)) || defined (G_OS_WIN32)
|
|
static gchar*
|
|
get_sys_name (gchar** architecture)
|
|
{
|
|
static gchar* sys_name = NULL;
|
|
static gchar* sys_architecture = NULL;
|
|
|
|
if (!sys_name)
|
|
{
|
|
#ifdef G_OS_WIN32
|
|
/* 6.1 Win7, 6.0 Vista, 5.1 XP and 5.0 Win2k */
|
|
guint version = g_win32_get_windows_version ();
|
|
sys_name = g_strdup_printf ("NT %d.%d", LOBYTE (version), HIBYTE (version));
|
|
#else
|
|
struct utsname name;
|
|
if (uname (&name) != -1)
|
|
{
|
|
sys_name = g_strdup (name.sysname);
|
|
sys_architecture = g_strdup (name.machine);
|
|
}
|
|
else
|
|
sys_name = "Linux";
|
|
#endif
|
|
}
|
|
|
|
if (architecture != NULL)
|
|
*architecture = sys_architecture;
|
|
return sys_name;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* midori_web_settings_get_system_name:
|
|
* @architecture: (out) (allow-none): location of a string, or %NULL
|
|
* @platform: (out) (allow-none): location of a string, or %NULL
|
|
*
|
|
* Determines the system name, architecture and platform.
|
|
* This function may write a %NULL value to @architecture.
|
|
*
|
|
* Returns: a string
|
|
*
|
|
* Since: 0.4.2
|
|
**/
|
|
const gchar*
|
|
midori_web_settings_get_system_name (gchar** architecture,
|
|
gchar** platform)
|
|
{
|
|
if (architecture != NULL)
|
|
*architecture = NULL;
|
|
|
|
if (platform != NULL)
|
|
*platform =
|
|
#if defined (G_OS_WIN32)
|
|
"Windows";
|
|
#elif defined(GDK_WINDOWING_QUARTZ)
|
|
"Macintosh;";
|
|
#elif defined(GDK_WINDOWING_DIRECTFB)
|
|
"DirectFB;";
|
|
#else
|
|
"X11;";
|
|
#endif
|
|
|
|
return
|
|
#if HAVE_OSX
|
|
"Mac OS X";
|
|
#elif defined (G_OS_UNIX) || defined (G_OS_WIN32)
|
|
get_sys_name (architecture);
|
|
#else
|
|
"Linux";
|
|
#endif
|
|
}
|
|
|
|
static const gchar*
|
|
get_uri_for_new_tab (MidoriWebSettings* web_settings,
|
|
MidoriNewTabType new_tab_type)
|
|
{
|
|
switch (new_tab_type)
|
|
{
|
|
case MIDORI_NEWTAB_BLANK_PAGE:
|
|
return "about:blank";
|
|
case MIDORI_NEWTAB_HOMEPAGE:
|
|
return "about:home";
|
|
case MIDORI_NEWTAB_SEARCH:
|
|
return "about:search";
|
|
case MIDORI_NEWTAB_CUSTOM:
|
|
return midori_settings_get_tabhome (MIDORI_SETTINGS (web_settings));
|
|
case MIDORI_NEWTAB_SPEED_DIAL:
|
|
return "about:dial";
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
static gchar*
|
|
generate_ident_string (MidoriWebSettings* web_settings,
|
|
MidoriIdentity identify_as)
|
|
{
|
|
const gchar* appname = "Midori/"
|
|
G_STRINGIFY (MIDORI_MAJOR_VERSION) "."
|
|
G_STRINGIFY (MIDORI_MINOR_VERSION);
|
|
|
|
const gchar* lang = pango_language_to_string (gtk_get_default_language ());
|
|
gchar* platform;
|
|
const gchar* os = midori_web_settings_get_system_name (NULL, &platform);
|
|
|
|
#ifndef HAVE_WEBKIT2
|
|
const int webcore_major = WEBKIT_USER_AGENT_MAJOR_VERSION;
|
|
const int webcore_minor = WEBKIT_USER_AGENT_MINOR_VERSION;
|
|
#else
|
|
const int webcore_major = 537;
|
|
const int webcore_minor = 32;
|
|
#endif
|
|
|
|
g_object_set (web_settings, "enable-site-specific-quirks",
|
|
identify_as != MIDORI_IDENT_GENUINE, NULL);
|
|
|
|
switch (identify_as)
|
|
{
|
|
case MIDORI_IDENT_GENUINE:
|
|
return g_strdup_printf ("Mozilla/5.0 (%s %s) AppleWebKit/%d.%d+ %s",
|
|
platform, os, webcore_major, webcore_minor, appname);
|
|
case MIDORI_IDENT_MIDORI:
|
|
case MIDORI_IDENT_CHROME:
|
|
return g_strdup_printf ("Mozilla/5.0 (%s %s) AppleWebKit/%d.%d "
|
|
"(KHTML, like Gecko) Chrome/18.0.1025.133 Safari/%d.%d %s",
|
|
platform, os, webcore_major, webcore_minor, webcore_major, webcore_minor, appname);
|
|
case MIDORI_IDENT_SAFARI:
|
|
return g_strdup_printf ("Mozilla/5.0 (Macintosh; U; Intel Mac OS X; %s) "
|
|
"AppleWebKit/%d+ (KHTML, like Gecko) Version/5.0 Safari/%d.%d+ %s",
|
|
lang, webcore_major, webcore_major, webcore_minor, appname);
|
|
case MIDORI_IDENT_IPHONE:
|
|
return g_strdup_printf ("Mozilla/5.0 (iPhone; U; CPU like Mac OS X; %s) "
|
|
"AppleWebKit/532+ (KHTML, like Gecko) Version/3.0 Mobile/1A538b Safari/419.3 %s",
|
|
lang, appname);
|
|
case MIDORI_IDENT_FIREFOX:
|
|
return g_strdup_printf ("Mozilla/5.0 (%s %s; rv:2.0.1) Gecko/20100101 Firefox/4.0.1 %s",
|
|
platform, os, appname);
|
|
case MIDORI_IDENT_EXPLORER:
|
|
return g_strdup_printf ("Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; %s) %s",
|
|
lang, appname);
|
|
default:
|
|
return g_strdup_printf ("%s", appname);
|
|
}
|
|
}
|
|
|
|
#ifndef HAVE_WEBKIT2
|
|
/* Provide a new way for SoupSession to assume an 'Accept-Language'
|
|
string automatically from the return value of g_get_language_names(),
|
|
properly formatted according to RFC2616.
|
|
Copyright (C) 2009 Mario Sanchez Prada <msanchez@igalia.com>
|
|
Copyright (C) 2009 Dan Winship <danw@gnome.org>
|
|
Mostly copied from libSoup 2.29, coding style adjusted */
|
|
|
|
/* Converts a language in POSIX format and to be RFC2616 compliant */
|
|
/* Based on code from epiphany-webkit (ephy_langs_append_languages()) */
|
|
|
|
static gchar *
|
|
sokoke_posix_lang_to_rfc2616 (const gchar *language)
|
|
{
|
|
if (!strchr (language, '.') && !strchr (language, '@') && language[0] != 'C')
|
|
/* change to lowercase and '_' to '-' */
|
|
return g_strdelimit (g_ascii_strdown (language, -1), "_", '-');
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/* Adds a quality value to a string (any value between 0 and 1). */
|
|
static gchar *
|
|
sokoke_add_quality_value (const gchar *str,
|
|
float qvalue)
|
|
{
|
|
if ((qvalue >= 0.0) && (qvalue <= 1.0))
|
|
{
|
|
int qv_int = (qvalue * 1000 + 0.5);
|
|
return g_strdup_printf ("%s;q=%d.%d",
|
|
str, (int) (qv_int / 1000), qv_int % 1000);
|
|
}
|
|
|
|
return g_strdup (str);
|
|
}
|
|
|
|
/* Returns a RFC2616 compliant languages list from system locales */
|
|
static gchar *
|
|
sokoke_accept_languages (const gchar* const * lang_names)
|
|
{
|
|
GString* langs = NULL;
|
|
char *cur_lang = NULL;
|
|
char *prev_lang = NULL;
|
|
float delta;
|
|
int i, n_lang_names;
|
|
|
|
/* Calculate delta for setting the quality values */
|
|
n_lang_names = g_strv_length ((gchar **)lang_names);
|
|
delta = 0.999 / (n_lang_names - 1);
|
|
|
|
/* Build the array of languages */
|
|
langs = g_string_sized_new (n_lang_names);
|
|
for (i = 0; lang_names[i] != NULL; i++)
|
|
{
|
|
cur_lang = sokoke_posix_lang_to_rfc2616 (lang_names[i]);
|
|
|
|
/* Apart from getting a valid RFC2616 compliant
|
|
language, also get rid of extra variants */
|
|
if (cur_lang && (!prev_lang ||
|
|
(!strcmp (prev_lang, cur_lang) || !strstr (prev_lang, cur_lang))))
|
|
{
|
|
|
|
gchar *qv_lang = NULL;
|
|
|
|
/* Save reference for further comparison */
|
|
prev_lang = cur_lang;
|
|
|
|
/* Add the quality value and append it */
|
|
qv_lang = sokoke_add_quality_value (cur_lang, 1 - i * delta);
|
|
if (langs->len > 0)
|
|
g_string_append_c (langs, ',');
|
|
g_string_append (langs, qv_lang);
|
|
}
|
|
}
|
|
|
|
/* Fallback: add "en" if list is empty */
|
|
if (langs->len == 0)
|
|
g_string_append (langs, "en");
|
|
|
|
return g_string_free (langs, FALSE);
|
|
}
|
|
|
|
|
|
static void
|
|
midori_web_settings_update_accept_language (MidoriWebSettings* settings)
|
|
{
|
|
gchar* languages = settings->http_accept_language;
|
|
/* Empty, use the system locales */
|
|
if (!(languages && *languages))
|
|
katze_assign (settings->accept, sokoke_accept_languages (g_get_language_names ()));
|
|
/* No =, no ., looks like a list of language names */
|
|
else if (!(strchr (languages, '=') && strchr (languages, '.')))
|
|
{
|
|
gchar ** lang_names = g_strsplit_set (languages, ",; ", -1);
|
|
katze_assign (settings->accept, sokoke_accept_languages ((const gchar* const *)lang_names));
|
|
g_strfreev (lang_names);
|
|
}
|
|
/* Presumably a well formatted list including priorities */
|
|
else
|
|
katze_assign (settings->accept, g_strdup (languages));
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_get_accept_language:
|
|
*
|
|
* Returns the value of the accept-language header to send to web servers
|
|
*
|
|
* Returns: the accept-language string
|
|
**/
|
|
const gchar*
|
|
midori_web_settings_get_accept_language (MidoriWebSettings* settings)
|
|
{
|
|
if (!settings->accept)
|
|
midori_web_settings_update_accept_language (settings);
|
|
return settings->accept;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
midori_web_settings_process_stylesheets (MidoriWebSettings* settings,
|
|
gint delta_len);
|
|
|
|
static void
|
|
base64_space_pad (gchar* base64,
|
|
guint len);
|
|
|
|
static void
|
|
midori_web_settings_set_property (GObject* object,
|
|
guint prop_id,
|
|
const GValue* value,
|
|
GParamSpec* pspec)
|
|
{
|
|
MidoriWebSettings* web_settings = MIDORI_WEB_SETTINGS (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_TOOLBAR_STYLE:
|
|
web_settings->toolbar_style = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_LOAD_ON_STARTUP:
|
|
web_settings->load_on_startup = g_value_get_enum (value);
|
|
break;
|
|
case PROP_PREFERRED_ENCODING:
|
|
web_settings->preferred_encoding = g_value_get_enum (value);
|
|
switch (web_settings->preferred_encoding)
|
|
{
|
|
case MIDORI_ENCODING_CHINESE:
|
|
g_object_set (object, "default-charset", "BIG5", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_CHINESE_SIMPLIFIED:
|
|
g_object_set (object, "default-charset", "GB18030", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_JAPANESE:
|
|
g_object_set (object, "default-charset", "SHIFT_JIS", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_KOREAN:
|
|
g_object_set (object, "default-charset", "EUC-KR", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_RUSSIAN:
|
|
g_object_set (object, "default-charset", "KOI8-R", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_UNICODE:
|
|
g_object_set (object, "default-charset", "UTF-8", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_WESTERN:
|
|
g_object_set (object, "default-charset", "ISO-8859-1", NULL);
|
|
break;
|
|
case MIDORI_ENCODING_CUSTOM:
|
|
g_object_set (object, "default-charset", "", NULL);
|
|
}
|
|
break;
|
|
|
|
case PROP_OPEN_NEW_PAGES_IN:
|
|
web_settings->open_new_pages_in = g_value_get_enum (value);
|
|
break;
|
|
|
|
case PROP_ENABLE_PLUGINS:
|
|
g_object_set (web_settings,
|
|
WEB_SETTINGS_STRING ("enable-plugins"), g_value_get_boolean (value),
|
|
#ifdef HAVE_WEBKIT2
|
|
"enable-java", g_value_get_boolean (value),
|
|
#else
|
|
"enable-java-applet", g_value_get_boolean (value),
|
|
#endif
|
|
NULL);
|
|
break;
|
|
case PROP_ENABLE_PAGE_CACHE:
|
|
g_object_set (web_settings, WEB_SETTINGS_STRING ("enable-page-cache"),
|
|
g_value_get_boolean (value), NULL);
|
|
break;
|
|
|
|
case PROP_PROXY_TYPE:
|
|
web_settings->proxy_type = g_value_get_enum (value);
|
|
break;
|
|
case PROP_IDENTIFY_AS:
|
|
web_settings->identify_as = g_value_get_enum (value);
|
|
if (web_settings->identify_as != MIDORI_IDENT_CUSTOM)
|
|
{
|
|
gchar* string = generate_ident_string (web_settings, web_settings->identify_as);
|
|
katze_assign (web_settings->ident_string, string);
|
|
g_object_set (web_settings, "user-agent", string, NULL);
|
|
}
|
|
break;
|
|
case PROP_USER_AGENT:
|
|
if (web_settings->identify_as == MIDORI_IDENT_CUSTOM)
|
|
katze_assign (web_settings->ident_string, g_value_dup_string (value));
|
|
g_object_set (web_settings, WEB_SETTINGS_STRING ("user-agent"),
|
|
web_settings->ident_string, NULL);
|
|
break;
|
|
case PROP_NEW_TAB:
|
|
web_settings->new_tab_type = g_value_get_enum (value);
|
|
const gchar* tabhome = get_uri_for_new_tab (web_settings, web_settings->new_tab_type);
|
|
midori_settings_set_tabhome (MIDORI_SETTINGS (web_settings), tabhome);
|
|
break;
|
|
case PROP_PREFERRED_LANGUAGES:
|
|
katze_assign (web_settings->http_accept_language, g_value_dup_string (value));
|
|
#ifdef HAVE_WEBKIT2
|
|
WebKitWebContext* context = webkit_web_context_get_default ();
|
|
gchar** languages = web_settings->http_accept_language
|
|
? g_strsplit_set (web_settings->http_accept_language, ",; ", -1)
|
|
: g_strdupv ((gchar**)g_get_language_names ());
|
|
webkit_web_context_set_preferred_languages (context, (const gchar* const*)languages);
|
|
webkit_web_context_set_spell_checking_languages (context, (const gchar* const*)languages);
|
|
g_strfreev (languages);
|
|
#else
|
|
g_object_set (web_settings, "spell-checking-languages",
|
|
web_settings->http_accept_language, NULL);
|
|
midori_web_settings_update_accept_language (web_settings);
|
|
#endif
|
|
break;
|
|
case PROP_SITE_DATA_RULES:
|
|
katze_assign (web_settings->site_data_rules, g_value_dup_string (value));
|
|
break;
|
|
case PROP_ENFORCE_FONT_FAMILY:
|
|
if ((web_settings->enforce_font_family = g_value_get_boolean (value)))
|
|
{
|
|
gchar* font_family = katze_object_get_string (web_settings,
|
|
"default-font-family");
|
|
gchar* monospace = katze_object_get_string (web_settings,
|
|
"monospace-font-family");
|
|
gchar* css = g_strdup_printf ("body * { font-family: %s !important; } "
|
|
"code, code *, pre, pre *, blockquote, blockquote *, "
|
|
"input, textarea { font-family: %s !important; }",
|
|
font_family, monospace);
|
|
midori_web_settings_add_style (web_settings, "enforce-font-family", css);
|
|
g_free (font_family);
|
|
g_free (monospace);
|
|
g_free (css);
|
|
}
|
|
else
|
|
midori_web_settings_remove_style (web_settings, "enforce-font-family");
|
|
break;
|
|
case PROP_ENABLE_FULLSCREEN:
|
|
g_object_set (web_settings, WEB_SETTINGS_STRING ("enable-fullscreen"),
|
|
g_value_get_boolean (value), NULL);
|
|
break;
|
|
case PROP_USER_STYLESHEET_URI:
|
|
{
|
|
gint old_len = web_settings->user_stylesheet_uri_cached
|
|
? strlen (web_settings->user_stylesheet_uri_cached) : 0;
|
|
gint new_len = 0;
|
|
if ((web_settings->user_stylesheet_uri = g_value_dup_string (value)))
|
|
{
|
|
gchar* import = g_strdup_printf ("@import url(\"%s\");",
|
|
web_settings->user_stylesheet_uri);
|
|
gchar* encoded = g_base64_encode ((const guchar*)import, strlen (import));
|
|
new_len = strlen (encoded);
|
|
base64_space_pad (encoded, new_len);
|
|
g_free (import);
|
|
katze_assign (web_settings->user_stylesheet_uri_cached, encoded);
|
|
}
|
|
/* Make original user-stylesheet-uri available to main.c */
|
|
g_object_set_data (G_OBJECT (web_settings), "user-stylesheet-uri",
|
|
web_settings->user_stylesheet_uri);
|
|
midori_web_settings_process_stylesheets (web_settings, new_len - old_len);
|
|
}
|
|
break;
|
|
case PROP_PRINT_WITHOUT_DIALOG:
|
|
web_settings->print_without_dialog = g_value_get_boolean(value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
midori_web_settings_get_property (GObject* object,
|
|
guint prop_id,
|
|
GValue* value,
|
|
GParamSpec* pspec)
|
|
{
|
|
MidoriWebSettings* web_settings = MIDORI_WEB_SETTINGS (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_TOOLBAR_STYLE:
|
|
g_value_set_enum (value, web_settings->toolbar_style);
|
|
break;
|
|
|
|
case PROP_LOAD_ON_STARTUP:
|
|
g_value_set_enum (value, web_settings->load_on_startup);
|
|
break;
|
|
case PROP_PREFERRED_ENCODING:
|
|
g_value_set_enum (value, web_settings->preferred_encoding);
|
|
break;
|
|
|
|
case PROP_CLOSE_BUTTONS_LEFT:
|
|
#if HAVE_OSX
|
|
g_value_set_boolean (value, TRUE);
|
|
#elif defined (G_OS_WIN32)
|
|
g_value_set_boolean (value, FALSE);
|
|
#else
|
|
if (!web_settings->close_buttons_left)
|
|
{
|
|
/* Look for close button in layout specified in index.theme */
|
|
GdkScreen* screen = gdk_screen_get_default ();
|
|
GtkSettings* settings = gtk_settings_get_for_screen (screen);
|
|
gchar* theme = katze_object_get_string (settings, "gtk-theme-name");
|
|
gchar* theme_file = g_build_filename ("themes", theme, "index.theme", NULL);
|
|
gchar* filename = midori_paths_get_data_filename (theme_file, FALSE);
|
|
g_free (theme_file);
|
|
web_settings->close_buttons_left = 1;
|
|
if (g_access (filename, F_OK) != 0)
|
|
katze_assign (filename,
|
|
g_build_filename (g_get_home_dir (), ".themes",
|
|
theme, "index.theme", NULL));
|
|
g_free (theme);
|
|
if (g_access (filename, F_OK) == 0)
|
|
{
|
|
GKeyFile* keyfile = g_key_file_new ();
|
|
gchar* button_layout;
|
|
g_key_file_load_from_file (keyfile, filename, 0, NULL);
|
|
button_layout = g_key_file_get_string (keyfile,
|
|
"X-GNOME-Metatheme", "ButtonLayout", NULL);
|
|
if (button_layout && strstr (button_layout, "close:"))
|
|
web_settings->close_buttons_left = 2;
|
|
g_free (button_layout);
|
|
g_key_file_free (keyfile);
|
|
}
|
|
g_free (filename);
|
|
}
|
|
g_value_set_boolean (value, web_settings->close_buttons_left == 2);
|
|
#endif
|
|
break;
|
|
case PROP_OPEN_NEW_PAGES_IN:
|
|
g_value_set_enum (value, web_settings->open_new_pages_in);
|
|
break;
|
|
|
|
case PROP_ENABLE_PLUGINS:
|
|
g_value_set_boolean (value, katze_object_get_boolean (web_settings,
|
|
WEB_SETTINGS_STRING ("enable-plugins")));
|
|
break;
|
|
case PROP_ENABLE_PAGE_CACHE:
|
|
g_value_set_boolean (value, katze_object_get_boolean (web_settings,
|
|
WEB_SETTINGS_STRING ("enable-page-cache")));
|
|
break;
|
|
|
|
case PROP_PROXY_TYPE:
|
|
g_value_set_enum (value, web_settings->proxy_type);
|
|
break;
|
|
case PROP_IDENTIFY_AS:
|
|
g_value_set_enum (value, web_settings->identify_as);
|
|
break;
|
|
case PROP_USER_AGENT:
|
|
if (!g_strcmp0 (web_settings->ident_string, ""))
|
|
{
|
|
gchar* string = generate_ident_string (web_settings, web_settings->identify_as);
|
|
katze_assign (web_settings->ident_string, string);
|
|
}
|
|
g_value_set_string (value, web_settings->ident_string);
|
|
break;
|
|
case PROP_NEW_TAB:
|
|
g_value_set_enum (value, web_settings->new_tab_type);
|
|
break;
|
|
case PROP_PREFERRED_LANGUAGES:
|
|
g_value_set_string (value, web_settings->http_accept_language);
|
|
break;
|
|
case PROP_SITE_DATA_RULES:
|
|
g_value_set_string (value, web_settings->site_data_rules);
|
|
break;
|
|
case PROP_ENFORCE_FONT_FAMILY:
|
|
g_value_set_boolean (value, web_settings->enforce_font_family);
|
|
break;
|
|
case PROP_ENABLE_FULLSCREEN:
|
|
g_value_set_boolean (value, katze_object_get_boolean (web_settings,
|
|
WEB_SETTINGS_STRING ("enable-fullscreen")));
|
|
break;
|
|
case PROP_USER_STYLESHEET_URI:
|
|
#ifdef HAVE_WEBKIT2
|
|
g_value_set_string (value, web_settings->user_stylesheet_uri);
|
|
#else
|
|
g_value_take_string (value, katze_object_get_string (web_settings,
|
|
WEB_SETTINGS_STRING ("user-stylesheet-uri")));
|
|
#endif
|
|
break;
|
|
case PROP_PRINT_WITHOUT_DIALOG:
|
|
g_value_set_boolean (value, web_settings->print_without_dialog);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_new:
|
|
*
|
|
* Creates a new #MidoriWebSettings instance with default values.
|
|
*
|
|
* You will typically want to assign this to a #MidoriWebView
|
|
* or #MidoriBrowser.
|
|
*
|
|
* Return value: (transfer full): a new #MidoriWebSettings
|
|
**/
|
|
MidoriWebSettings*
|
|
midori_web_settings_new (void)
|
|
{
|
|
MidoriWebSettings* web_settings = g_object_new (MIDORI_TYPE_WEB_SETTINGS,
|
|
NULL);
|
|
|
|
return web_settings;
|
|
}
|
|
|
|
static void
|
|
midori_web_settings_process_stylesheets (MidoriWebSettings* settings,
|
|
gint delta_len)
|
|
{
|
|
GHashTableIter it;
|
|
GString* css;
|
|
gchar* encoded;
|
|
gpointer value;
|
|
static guint length = 0;
|
|
|
|
g_return_if_fail ((gint)length >= -delta_len);
|
|
|
|
length += delta_len;
|
|
|
|
/* Precalculate size to avoid re-allocations */
|
|
css = g_string_sized_new (length);
|
|
|
|
if (settings->user_stylesheet_uri_cached != NULL)
|
|
g_string_append (css, settings->user_stylesheet_uri_cached);
|
|
|
|
if (settings->user_stylesheets != NULL)
|
|
{
|
|
g_hash_table_iter_init (&it, settings->user_stylesheets);
|
|
while (g_hash_table_iter_next (&it, NULL, &value))
|
|
g_string_append (css, (gchar*)value);
|
|
}
|
|
|
|
/* data: uri prefix from Source/WebCore/page/Page.cpp:700 in WebKit */
|
|
encoded = g_strconcat ("data:text/css;charset=utf-8;base64,", css->str, NULL);
|
|
#ifdef HAVE_WEBKIT2
|
|
/* TODO: webkit_web_view_group_add_user_style_sheet */
|
|
#else
|
|
g_object_set (settings, WEB_SETTINGS_STRING ("user-stylesheet-uri"), encoded, NULL);
|
|
#endif
|
|
g_free (encoded);
|
|
g_string_free (css, TRUE);
|
|
}
|
|
|
|
static void
|
|
base64_space_pad (gchar* base64,
|
|
guint len)
|
|
{
|
|
/* Replace '=' padding at the end with encoded spaces
|
|
so WebKit will accept concatenations to this string */
|
|
if (len > 2 && base64[len - 2] == '=')
|
|
{
|
|
base64[len - 3] += 2;
|
|
base64[len - 2] = 'A';
|
|
}
|
|
if (len > 1 && base64[len - 1] == '=')
|
|
base64[len - 1] = 'g';
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_add_style:
|
|
* @settings: the MidoriWebSettings instance to modify
|
|
* @rule_id: a static string identifier
|
|
* @style: a CSS stylesheet
|
|
*
|
|
* Adds or replaces a custom stylesheet.
|
|
*
|
|
* Since: 0.4.2
|
|
**/
|
|
void
|
|
midori_web_settings_add_style (MidoriWebSettings* settings,
|
|
const gchar* rule_id,
|
|
const gchar* style)
|
|
{
|
|
gchar* base64;
|
|
guint len;
|
|
|
|
g_return_if_fail (MIDORI_IS_WEB_SETTINGS (settings));
|
|
g_return_if_fail (rule_id != NULL);
|
|
g_return_if_fail (style != NULL);
|
|
|
|
len = strlen (style);
|
|
base64 = g_base64_encode ((const guchar*)style, len);
|
|
len = ((len + 2) / 3) * 4;
|
|
base64_space_pad (base64, len);
|
|
|
|
if (settings->user_stylesheets == NULL)
|
|
settings->user_stylesheets = g_hash_table_new_full (g_str_hash, NULL,
|
|
NULL, g_free);
|
|
|
|
g_hash_table_insert (settings->user_stylesheets, (gchar*)rule_id, base64);
|
|
midori_web_settings_process_stylesheets (settings, len);
|
|
}
|
|
|
|
/**
|
|
* midori_web_settings_remove_style:
|
|
* @settings: the MidoriWebSettings instance to modify
|
|
* @rule_id: the string identifier used previously
|
|
*
|
|
* Removes a stylesheet from midori settings.
|
|
*
|
|
* Since: 0.4.2
|
|
**/
|
|
void
|
|
midori_web_settings_remove_style (MidoriWebSettings* settings,
|
|
const gchar* rule_id)
|
|
{
|
|
gchar* str;
|
|
|
|
g_return_if_fail (MIDORI_IS_WEB_SETTINGS (settings));
|
|
g_return_if_fail (rule_id != NULL);
|
|
|
|
if (settings->user_stylesheets != NULL)
|
|
{
|
|
if ((str = g_hash_table_lookup (settings->user_stylesheets, rule_id)))
|
|
{
|
|
guint len = strlen (str);
|
|
g_hash_table_remove (settings->user_stylesheets, rule_id);
|
|
midori_web_settings_process_stylesheets (settings, -len);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* midori_settings_new_full:
|
|
* @extensions: (out) (allow-none): a pointer into which
|
|
* to write an array of names of extensions which preferences
|
|
* indicate should be activated, or %NULL.
|
|
*
|
|
* Creates a new #MidoriWebSettings instance, loading
|
|
* configuration from disk according to preferences and
|
|
* invocation mode.
|
|
*
|
|
* You will typically want to assign this to a #MidoriWebView
|
|
* or #MidoriBrowser.
|
|
*
|
|
* Return value: (transfer full): a new #MidoriWebSettings
|
|
**/
|
|
MidoriWebSettings*
|
|
midori_settings_new_full (gchar*** extensions)
|
|
{
|
|
MidoriWebSettings* settings = midori_web_settings_new ();
|
|
gchar* config_file = midori_paths_get_config_filename_for_reading ("config");
|
|
GKeyFile* key_file = g_key_file_new ();
|
|
GError* error = NULL;
|
|
GObjectClass* class;
|
|
guint i, n_properties;
|
|
GParamSpec** pspecs;
|
|
GParamSpec* pspec;
|
|
GType type;
|
|
const gchar* property;
|
|
gchar* str;
|
|
gint integer;
|
|
gfloat number;
|
|
gboolean boolean;
|
|
|
|
if (!g_key_file_load_from_file (key_file, config_file,
|
|
G_KEY_FILE_KEEP_COMMENTS, &error))
|
|
{
|
|
if (error->code == G_FILE_ERROR_NOENT)
|
|
{
|
|
GError* inner_error = NULL;
|
|
katze_assign (config_file, midori_paths_get_preset_filename (NULL, "config"));
|
|
g_key_file_load_from_file (key_file, config_file,
|
|
G_KEY_FILE_KEEP_COMMENTS, &inner_error);
|
|
if (inner_error != NULL)
|
|
{
|
|
printf (_("The configuration couldn't be loaded: %s\n"),
|
|
inner_error->message);
|
|
g_error_free (inner_error);
|
|
}
|
|
}
|
|
else
|
|
printf (_("The configuration couldn't be loaded: %s\n"),
|
|
error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
class = G_OBJECT_GET_CLASS (settings);
|
|
pspecs = g_object_class_list_properties (class, &n_properties);
|
|
for (i = 0; i < n_properties; i++)
|
|
{
|
|
pspec = pspecs[i];
|
|
if (!(pspec->flags & G_PARAM_WRITABLE))
|
|
continue;
|
|
|
|
type = G_PARAM_SPEC_TYPE (pspec);
|
|
property = g_param_spec_get_name (pspec);
|
|
if (!g_key_file_has_key (key_file, "settings", property, NULL))
|
|
continue;
|
|
|
|
if (type == G_TYPE_PARAM_STRING)
|
|
{
|
|
str = g_key_file_get_string (key_file, "settings", property, NULL);
|
|
g_object_set (settings, property, str, NULL);
|
|
g_free (str);
|
|
}
|
|
else if (type == G_TYPE_PARAM_INT || type == G_TYPE_PARAM_UINT)
|
|
{
|
|
integer = g_key_file_get_integer (key_file, "settings", property, NULL);
|
|
g_object_set (settings, property, integer, NULL);
|
|
}
|
|
else if (type == G_TYPE_PARAM_FLOAT || type == G_TYPE_PARAM_DOUBLE)
|
|
{
|
|
number = g_key_file_get_double (key_file, "settings", property, NULL);
|
|
g_object_set (settings, property, number, NULL);
|
|
}
|
|
else if (type == G_TYPE_PARAM_BOOLEAN)
|
|
{
|
|
boolean = g_key_file_get_boolean (key_file, "settings", property, NULL);
|
|
g_object_set (settings, property, boolean, NULL);
|
|
}
|
|
else if (type == G_TYPE_PARAM_ENUM)
|
|
{
|
|
GEnumClass* enum_class = G_ENUM_CLASS (
|
|
g_type_class_peek (pspec->value_type));
|
|
GEnumValue* enum_value;
|
|
str = g_key_file_get_string (key_file, "settings", property, NULL);
|
|
enum_value = g_enum_get_value_by_name (enum_class, str);
|
|
if (enum_value)
|
|
g_object_set (settings, property, enum_value->value, NULL);
|
|
else
|
|
g_warning (_("Value '%s' is invalid for %s"),
|
|
str, property);
|
|
g_free (str);
|
|
}
|
|
else
|
|
g_warning (_("Invalid configuration value '%s'"), property);
|
|
}
|
|
g_free (pspecs);
|
|
|
|
if (extensions != NULL)
|
|
*extensions = g_key_file_get_keys (key_file, "extensions", NULL, NULL);
|
|
g_key_file_free (key_file);
|
|
|
|
/* Load accelerators */
|
|
katze_assign (config_file, midori_paths_get_config_filename_for_reading ("accels"));
|
|
if (g_access (config_file, F_OK) != 0)
|
|
katze_assign (config_file, midori_paths_get_preset_filename (NULL, "accels"));
|
|
gtk_accel_map_load (config_file);
|
|
g_free (config_file);
|
|
|
|
return settings;
|
|
}
|
|
|
|
/**
|
|
* midori_settings_save_to_file:
|
|
* @settings: a MidoriWebSettings instance to save
|
|
* @app: (type MidoriApp) (allow-none): a MidoriApp instance
|
|
* @filename: the filename into which to save settings
|
|
* @error: (out) (allow-none): return location for a GError, or %NULL
|
|
*
|
|
* Saves a #MidoriWebSettings instance to disk at the path given by @filename.
|
|
*
|
|
* Also saves the list of activated extensions from @app.
|
|
*
|
|
* Return value: %TRUE if no error occurred; %FALSE if an error
|
|
* occurred, in which case @error will contain detailed information
|
|
**/
|
|
gboolean
|
|
midori_settings_save_to_file (MidoriWebSettings* settings,
|
|
GObject* app,
|
|
const gchar* filename,
|
|
GError** error)
|
|
{
|
|
GKeyFile* key_file;
|
|
GObjectClass* class;
|
|
guint i, n_properties;
|
|
GParamSpec** pspecs;
|
|
GParamSpec* pspec;
|
|
GType type;
|
|
const gchar* property;
|
|
gboolean saved;
|
|
|
|
key_file = g_key_file_new ();
|
|
class = G_OBJECT_GET_CLASS (settings);
|
|
pspecs = g_object_class_list_properties (class, &n_properties);
|
|
for (i = 0; i < n_properties; i++)
|
|
{
|
|
pspec = pspecs[i];
|
|
type = G_PARAM_SPEC_TYPE (pspec);
|
|
property = g_param_spec_get_name (pspec);
|
|
if (!(pspec->flags & G_PARAM_WRITABLE))
|
|
continue;
|
|
if (type == G_TYPE_PARAM_STRING)
|
|
{
|
|
gchar* string;
|
|
const gchar* def_string = G_PARAM_SPEC_STRING (pspec)->default_value;
|
|
if (!strcmp (property, "user-stylesheet-uri"))
|
|
{
|
|
const gchar* user_stylesheet_uri = g_object_get_data (G_OBJECT (settings), property);
|
|
if (user_stylesheet_uri)
|
|
{
|
|
g_key_file_set_string (key_file, "settings", property,
|
|
user_stylesheet_uri);
|
|
}
|
|
else
|
|
g_key_file_remove_key (key_file, "settings", property, NULL);
|
|
continue;
|
|
}
|
|
|
|
g_object_get (settings, property, &string, NULL);
|
|
if (!def_string)
|
|
def_string = "";
|
|
if (strcmp (string ? string : "", def_string))
|
|
g_key_file_set_string (key_file, "settings", property, string ? string : "");
|
|
g_free (string);
|
|
}
|
|
else if (type == G_TYPE_PARAM_INT)
|
|
{
|
|
gint integer;
|
|
g_object_get (settings, property, &integer, NULL);
|
|
if (integer != G_PARAM_SPEC_INT (pspec)->default_value)
|
|
g_key_file_set_integer (key_file, "settings", property, integer);
|
|
}
|
|
else if (type == G_TYPE_PARAM_UINT)
|
|
{
|
|
guint integer;
|
|
g_object_get (settings, property, &integer, NULL);
|
|
if (integer != G_PARAM_SPEC_UINT (pspec)->default_value)
|
|
g_key_file_set_integer (key_file, "settings", property, integer);
|
|
}
|
|
else if (type == G_TYPE_PARAM_DOUBLE)
|
|
{
|
|
gdouble number;
|
|
g_object_get (settings, property, &number, NULL);
|
|
if (number != G_PARAM_SPEC_DOUBLE (pspec)->default_value)
|
|
g_key_file_set_double (key_file, "settings", property, number);
|
|
}
|
|
else if (type == G_TYPE_PARAM_FLOAT)
|
|
{
|
|
gfloat number;
|
|
g_object_get (settings, property, &number, NULL);
|
|
if (number != G_PARAM_SPEC_FLOAT (pspec)->default_value)
|
|
g_key_file_set_double (key_file, "settings", property, number);
|
|
}
|
|
else if (type == G_TYPE_PARAM_BOOLEAN)
|
|
{
|
|
gboolean truth;
|
|
g_object_get (settings, property, &truth, NULL);
|
|
if (truth != G_PARAM_SPEC_BOOLEAN (pspec)->default_value)
|
|
g_key_file_set_boolean (key_file, "settings", property, truth);
|
|
}
|
|
else if (type == G_TYPE_PARAM_ENUM)
|
|
{
|
|
GEnumClass* enum_class = G_ENUM_CLASS (
|
|
g_type_class_peek (pspec->value_type));
|
|
gint integer;
|
|
GEnumValue* enum_value;
|
|
g_object_get (settings, property, &integer, NULL);
|
|
enum_value = g_enum_get_value (enum_class, integer);
|
|
if (integer != G_PARAM_SPEC_ENUM (pspec)->default_value)
|
|
g_key_file_set_string (key_file, "settings", property,
|
|
enum_value->value_name);
|
|
}
|
|
else
|
|
g_warning (_("Invalid configuration value '%s'"), property);
|
|
}
|
|
g_free (pspecs);
|
|
|
|
if (app != NULL)
|
|
{
|
|
/* Take frozen list of active extensions until preferences reset it */
|
|
gchar** _extensions;
|
|
KatzeArray* extensions;
|
|
if ((_extensions = g_object_get_data (G_OBJECT (app), "extensions")))
|
|
{
|
|
i = 0;
|
|
while (_extensions[i])
|
|
g_key_file_set_boolean (key_file, "extensions", _extensions[i++], TRUE);
|
|
}
|
|
else if ((extensions = katze_object_get_object (app, "extensions")))
|
|
{
|
|
MidoriExtension* extension;
|
|
KATZE_ARRAY_FOREACH_ITEM (extension, extensions)
|
|
if (midori_extension_is_active (extension))
|
|
{
|
|
const gchar* extension_filename = g_object_get_data (G_OBJECT (extension), "filename");
|
|
g_return_val_if_fail (extension_filename != NULL, FALSE);
|
|
if (extension_filename && strchr (extension_filename, '/'))
|
|
g_warning ("%s: %s unexpected /", G_STRFUNC, extension_filename);
|
|
gchar* key = katze_object_get_string (extension, "key");
|
|
gchar* subname = key ? g_strdup_printf ("%s/%s", extension_filename, key) : g_strdup (extension_filename);
|
|
g_key_file_set_boolean (key_file, "extensions", subname, TRUE);
|
|
g_free (key);
|
|
g_free (subname);
|
|
}
|
|
g_object_unref (extensions);
|
|
}
|
|
}
|
|
|
|
saved = sokoke_key_file_save_to_file (key_file, filename, error);
|
|
g_key_file_free (key_file);
|
|
return saved;
|
|
}
|