| /* |
| * Copyright (C) 2011 Igalia S.L. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library 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 |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| #include "config.h" |
| #include "WebKitWebContext.h" |
| |
| #include "APIString.h" |
| #include "WebBatteryManagerProxy.h" |
| #include "WebCertificateInfo.h" |
| #include "WebCookieManagerProxy.h" |
| #include "WebGeolocationManagerProxy.h" |
| #include "WebKitBatteryProvider.h" |
| #include "WebKitCertificateInfoPrivate.h" |
| #include "WebKitCookieManagerPrivate.h" |
| #include "WebKitDownloadClient.h" |
| #include "WebKitDownloadPrivate.h" |
| #include "WebKitFaviconDatabasePrivate.h" |
| #include "WebKitGeolocationProvider.h" |
| #include "WebKitInjectedBundleClient.h" |
| #include "WebKitPluginPrivate.h" |
| #include "WebKitPrivate.h" |
| #include "WebKitRequestManagerClient.h" |
| #include "WebKitSecurityManagerPrivate.h" |
| #include "WebKitTextChecker.h" |
| #include "WebKitURISchemeRequestPrivate.h" |
| #include "WebKitWebContextPrivate.h" |
| #include "WebKitWebViewBasePrivate.h" |
| #include "WebKitWebViewGroupPrivate.h" |
| #include "WebResourceCacheManagerProxy.h" |
| #include <WebCore/FileSystem.h> |
| #include <WebCore/IconDatabase.h> |
| #include <WebCore/Language.h> |
| #include <wtf/HashMap.h> |
| #include <wtf/OwnPtr.h> |
| #include <wtf/PassRefPtr.h> |
| #include <wtf/RefCounted.h> |
| #include <wtf/gobject/GRefPtr.h> |
| #include <wtf/gobject/GUniquePtr.h> |
| #include <wtf/text/CString.h> |
| |
| using namespace WebKit; |
| |
| /** |
| * SECTION: WebKitWebContext |
| * @Short_description: Manages aspects common to all #WebKitWebView<!-- -->s |
| * @Title: WebKitWebContext |
| * |
| * The #WebKitWebContext manages all aspects common to all |
| * #WebKitWebView<!-- -->s. |
| * |
| * You can define the #WebKitCacheModel with |
| * webkit_web_context_set_cache_model(), depending on the needs of |
| * your application. You can access the #WebKitCookieManager or the |
| * #WebKitSecurityManager to specify the behaviour of your application |
| * regarding cookies and security, using |
| * webkit_web_context_get_cookie_manager() and |
| * webkit_web_context_get_security_manager() for that. |
| * |
| * It is also possible to change your preferred language or enable |
| * spell checking, using webkit_web_context_set_preferred_languages(), |
| * webkit_web_context_set_spell_checking_languages() and |
| * webkit_web_context_set_spell_checking_enabled(). |
| * |
| * You can use webkit_web_context_register_uri_scheme() to register |
| * custom URI schemes, and manage several other settings. |
| * |
| */ |
| |
| enum { |
| DOWNLOAD_STARTED, |
| INITIALIZE_WEB_EXTENSIONS, |
| |
| LAST_SIGNAL |
| }; |
| |
| class WebKitURISchemeHandler: public RefCounted<WebKitURISchemeHandler> { |
| public: |
| WebKitURISchemeHandler() |
| : m_callback(0) |
| , m_userData(0) |
| , m_destroyNotify(0) |
| { |
| } |
| WebKitURISchemeHandler(WebKitURISchemeRequestCallback callback, void* userData, GDestroyNotify destroyNotify) |
| : m_callback(callback) |
| , m_userData(userData) |
| , m_destroyNotify(destroyNotify) |
| { |
| } |
| |
| ~WebKitURISchemeHandler() |
| { |
| if (m_destroyNotify) |
| m_destroyNotify(m_userData); |
| } |
| |
| bool hasCallback() |
| { |
| return m_callback; |
| } |
| |
| void performCallback(WebKitURISchemeRequest* request) |
| { |
| ASSERT(m_callback); |
| |
| m_callback(request, m_userData); |
| } |
| |
| private: |
| WebKitURISchemeRequestCallback m_callback; |
| void* m_userData; |
| GDestroyNotify m_destroyNotify; |
| }; |
| |
| typedef HashMap<String, RefPtr<WebKitURISchemeHandler> > URISchemeHandlerMap; |
| typedef HashMap<uint64_t, GRefPtr<WebKitURISchemeRequest> > URISchemeRequestMap; |
| |
| struct _WebKitWebContextPrivate { |
| RefPtr<WebContext> context; |
| |
| GRefPtr<WebKitCookieManager> cookieManager; |
| GRefPtr<WebKitFaviconDatabase> faviconDatabase; |
| GRefPtr<WebKitSecurityManager> securityManager; |
| RefPtr<WebSoupCustomProtocolRequestManager> requestManager; |
| URISchemeHandlerMap uriSchemeHandlers; |
| URISchemeRequestMap uriSchemeRequests; |
| #if ENABLE(GEOLOCATION) |
| RefPtr<WebKitGeolocationProvider> geolocationProvider; |
| #endif |
| #if ENABLE(BATTERY_STATUS) |
| RefPtr<WebKitBatteryProvider> batteryProvider; |
| #endif |
| #if ENABLE(SPELLCHECK) |
| OwnPtr<WebKitTextChecker> textChecker; |
| #endif |
| CString faviconDatabaseDirectory; |
| WebKitTLSErrorsPolicy tlsErrorsPolicy; |
| |
| HashMap<uint64_t, WebKitWebView*> webViews; |
| GRefPtr<WebKitWebViewGroup> defaultWebViewGroup; |
| |
| CString webExtensionsDirectory; |
| GRefPtr<GVariant> webExtensionsInitializationUserData; |
| }; |
| |
| static guint signals[LAST_SIGNAL] = { 0, }; |
| |
| WEBKIT_DEFINE_TYPE(WebKitWebContext, webkit_web_context, G_TYPE_OBJECT) |
| |
| static void webkit_web_context_class_init(WebKitWebContextClass* webContextClass) |
| { |
| GObjectClass* gObjectClass = G_OBJECT_CLASS(webContextClass); |
| |
| /** |
| * WebKitWebContext::download-started: |
| * @context: the #WebKitWebContext |
| * @download: the #WebKitDownload associated with this event |
| * |
| * This signal is emitted when a new download request is made. |
| */ |
| signals[DOWNLOAD_STARTED] = |
| g_signal_new("download-started", |
| G_TYPE_FROM_CLASS(gObjectClass), |
| G_SIGNAL_RUN_LAST, |
| 0, 0, 0, |
| g_cclosure_marshal_VOID__OBJECT, |
| G_TYPE_NONE, 1, |
| WEBKIT_TYPE_DOWNLOAD); |
| |
| /** |
| * WebKitWebContext::initialize-web-extensions: |
| * @context: the #WebKitWebContext |
| * |
| * This signal is emitted when a new web process is about to be |
| * launched. It signals the most appropriate moment to use |
| * webkit_web_context_set_web_extensions_initialization_user_data() |
| * and webkit_web_context_set_web_extensions_directory(). |
| * |
| * Since: 2.4 |
| */ |
| signals[INITIALIZE_WEB_EXTENSIONS] = |
| g_signal_new("initialize-web-extensions", |
| G_TYPE_FROM_CLASS(gObjectClass), |
| G_SIGNAL_RUN_LAST, |
| 0, nullptr, nullptr, |
| g_cclosure_marshal_VOID__VOID, |
| G_TYPE_NONE, 0); |
| } |
| |
| static CString injectedBundleDirectory() |
| { |
| const char* bundleDirectory = g_getenv("WEBKIT_INJECTED_BUNDLE_PATH"); |
| if (bundleDirectory && g_file_test(bundleDirectory, G_FILE_TEST_IS_DIR)) |
| return bundleDirectory; |
| |
| static const char* injectedBundlePath = LIBDIR G_DIR_SEPARATOR_S "webkit2gtk-" WEBKITGTK_API_VERSION_STRING |
| G_DIR_SEPARATOR_S "injected-bundle" G_DIR_SEPARATOR_S; |
| return injectedBundlePath; |
| } |
| |
| static CString injectedBundleFilename() |
| { |
| GUniquePtr<char> bundleFilename(g_build_filename(injectedBundleDirectory().data(), "libwebkit2gtkinjectedbundle.so", NULL)); |
| return bundleFilename.get(); |
| } |
| |
| static gpointer createDefaultWebContext(gpointer) |
| { |
| static GRefPtr<WebKitWebContext> webContext = adoptGRef(WEBKIT_WEB_CONTEXT(g_object_new(WEBKIT_TYPE_WEB_CONTEXT, NULL))); |
| WebKitWebContextPrivate* priv = webContext->priv; |
| |
| priv->context = WebContext::create(WebCore::filenameToString(injectedBundleFilename().data())); |
| priv->requestManager = webContext->priv->context->supplement<WebSoupCustomProtocolRequestManager>(); |
| priv->context->setCacheModel(CacheModelPrimaryWebBrowser); |
| #if ENABLE(NETWORK_PROCESS) |
| // FIXME: Temporary use an env var until we have API to set the process model. See https://bugs.webkit.org/show_bug.cgi?id=125463. |
| priv->context->setUsesNetworkProcess(g_getenv("WEBKIT_USE_NETWORK_PROCESS")); |
| #endif |
| priv->tlsErrorsPolicy = WEBKIT_TLS_ERRORS_POLICY_IGNORE; |
| |
| attachInjectedBundleClientToContext(webContext.get()); |
| attachDownloadClientToContext(webContext.get()); |
| attachRequestManagerClientToContext(webContext.get()); |
| |
| #if ENABLE(GEOLOCATION) |
| priv->geolocationProvider = WebKitGeolocationProvider::create(priv->context->supplement<WebGeolocationManagerProxy>()); |
| #endif |
| #if ENABLE(BATTERY_STATUS) |
| priv->batteryProvider = WebKitBatteryProvider::create(priv->context->supplement<WebBatteryManagerProxy>()); |
| #endif |
| #if ENABLE(SPELLCHECK) |
| priv->textChecker = WebKitTextChecker::create(); |
| #endif |
| return webContext.get(); |
| } |
| |
| /** |
| * webkit_web_context_get_default: |
| * |
| * Gets the default web context |
| * |
| * Returns: (transfer none): a #WebKitWebContext |
| */ |
| WebKitWebContext* webkit_web_context_get_default(void) |
| { |
| static GOnce onceInit = G_ONCE_INIT; |
| return WEBKIT_WEB_CONTEXT(g_once(&onceInit, createDefaultWebContext, 0)); |
| } |
| |
| /** |
| * webkit_web_context_set_cache_model: |
| * @context: the #WebKitWebContext |
| * @cache_model: a #WebKitCacheModel |
| * |
| * Specifies a usage model for WebViews, which WebKit will use to |
| * determine its caching behavior. All web views follow the cache |
| * model. This cache model determines the RAM and disk space to use |
| * for caching previously viewed content . |
| * |
| * Research indicates that users tend to browse within clusters of |
| * documents that hold resources in common, and to revisit previously |
| * visited documents. WebKit and the frameworks below it include |
| * built-in caches that take advantage of these patterns, |
| * substantially improving document load speed in browsing |
| * situations. The WebKit cache model controls the behaviors of all of |
| * these caches, including various WebCore caches. |
| * |
| * Browsers can improve document load speed substantially by |
| * specifying %WEBKIT_CACHE_MODEL_WEB_BROWSER. Applications without a |
| * browsing interface can reduce memory usage substantially by |
| * specifying %WEBKIT_CACHE_MODEL_DOCUMENT_VIEWER. The default value is |
| * %WEBKIT_CACHE_MODEL_WEB_BROWSER. |
| */ |
| void webkit_web_context_set_cache_model(WebKitWebContext* context, WebKitCacheModel model) |
| { |
| CacheModel cacheModel; |
| |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| switch (model) { |
| case WEBKIT_CACHE_MODEL_DOCUMENT_VIEWER: |
| cacheModel = CacheModelDocumentViewer; |
| break; |
| case WEBKIT_CACHE_MODEL_WEB_BROWSER: |
| cacheModel = CacheModelPrimaryWebBrowser; |
| break; |
| case WEBKIT_CACHE_MODEL_DOCUMENT_BROWSER: |
| cacheModel = CacheModelDocumentBrowser; |
| break; |
| default: |
| g_assert_not_reached(); |
| } |
| |
| if (cacheModel != context->priv->context->cacheModel()) |
| context->priv->context->setCacheModel(cacheModel); |
| } |
| |
| /** |
| * webkit_web_context_get_cache_model: |
| * @context: the #WebKitWebContext |
| * |
| * Returns the current cache model. For more information about this |
| * value check the documentation of the function |
| * webkit_web_context_set_cache_model(). |
| * |
| * Returns: the current #WebKitCacheModel |
| */ |
| WebKitCacheModel webkit_web_context_get_cache_model(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), WEBKIT_CACHE_MODEL_WEB_BROWSER); |
| |
| switch (context->priv->context->cacheModel()) { |
| case CacheModelDocumentViewer: |
| return WEBKIT_CACHE_MODEL_DOCUMENT_VIEWER; |
| case CacheModelPrimaryWebBrowser: |
| return WEBKIT_CACHE_MODEL_WEB_BROWSER; |
| case CacheModelDocumentBrowser: |
| return WEBKIT_CACHE_MODEL_DOCUMENT_BROWSER; |
| default: |
| g_assert_not_reached(); |
| } |
| |
| return WEBKIT_CACHE_MODEL_WEB_BROWSER; |
| } |
| |
| /** |
| * webkit_web_context_clear_cache: |
| * @context: a #WebKitWebContext |
| * |
| * Clears all resources currently cached. |
| * See also webkit_web_context_set_cache_model(). |
| */ |
| void webkit_web_context_clear_cache(WebKitWebContext* context) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| context->priv->context->supplement<WebResourceCacheManagerProxy>()->clearCacheForAllOrigins(AllResourceCaches); |
| } |
| |
| typedef HashMap<DownloadProxy*, GRefPtr<WebKitDownload> > DownloadsMap; |
| |
| static DownloadsMap& downloadsMap() |
| { |
| DEFINE_STATIC_LOCAL(DownloadsMap, downloads, ()); |
| return downloads; |
| } |
| |
| /** |
| * webkit_web_context_download_uri: |
| * @context: a #WebKitWebContext |
| * @uri: the URI to download |
| * |
| * Requests downloading of the specified URI string. The download operation |
| * will not be associated to any #WebKitWebView, if you are interested in |
| * starting a download from a particular #WebKitWebView use |
| * webkit_web_view_download_uri() instead. |
| * |
| * Returns: (transfer full): a new #WebKitDownload representing the |
| * the download operation. |
| */ |
| WebKitDownload* webkit_web_context_download_uri(WebKitWebContext* context, const gchar* uri) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| g_return_val_if_fail(uri, 0); |
| |
| return webkitWebContextStartDownload(context, uri, 0); |
| } |
| |
| /** |
| * webkit_web_context_get_cookie_manager: |
| * @context: a #WebKitWebContext |
| * |
| * Get the #WebKitCookieManager of @context. |
| * |
| * Returns: (transfer none): the #WebKitCookieManager of @context. |
| */ |
| WebKitCookieManager* webkit_web_context_get_cookie_manager(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| |
| WebKitWebContextPrivate* priv = context->priv; |
| if (!priv->cookieManager) |
| priv->cookieManager = adoptGRef(webkitCookieManagerCreate(priv->context->supplement<WebCookieManagerProxy>())); |
| |
| return priv->cookieManager.get(); |
| } |
| |
| static void ensureFaviconDatabase(WebKitWebContext* context) |
| { |
| WebKitWebContextPrivate* priv = context->priv; |
| if (priv->faviconDatabase) |
| return; |
| |
| priv->faviconDatabase = adoptGRef(webkitFaviconDatabaseCreate(priv->context->iconDatabase())); |
| } |
| |
| /** |
| * webkit_web_context_set_favicon_database_directory: |
| * @context: a #WebKitWebContext |
| * @path: (allow-none): an absolute path to the icon database |
| * directory or %NULL to use the defaults |
| * |
| * Set the directory path to be used to store the favicons database |
| * for @context on disk. Passing %NULL as @path means using the |
| * default directory for the platform (see g_get_user_data_dir()). |
| * |
| * Calling this method also means enabling the favicons database for |
| * its use from the applications, so that's why it's expected to be |
| * called only once. Further calls for the same instance of |
| * #WebKitWebContext won't cause any effect. |
| */ |
| void webkit_web_context_set_favicon_database_directory(WebKitWebContext* context, const gchar* path) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| WebKitWebContextPrivate* priv = context->priv; |
| WebIconDatabase* iconDatabase = priv->context->iconDatabase(); |
| if (iconDatabase->isOpen()) |
| return; |
| |
| ensureFaviconDatabase(context); |
| |
| // Use default if 0 is passed as parameter. |
| String directoryPath = WebCore::filenameToString(path); |
| priv->faviconDatabaseDirectory = directoryPath.isEmpty() |
| ? priv->context->iconDatabasePath().utf8() |
| : directoryPath.utf8(); |
| |
| // Build the full path to the icon database file on disk. |
| GUniquePtr<gchar> faviconDatabasePath(g_build_filename(priv->faviconDatabaseDirectory.data(), |
| WebCore::IconDatabase::defaultDatabaseFilename().utf8().data(), nullptr)); |
| |
| // Setting the path will cause the icon database to be opened. |
| priv->context->setIconDatabasePath(WebCore::filenameToString(faviconDatabasePath.get())); |
| } |
| |
| /** |
| * webkit_web_context_get_favicon_database_directory: |
| * @context: a #WebKitWebContext |
| * |
| * Get the directory path being used to store the favicons database |
| * for @context, or %NULL if |
| * webkit_web_context_set_favicon_database_directory() hasn't been |
| * called yet. |
| * |
| * This function will always return the same path after having called |
| * webkit_web_context_set_favicon_database_directory() for the first |
| * time. |
| * |
| * Returns: (transfer none): the path of the directory of the favicons |
| * database associated with @context, or %NULL. |
| */ |
| const gchar* webkit_web_context_get_favicon_database_directory(WebKitWebContext *context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| |
| WebKitWebContextPrivate* priv = context->priv; |
| if (priv->faviconDatabaseDirectory.isNull()) |
| return 0; |
| |
| return priv->faviconDatabaseDirectory.data(); |
| } |
| |
| /** |
| * webkit_web_context_get_favicon_database: |
| * @context: a #WebKitWebContext |
| * |
| * Get the #WebKitFaviconDatabase associated with @context. |
| * |
| * To initialize the database you need to call |
| * webkit_web_context_set_favicon_database_directory(). |
| * |
| * Returns: (transfer none): the #WebKitFaviconDatabase of @context. |
| */ |
| WebKitFaviconDatabase* webkit_web_context_get_favicon_database(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| |
| ensureFaviconDatabase(context); |
| return context->priv->faviconDatabase.get(); |
| } |
| |
| /** |
| * webkit_web_context_get_security_manager: |
| * @context: a #WebKitWebContext |
| * |
| * Get the #WebKitSecurityManager of @context. |
| * |
| * Returns: (transfer none): the #WebKitSecurityManager of @context. |
| */ |
| WebKitSecurityManager* webkit_web_context_get_security_manager(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| |
| WebKitWebContextPrivate* priv = context->priv; |
| if (!priv->securityManager) |
| priv->securityManager = adoptGRef(webkitSecurityManagerCreate(context)); |
| |
| return priv->securityManager.get(); |
| } |
| |
| /** |
| * webkit_web_context_set_additional_plugins_directory: |
| * @context: a #WebKitWebContext |
| * @directory: the directory to add |
| * |
| * Set an additional directory where WebKit will look for plugins. |
| */ |
| void webkit_web_context_set_additional_plugins_directory(WebKitWebContext* context, const char* directory) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(directory); |
| |
| context->priv->context->setAdditionalPluginsDirectory(WebCore::filenameToString(directory)); |
| } |
| |
| static void destroyPluginList(GList* plugins) |
| { |
| g_list_free_full(plugins, g_object_unref); |
| } |
| |
| static void webkitWebContextGetPluginThread(GTask* task, gpointer object, gpointer taskData, GCancellable*) |
| { |
| Vector<PluginModuleInfo> plugins = WEBKIT_WEB_CONTEXT(object)->priv->context->pluginInfoStore().plugins(); |
| GList* returnValue = 0; |
| for (size_t i = 0; i < plugins.size(); ++i) |
| returnValue = g_list_prepend(returnValue, webkitPluginCreate(plugins[i])); |
| g_task_return_pointer(task, returnValue, reinterpret_cast<GDestroyNotify>(destroyPluginList)); |
| } |
| |
| /** |
| * webkit_web_context_get_plugins: |
| * @context: a #WebKitWebContext |
| * @cancellable: (allow-none): a #GCancellable or %NULL to ignore |
| * @callback: (scope async): a #GAsyncReadyCallback to call when the request is satisfied |
| * @user_data: (closure): the data to pass to callback function |
| * |
| * Asynchronously get the list of installed plugins. |
| * |
| * When the operation is finished, @callback will be called. You can then call |
| * webkit_web_context_get_plugins_finish() to get the result of the operation. |
| */ |
| void webkit_web_context_get_plugins(WebKitWebContext* context, GCancellable* cancellable, GAsyncReadyCallback callback, gpointer userData) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| GRefPtr<GTask> task = adoptGRef(g_task_new(context, cancellable, callback, userData)); |
| g_task_run_in_thread(task.get(), webkitWebContextGetPluginThread); |
| } |
| |
| /** |
| * webkit_web_context_get_plugins_finish: |
| * @context: a #WebKitWebContext |
| * @result: a #GAsyncResult |
| * @error: return location for error or %NULL to ignore |
| * |
| * Finish an asynchronous operation started with webkit_web_context_get_plugins. |
| * |
| * Returns: (element-type WebKitPlugin) (transfer full): a #GList of #WebKitPlugin. You must free the #GList with |
| * g_list_free() and unref the #WebKitPlugin<!-- -->s with g_object_unref() when you're done with them. |
| */ |
| GList* webkit_web_context_get_plugins_finish(WebKitWebContext* context, GAsyncResult* result, GError** error) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| g_return_val_if_fail(g_task_is_valid(result, context), 0); |
| |
| return static_cast<GList*>(g_task_propagate_pointer(G_TASK(result), error)); |
| } |
| |
| /** |
| * webkit_web_context_register_uri_scheme: |
| * @context: a #WebKitWebContext |
| * @scheme: the network scheme to register |
| * @callback: (scope async): a #WebKitURISchemeRequestCallback |
| * @user_data: data to pass to callback function |
| * @user_data_destroy_func: destroy notify for @user_data |
| * |
| * Register @scheme in @context, so that when an URI request with @scheme is made in the |
| * #WebKitWebContext, the #WebKitURISchemeRequestCallback registered will be called with a |
| * #WebKitURISchemeRequest. |
| * It is possible to handle URI scheme requests asynchronously, by calling g_object_ref() on the |
| * #WebKitURISchemeRequest and calling webkit_uri_scheme_request_finish() later |
| * when the data of the request is available or |
| * webkit_uri_scheme_request_finish_error() in case of error. |
| * |
| * <informalexample><programlisting> |
| * static void |
| * about_uri_scheme_request_cb (WebKitURISchemeRequest *request, |
| * gpointer user_data) |
| * { |
| * GInputStream *stream; |
| * gsize stream_length; |
| * const gchar *path; |
| * |
| * path = webkit_uri_scheme_request_get_path (request); |
| * if (!g_strcmp0 (path, "plugins")) { |
| * /<!-- -->* Create a GInputStream with the contents of plugins about page, and set its length to stream_length *<!-- -->/ |
| * } else if (!g_strcmp0 (path, "memory")) { |
| * /<!-- -->* Create a GInputStream with the contents of memory about page, and set its length to stream_length *<!-- -->/ |
| * } else if (!g_strcmp0 (path, "applications")) { |
| * /<!-- -->* Create a GInputStream with the contents of applications about page, and set its length to stream_length *<!-- -->/ |
| * } else if (!g_strcmp0 (path, "example")) { |
| * gchar *contents; |
| * |
| * contents = g_strdup_printf ("<html><body><p>Example about page</p></body></html>"); |
| * stream_length = strlen (contents); |
| * stream = g_memory_input_stream_new_from_data (contents, stream_length, g_free); |
| * } else { |
| * GError *error; |
| * |
| * error = g_error_new (ABOUT_HANDLER_ERROR, ABOUT_HANDLER_ERROR_INVALID, "Invalid about:%s page.", path); |
| * webkit_uri_scheme_request_finish_error (request, error); |
| * g_error_free (error); |
| * return; |
| * } |
| * webkit_uri_scheme_request_finish (request, stream, stream_length, "text/html"); |
| * g_object_unref (stream); |
| * } |
| * </programlisting></informalexample> |
| */ |
| void webkit_web_context_register_uri_scheme(WebKitWebContext* context, const char* scheme, WebKitURISchemeRequestCallback callback, gpointer userData, GDestroyNotify destroyNotify) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(scheme); |
| g_return_if_fail(callback); |
| |
| RefPtr<WebKitURISchemeHandler> handler = adoptRef(new WebKitURISchemeHandler(callback, userData, destroyNotify)); |
| context->priv->uriSchemeHandlers.set(String::fromUTF8(scheme), handler.get()); |
| context->priv->requestManager->registerSchemeForCustomProtocol(String::fromUTF8(scheme)); |
| } |
| |
| /** |
| * webkit_web_context_get_spell_checking_enabled: |
| * @context: a #WebKitWebContext |
| * |
| * Get whether spell checking feature is currently enabled. |
| * |
| * Returns: %TRUE If spell checking is enabled, or %FALSE otherwise. |
| */ |
| gboolean webkit_web_context_get_spell_checking_enabled(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), FALSE); |
| |
| #if ENABLE(SPELLCHECK) |
| return context->priv->textChecker->isSpellCheckingEnabled(); |
| #else |
| return false; |
| #endif |
| } |
| |
| /** |
| * webkit_web_context_set_spell_checking_enabled: |
| * @context: a #WebKitWebContext |
| * @enabled: Value to be set |
| * |
| * Enable or disable the spell checking feature. |
| */ |
| void webkit_web_context_set_spell_checking_enabled(WebKitWebContext* context, gboolean enabled) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| #if ENABLE(SPELLCHECK) |
| context->priv->textChecker->setSpellCheckingEnabled(enabled); |
| #endif |
| } |
| |
| /** |
| * webkit_web_context_get_spell_checking_languages: |
| * @context: a #WebKitWebContext |
| * |
| * Get the the list of spell checking languages associated with |
| * @context, or %NULL if no languages have been previously set. |
| * |
| * See webkit_web_context_set_spell_checking_languages() for more |
| * details on the format of the languages in the list. |
| * |
| * Returns: (array zero-terminated=1) (element-type utf8) (transfer none): A %NULL-terminated |
| * array of languages if available, or %NULL otherwise. |
| */ |
| const gchar* const* webkit_web_context_get_spell_checking_languages(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), 0); |
| |
| #if ENABLE(SPELLCHECK) |
| return context->priv->textChecker->getSpellCheckingLanguages(); |
| #else |
| return 0; |
| #endif |
| } |
| |
| /** |
| * webkit_web_context_set_spell_checking_languages: |
| * @context: a #WebKitWebContext |
| * @languages: (array zero-terminated=1) (transfer none): a %NULL-terminated list of spell checking languages |
| * |
| * Set the list of spell checking languages to be used for spell |
| * checking. |
| * |
| * The locale string typically is in the form lang_COUNTRY, where lang |
| * is an ISO-639 language code, and COUNTRY is an ISO-3166 country code. |
| * For instance, sv_FI for Swedish as written in Finland or pt_BR |
| * for Portuguese as written in Brazil. |
| * |
| * You need to call this function with a valid list of languages at |
| * least once in order to properly enable the spell checking feature |
| * in WebKit. |
| */ |
| void webkit_web_context_set_spell_checking_languages(WebKitWebContext* context, const gchar* const* languages) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(languages); |
| |
| #if ENABLE(SPELLCHECK) |
| context->priv->textChecker->setSpellCheckingLanguages(languages); |
| #endif |
| } |
| |
| /** |
| * webkit_web_context_set_preferred_languages: |
| * @context: a #WebKitWebContext |
| * @languages: (allow-none) (array zero-terminated=1) (element-type utf8) (transfer none): a %NULL-terminated list of language identifiers |
| * |
| * Set the list of preferred languages, sorted from most desirable |
| * to least desirable. The list will be used to build the "Accept-Language" |
| * header that will be included in the network requests started by |
| * the #WebKitWebContext. |
| */ |
| void webkit_web_context_set_preferred_languages(WebKitWebContext* context, const gchar* const* languageList) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| if (!languageList || !g_strv_length(const_cast<char**>(languageList))) |
| return; |
| |
| Vector<String> languages; |
| for (size_t i = 0; languageList[i]; ++i) |
| languages.append(String::fromUTF8(languageList[i]).lower().replace("_", "-")); |
| |
| WebCore::overrideUserPreferredLanguages(languages); |
| WebCore::languageDidChange(); |
| } |
| |
| /** |
| * webkit_web_context_set_tls_errors_policy: |
| * @context: a #WebKitWebContext |
| * @policy: a #WebKitTLSErrorsPolicy |
| * |
| * Set the TLS errors policy of @context as @policy |
| */ |
| void webkit_web_context_set_tls_errors_policy(WebKitWebContext* context, WebKitTLSErrorsPolicy policy) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| if (context->priv->tlsErrorsPolicy == policy) |
| return; |
| |
| context->priv->tlsErrorsPolicy = policy; |
| bool ignoreTLSErrors = policy == WEBKIT_TLS_ERRORS_POLICY_IGNORE; |
| if (context->priv->context->ignoreTLSErrors() != ignoreTLSErrors) |
| context->priv->context->setIgnoreTLSErrors(ignoreTLSErrors); |
| } |
| |
| /** |
| * webkit_web_context_get_tls_errors_policy: |
| * @context: a #WebKitWebContext |
| * |
| * Get the TLS errors policy of @context |
| * |
| * Returns: a #WebKitTLSErrorsPolicy |
| */ |
| WebKitTLSErrorsPolicy webkit_web_context_get_tls_errors_policy(WebKitWebContext* context) |
| { |
| g_return_val_if_fail(WEBKIT_IS_WEB_CONTEXT(context), WEBKIT_TLS_ERRORS_POLICY_IGNORE); |
| |
| return context->priv->tlsErrorsPolicy; |
| } |
| |
| /** |
| * webkit_web_context_set_web_extensions_directory: |
| * @context: a #WebKitWebContext |
| * @directory: the directory to add |
| * |
| * Set the directory where WebKit will look for Web Extensions. |
| * This method must be called before loading anything in this context, |
| * otherwise it will not have any effect. You can connect to |
| * #WebKitWebContext::initialize-web-extensions to call this method |
| * before anything is loaded. |
| */ |
| void webkit_web_context_set_web_extensions_directory(WebKitWebContext* context, const char* directory) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(directory); |
| |
| context->priv->webExtensionsDirectory = directory; |
| } |
| |
| /** |
| * webkit_web_context_set_web_extensions_initialization_user_data: |
| * @context: a #WebKitWebContext |
| * @user_data: a #GVariant |
| * |
| * Set user data to be passed to Web Extensions on initialization. |
| * The data will be passed to the |
| * #WebKitWebExtensionInitializeWithUserDataFunction. |
| * This method must be called before loading anything in this context, |
| * otherwise it will not have any effect. You can connect to |
| * #WebKitWebContext::initialize-web-extensions to call this method |
| * before anything is loaded. |
| * |
| * Since: 2.4 |
| */ |
| void webkit_web_context_set_web_extensions_initialization_user_data(WebKitWebContext* context, GVariant* userData) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(userData); |
| |
| context->priv->webExtensionsInitializationUserData = userData; |
| } |
| |
| /** |
| * webkit_web_context_set_disk_cache_directory: |
| * @context: a #WebKitWebContext |
| * @directory: the directory to set |
| * |
| * Set the directory where disk cache files will be stored |
| * This method must be called before loading anything in this context, otherwise |
| * it will not have any effect. |
| */ |
| void webkit_web_context_set_disk_cache_directory(WebKitWebContext* context, const char* directory) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(directory); |
| |
| context->priv->context->setDiskCacheDirectory(WebCore::filenameToString(directory)); |
| } |
| |
| /** |
| * webkit_web_context_prefetch_dns: |
| * @context: a #WebKitWebContext |
| * @hostname: a hostname to be resolved |
| * |
| * Resolve the domain name of the given @hostname in advance, so that if a URI |
| * of @hostname is requested the load will be performed more quickly. |
| */ |
| void webkit_web_context_prefetch_dns(WebKitWebContext* context, const char* hostname) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(hostname); |
| |
| ImmutableDictionary::MapType message; |
| message.set(String::fromUTF8("Hostname"), API::String::create(String::fromUTF8(hostname))); |
| context->priv->context->postMessageToInjectedBundle(String::fromUTF8("PrefetchDNS"), ImmutableDictionary::create(std::move(message)).get()); |
| } |
| |
| /** |
| * webkit_web_context_allow_tls_certificate_for_host: |
| * @context: a #WebKitWebContext |
| * @info: a #WebKitCertificateInfo |
| * @host: the host for which a certificate is to be allowed |
| * |
| * Ignore further TLS errors on the @host for the certificate present in @info. |
| * |
| * Since: 2.4 |
| */ |
| void webkit_web_context_allow_tls_certificate_for_host(WebKitWebContext* context, WebKitCertificateInfo* info, const gchar* host) |
| { |
| g_return_if_fail(WEBKIT_IS_WEB_CONTEXT(context)); |
| g_return_if_fail(info); |
| g_return_if_fail(host); |
| |
| RefPtr<WebCertificateInfo> webCertificateInfo = WebCertificateInfo::create(webkitCertificateInfoGetCertificateInfo(info)); |
| context->priv->context->allowSpecificHTTPSCertificateForHost(webCertificateInfo.get(), String::fromUTF8(host)); |
| } |
| |
| WebKitDownload* webkitWebContextGetOrCreateDownload(DownloadProxy* downloadProxy) |
| { |
| GRefPtr<WebKitDownload> download = downloadsMap().get(downloadProxy); |
| if (download) |
| return download.get(); |
| |
| download = adoptGRef(webkitDownloadCreate(downloadProxy)); |
| downloadsMap().set(downloadProxy, download.get()); |
| return download.get(); |
| } |
| |
| WebKitDownload* webkitWebContextStartDownload(WebKitWebContext* context, const char* uri, WebPageProxy* initiatingPage) |
| { |
| WebCore::ResourceRequest request(String::fromUTF8(uri)); |
| DownloadProxy* downloadProxy = context->priv->context->download(initiatingPage, request); |
| WebKitDownload* download = webkitDownloadCreateForRequest(downloadProxy, request); |
| downloadsMap().set(downloadProxy, download); |
| return download; |
| } |
| |
| void webkitWebContextRemoveDownload(DownloadProxy* downloadProxy) |
| { |
| downloadsMap().remove(downloadProxy); |
| } |
| |
| void webkitWebContextDownloadStarted(WebKitWebContext* context, WebKitDownload* download) |
| { |
| g_signal_emit(context, signals[DOWNLOAD_STARTED], 0, download); |
| } |
| |
| GVariant* webkitWebContextInitializeWebExtensions(WebKitWebContext* context) |
| { |
| g_signal_emit(context, signals[INITIALIZE_WEB_EXTENSIONS], 0); |
| return g_variant_new("(msmv)", |
| context->priv->webExtensionsDirectory.data(), |
| context->priv->webExtensionsInitializationUserData.get()); |
| } |
| |
| WebContext* webkitWebContextGetContext(WebKitWebContext* context) |
| { |
| g_assert(WEBKIT_IS_WEB_CONTEXT(context)); |
| |
| return context->priv->context.get(); |
| } |
| |
| WebSoupCustomProtocolRequestManager* webkitWebContextGetRequestManager(WebKitWebContext* context) |
| { |
| return context->priv->requestManager.get(); |
| } |
| |
| void webkitWebContextStartLoadingCustomProtocol(WebKitWebContext* context, uint64_t customProtocolID, API::URLRequest* urlRequest) |
| { |
| // FIXME: We need to figure out how to get the initiating page. |
| GRefPtr<WebKitURISchemeRequest> request = adoptGRef(webkitURISchemeRequestCreate(customProtocolID, context, urlRequest, nullptr)); |
| String scheme(String::fromUTF8(webkit_uri_scheme_request_get_scheme(request.get()))); |
| RefPtr<WebKitURISchemeHandler> handler = context->priv->uriSchemeHandlers.get(scheme); |
| ASSERT(handler.get()); |
| if (!handler->hasCallback()) |
| return; |
| |
| context->priv->uriSchemeRequests.set(customProtocolID, request.get()); |
| handler->performCallback(request.get()); |
| } |
| |
| void webkitWebContextStopLoadingCustomProtocol(WebKitWebContext* context, uint64_t customProtocolID) |
| { |
| GRefPtr<WebKitURISchemeRequest> request = context->priv->uriSchemeRequests.get(customProtocolID); |
| if (!request.get()) |
| return; |
| webkitURISchemeRequestCancel(request.get()); |
| } |
| |
| void webkitWebContextDidFinishLoadingCustomProtocol(WebKitWebContext* context, uint64_t customProtocolID) |
| { |
| context->priv->uriSchemeRequests.remove(customProtocolID); |
| } |
| |
| void webkitWebContextCreatePageForWebView(WebKitWebContext* context, WebKitWebView* webView, WebKitWebViewGroup* webViewGroup) |
| { |
| WebKitWebViewBase* webViewBase = WEBKIT_WEB_VIEW_BASE(webView); |
| WebPageGroup* pageGroup = webViewGroup ? webkitWebViewGroupGetPageGroup(webViewGroup) : 0; |
| webkitWebViewBaseCreateWebPage(webViewBase, context->priv->context.get(), pageGroup); |
| |
| WebPageProxy* page = webkitWebViewBaseGetPage(webViewBase); |
| context->priv->webViews.set(page->pageID(), webView); |
| |
| if (!pageGroup && !context->priv->defaultWebViewGroup) |
| context->priv->defaultWebViewGroup = adoptGRef(webkitWebViewGroupCreate(&page->pageGroup())); |
| } |
| |
| void webkitWebContextWebViewDestroyed(WebKitWebContext* context, WebKitWebView* webView) |
| { |
| WebPageProxy* page = webkitWebViewBaseGetPage(WEBKIT_WEB_VIEW_BASE(webView)); |
| context->priv->webViews.remove(page->pageID()); |
| } |
| |
| WebKitWebView* webkitWebContextGetWebViewForPage(WebKitWebContext* context, WebPageProxy* page) |
| { |
| return page ? context->priv->webViews.get(page->pageID()) : 0; |
| } |
| |
| WebKitWebViewGroup* webkitWebContextGetDefaultWebViewGroup(WebKitWebContext* context) |
| { |
| return context->priv->defaultWebViewGroup.get(); |
| } |