2014-10-31 18:17:05 +00:00
|
|
|
// Copyright (c) 2014 GitHub, Inc.
|
2014-04-25 09:49:37 +00:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2014-04-24 08:45:25 +00:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "atom/browser/api/atom_api_web_contents.h"
|
|
|
|
|
2018-09-13 00:25:56 +00:00
|
|
|
#include <memory>
|
2015-04-25 04:43:52 +00:00
|
|
|
#include <set>
|
2015-12-11 15:54:32 +00:00
|
|
|
#include <string>
|
2018-09-13 00:25:56 +00:00
|
|
|
#include <utility>
|
2015-04-25 04:43:52 +00:00
|
|
|
|
2018-02-22 03:49:17 +00:00
|
|
|
#include "atom/browser/api/atom_api_browser_window.h"
|
2016-01-21 18:22:23 +00:00
|
|
|
#include "atom/browser/api/atom_api_debugger.h"
|
2015-06-16 09:23:29 +00:00
|
|
|
#include "atom/browser/api/atom_api_session.h"
|
2015-05-11 06:40:40 +00:00
|
|
|
#include "atom/browser/atom_browser_client.h"
|
2014-10-23 06:04:13 +00:00
|
|
|
#include "atom/browser/atom_browser_context.h"
|
2015-06-18 09:01:23 +00:00
|
|
|
#include "atom/browser/atom_browser_main_parts.h"
|
2017-04-28 00:28:48 +00:00
|
|
|
#include "atom/browser/atom_javascript_dialog_manager.h"
|
2018-09-15 15:42:43 +00:00
|
|
|
#include "atom/browser/atom_navigation_throttle.h"
|
2018-11-08 15:57:28 +00:00
|
|
|
#include "atom/browser/browser.h"
|
2017-04-06 03:15:27 +00:00
|
|
|
#include "atom/browser/child_web_contents_tracker.h"
|
2016-06-01 05:54:41 +00:00
|
|
|
#include "atom/browser/lib/bluetooth_chooser.h"
|
2014-12-18 20:58:17 +00:00
|
|
|
#include "atom/browser/native_window.h"
|
2016-04-08 18:19:36 +00:00
|
|
|
#include "atom/browser/net/atom_network_delegate.h"
|
2016-07-03 03:26:43 +00:00
|
|
|
#include "atom/browser/ui/drag_util.h"
|
2018-10-19 13:50:30 +00:00
|
|
|
#include "atom/browser/ui/inspectable_web_contents.h"
|
|
|
|
#include "atom/browser/ui/inspectable_web_contents_view.h"
|
2016-01-23 13:29:47 +00:00
|
|
|
#include "atom/browser/web_contents_permission_helper.h"
|
2015-09-04 16:44:22 +00:00
|
|
|
#include "atom/browser/web_contents_preferences.h"
|
2017-01-30 11:18:40 +00:00
|
|
|
#include "atom/browser/web_contents_zoom_controller.h"
|
2015-06-25 06:28:13 +00:00
|
|
|
#include "atom/browser/web_view_guest_delegate.h"
|
2014-04-25 05:10:16 +00:00
|
|
|
#include "atom/common/api/api_messages.h"
|
2018-04-14 02:04:23 +00:00
|
|
|
#include "atom/common/api/atom_api_native_image.h"
|
2015-08-07 11:34:00 +00:00
|
|
|
#include "atom/common/api/event_emitter_caller.h"
|
2016-04-14 12:52:17 +00:00
|
|
|
#include "atom/common/color_util.h"
|
2016-04-08 06:54:33 +00:00
|
|
|
#include "atom/common/mouse_util.h"
|
2015-09-18 03:06:38 +00:00
|
|
|
#include "atom/common/native_mate_converters/blink_converter.h"
|
2015-08-07 10:10:19 +00:00
|
|
|
#include "atom/common/native_mate_converters/callback.h"
|
2015-10-31 13:39:07 +00:00
|
|
|
#include "atom/common/native_mate_converters/content_converter.h"
|
2015-07-24 09:39:11 +00:00
|
|
|
#include "atom/common/native_mate_converters/file_path_converter.h"
|
2014-10-24 06:37:14 +00:00
|
|
|
#include "atom/common/native_mate_converters/gfx_converter.h"
|
2014-04-24 09:00:41 +00:00
|
|
|
#include "atom/common/native_mate_converters/gurl_converter.h"
|
2015-04-20 06:50:04 +00:00
|
|
|
#include "atom/common/native_mate_converters/image_converter.h"
|
2016-10-13 15:14:23 +00:00
|
|
|
#include "atom/common/native_mate_converters/net_converter.h"
|
2018-04-12 20:20:01 +00:00
|
|
|
#include "atom/common/native_mate_converters/network_converter.h"
|
2014-04-24 09:00:41 +00:00
|
|
|
#include "atom/common/native_mate_converters/string16_converter.h"
|
2014-04-25 05:10:16 +00:00
|
|
|
#include "atom/common/native_mate_converters/value_converter.h"
|
2016-04-08 06:54:33 +00:00
|
|
|
#include "atom/common/options_switches.h"
|
2017-08-07 22:38:17 +00:00
|
|
|
#include "base/message_loop/message_loop.h"
|
2014-04-25 08:13:16 +00:00
|
|
|
#include "base/strings/utf_string_conversions.h"
|
2018-09-18 18:00:31 +00:00
|
|
|
#include "base/threading/thread_restrictions.h"
|
2016-11-30 07:30:03 +00:00
|
|
|
#include "base/threading/thread_task_runner_handle.h"
|
2017-02-06 09:18:04 +00:00
|
|
|
#include "base/values.h"
|
2018-04-02 03:47:00 +00:00
|
|
|
#include "chrome/browser/browser_process.h"
|
2017-01-24 01:51:11 +00:00
|
|
|
#include "chrome/browser/ssl/security_state_tab_helper.h"
|
2018-04-26 10:17:55 +00:00
|
|
|
#include "content/browser/frame_host/frame_tree_node.h"
|
|
|
|
#include "content/browser/frame_host/render_frame_host_manager.h"
|
2016-04-08 06:54:33 +00:00
|
|
|
#include "content/browser/renderer_host/render_widget_host_impl.h"
|
2017-03-04 02:09:16 +00:00
|
|
|
#include "content/browser/renderer_host/render_widget_host_view_base.h"
|
2019-01-12 01:00:43 +00:00
|
|
|
#include "content/common/widget_messages.h"
|
2017-06-30 22:47:41 +00:00
|
|
|
#include "content/public/browser/child_process_security_policy.h"
|
2018-04-12 07:30:51 +00:00
|
|
|
#include "content/public/browser/download_request_utils.h"
|
2015-04-20 06:50:04 +00:00
|
|
|
#include "content/public/browser/favicon_status.h"
|
2015-09-18 06:20:31 +00:00
|
|
|
#include "content/public/browser/native_web_keyboard_event.h"
|
2014-12-09 22:38:43 +00:00
|
|
|
#include "content/public/browser/navigation_details.h"
|
2015-03-05 14:39:27 +00:00
|
|
|
#include "content/public/browser/navigation_entry.h"
|
2016-07-14 00:19:28 +00:00
|
|
|
#include "content/public/browser/navigation_handle.h"
|
2015-05-12 19:35:56 +00:00
|
|
|
#include "content/public/browser/plugin_service.h"
|
2014-07-28 07:29:51 +00:00
|
|
|
#include "content/public/browser/render_frame_host.h"
|
2018-03-16 07:28:06 +00:00
|
|
|
#include "content/public/browser/render_process_host.h"
|
|
|
|
#include "content/public/browser/render_view_host.h"
|
2016-03-08 14:28:53 +00:00
|
|
|
#include "content/public/browser/render_widget_host.h"
|
2015-09-18 06:20:31 +00:00
|
|
|
#include "content/public/browser/render_widget_host_view.h"
|
2015-04-30 13:15:19 +00:00
|
|
|
#include "content/public/browser/service_worker_context.h"
|
2014-10-24 12:20:56 +00:00
|
|
|
#include "content/public/browser/site_instance.h"
|
2016-08-26 22:30:02 +00:00
|
|
|
#include "content/public/browser/storage_partition.h"
|
2014-04-24 08:45:25 +00:00
|
|
|
#include "content/public/browser/web_contents.h"
|
2015-11-02 15:28:45 +00:00
|
|
|
#include "content/public/common/context_menu_params.h"
|
2017-02-06 09:18:04 +00:00
|
|
|
#include "native_mate/converter.h"
|
2014-10-23 06:04:13 +00:00
|
|
|
#include "native_mate/dictionary.h"
|
2014-04-24 08:45:25 +00:00
|
|
|
#include "native_mate/object_template_builder.h"
|
2015-05-20 10:44:57 +00:00
|
|
|
#include "net/url_request/url_request_context.h"
|
2019-01-12 01:00:43 +00:00
|
|
|
#include "third_party/blink/public/mojom/frame/find_in_page.mojom.h"
|
2018-07-20 16:08:18 +00:00
|
|
|
#include "third_party/blink/public/platform/web_input_event.h"
|
2016-07-21 12:03:38 +00:00
|
|
|
#include "ui/display/screen.h"
|
2017-04-13 10:21:30 +00:00
|
|
|
#include "ui/events/base_event_utils.h"
|
2014-04-24 08:45:25 +00:00
|
|
|
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
|
|
|
#include "atom/browser/osr/osr_output_device.h"
|
|
|
|
#include "atom/browser/osr/osr_render_widget_host_view.h"
|
|
|
|
#include "atom/browser/osr/osr_web_contents_view.h"
|
|
|
|
#endif
|
|
|
|
|
2016-08-15 21:13:24 +00:00
|
|
|
#if !defined(OS_MACOSX)
|
|
|
|
#include "ui/aura/window.h"
|
|
|
|
#endif
|
2014-10-23 06:04:13 +00:00
|
|
|
|
2018-02-22 08:56:48 +00:00
|
|
|
#if defined(OS_LINUX) || defined(OS_WIN)
|
|
|
|
#include "content/public/common/renderer_preferences.h"
|
|
|
|
#include "ui/gfx/font_render_params.h"
|
|
|
|
#endif
|
|
|
|
|
2018-10-13 01:57:04 +00:00
|
|
|
#if BUILDFLAG(ENABLE_PRINTING)
|
|
|
|
#include "chrome/browser/printing/print_view_manager_basic.h"
|
2018-11-09 03:42:34 +00:00
|
|
|
#include "components/printing/common/print_messages.h"
|
2018-10-13 01:57:04 +00:00
|
|
|
#endif
|
|
|
|
|
2016-09-06 08:24:37 +00:00
|
|
|
#include "atom/common/node_includes.h"
|
|
|
|
|
2015-05-29 03:12:55 +00:00
|
|
|
namespace mate {
|
|
|
|
|
2018-11-09 03:42:34 +00:00
|
|
|
#if BUILDFLAG(ENABLE_PRINTING)
|
2018-04-18 01:55:30 +00:00
|
|
|
template <>
|
2017-02-06 09:18:04 +00:00
|
|
|
struct Converter<printing::PrinterBasicInfo> {
|
2017-05-18 17:26:22 +00:00
|
|
|
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
|
|
|
|
const printing::PrinterBasicInfo& val) {
|
|
|
|
mate::Dictionary dict(isolate, v8::Object::New(isolate));
|
|
|
|
dict.Set("name", val.printer_name);
|
|
|
|
dict.Set("description", val.printer_description);
|
|
|
|
dict.Set("status", val.printer_status);
|
2017-05-18 17:51:44 +00:00
|
|
|
dict.Set("isDefault", val.is_default ? true : false);
|
2017-05-18 17:26:22 +00:00
|
|
|
dict.Set("options", val.options);
|
|
|
|
return dict.GetHandle();
|
|
|
|
}
|
2017-02-06 09:18:04 +00:00
|
|
|
};
|
2018-11-09 03:42:34 +00:00
|
|
|
#endif
|
2017-02-06 09:18:04 +00:00
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
template <>
|
2015-06-23 08:34:22 +00:00
|
|
|
struct Converter<WindowOpenDisposition> {
|
|
|
|
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
|
|
|
|
WindowOpenDisposition val) {
|
|
|
|
std::string disposition = "other";
|
|
|
|
switch (val) {
|
2017-01-24 07:50:02 +00:00
|
|
|
case WindowOpenDisposition::CURRENT_TAB:
|
|
|
|
disposition = "default";
|
|
|
|
break;
|
|
|
|
case WindowOpenDisposition::NEW_FOREGROUND_TAB:
|
|
|
|
disposition = "foreground-tab";
|
|
|
|
break;
|
|
|
|
case WindowOpenDisposition::NEW_BACKGROUND_TAB:
|
|
|
|
disposition = "background-tab";
|
|
|
|
break;
|
|
|
|
case WindowOpenDisposition::NEW_POPUP:
|
|
|
|
case WindowOpenDisposition::NEW_WINDOW:
|
|
|
|
disposition = "new-window";
|
|
|
|
break;
|
|
|
|
case WindowOpenDisposition::SAVE_TO_DISK:
|
|
|
|
disposition = "save-to-disk";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2015-06-23 08:34:22 +00:00
|
|
|
}
|
|
|
|
return mate::ConvertToV8(isolate, disposition);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
template <>
|
2015-10-14 04:41:31 +00:00
|
|
|
struct Converter<content::SavePageType> {
|
2018-04-18 01:55:30 +00:00
|
|
|
static bool FromV8(v8::Isolate* isolate,
|
|
|
|
v8::Local<v8::Value> val,
|
2015-10-14 04:41:31 +00:00
|
|
|
content::SavePageType* out) {
|
|
|
|
std::string save_type;
|
|
|
|
if (!ConvertFromV8(isolate, val, &save_type))
|
|
|
|
return false;
|
2015-12-07 11:56:23 +00:00
|
|
|
save_type = base::ToLowerASCII(save_type);
|
2015-11-02 13:19:00 +00:00
|
|
|
if (save_type == "htmlonly") {
|
2015-10-14 04:41:31 +00:00
|
|
|
*out = content::SAVE_PAGE_TYPE_AS_ONLY_HTML;
|
2015-11-02 13:19:00 +00:00
|
|
|
} else if (save_type == "htmlcomplete") {
|
2015-10-14 04:41:31 +00:00
|
|
|
*out = content::SAVE_PAGE_TYPE_AS_COMPLETE_HTML;
|
2015-11-02 13:19:00 +00:00
|
|
|
} else if (save_type == "mhtml") {
|
2015-10-14 04:41:31 +00:00
|
|
|
*out = content::SAVE_PAGE_TYPE_AS_MHTML;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
template <>
|
2016-06-14 16:09:54 +00:00
|
|
|
struct Converter<atom::api::WebContents::Type> {
|
|
|
|
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
|
|
|
|
atom::api::WebContents::Type val) {
|
2016-06-14 17:05:25 +00:00
|
|
|
using Type = atom::api::WebContents::Type;
|
2016-06-14 16:09:54 +00:00
|
|
|
std::string type = "";
|
|
|
|
switch (val) {
|
2018-04-18 01:55:30 +00:00
|
|
|
case Type::BACKGROUND_PAGE:
|
|
|
|
type = "backgroundPage";
|
|
|
|
break;
|
|
|
|
case Type::BROWSER_WINDOW:
|
|
|
|
type = "window";
|
|
|
|
break;
|
|
|
|
case Type::BROWSER_VIEW:
|
|
|
|
type = "browserView";
|
|
|
|
break;
|
|
|
|
case Type::REMOTE:
|
|
|
|
type = "remote";
|
|
|
|
break;
|
|
|
|
case Type::WEB_VIEW:
|
|
|
|
type = "webview";
|
|
|
|
break;
|
|
|
|
case Type::OFF_SCREEN:
|
|
|
|
type = "offscreen";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2016-06-14 16:09:54 +00:00
|
|
|
}
|
|
|
|
return mate::ConvertToV8(isolate, type);
|
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
static bool FromV8(v8::Isolate* isolate,
|
|
|
|
v8::Local<v8::Value> val,
|
2016-06-14 16:23:03 +00:00
|
|
|
atom::api::WebContents::Type* out) {
|
2016-06-14 17:05:25 +00:00
|
|
|
using Type = atom::api::WebContents::Type;
|
2016-06-14 16:23:03 +00:00
|
|
|
std::string type;
|
|
|
|
if (!ConvertFromV8(isolate, val, &type))
|
|
|
|
return false;
|
Implement initial, experimental BrowserView API
Right now, `<webview>` is the only way to embed additional content in a
`BrowserWindow`. Unfortunately `<webview>` suffers from a [number of
problems](https://github.com/electron/electron/issues?utf8=%E2%9C%93&q=is%3Aissue%20is%3Aopen%20label%3Awebview%20).
To make matters worse, many of these are upstream Chromium bugs instead
of Electron-specific bugs.
For us at [Figma](https://www.figma.com), the main issue is very slow
performance.
Despite the upstream improvements to `<webview>` through the OOPIF work, it is
probable that there will continue to be `<webview>`-specific bugs in the
future.
Therefore, this introduces a `<webview>` alternative to called `BrowserView`,
which...
- is a thin wrapper around `api::WebContents` (so bugs in `BrowserView` will
likely also be bugs in `BrowserWindow` web contents)
- is instantiated in the main process like `BrowserWindow` (and unlike
`<webview>`, which lives in the DOM of a `BrowserWindow` web contents)
- needs to be added to a `BrowserWindow` to display something on the screen
This implements the most basic API. The API is expected to evolve and change in
the near future and has consequently been marked as experimental. Please do not
use this API in production unless you are prepared to deal with breaking
changes.
In the future, we will want to change the API to support multiple
`BrowserView`s per window. We will also want to consider z-ordering
auto-resizing, and possibly even nested views.
2017-04-11 17:47:30 +00:00
|
|
|
if (type == "backgroundPage") {
|
2016-06-14 17:05:25 +00:00
|
|
|
*out = Type::BACKGROUND_PAGE;
|
Implement initial, experimental BrowserView API
Right now, `<webview>` is the only way to embed additional content in a
`BrowserWindow`. Unfortunately `<webview>` suffers from a [number of
problems](https://github.com/electron/electron/issues?utf8=%E2%9C%93&q=is%3Aissue%20is%3Aopen%20label%3Awebview%20).
To make matters worse, many of these are upstream Chromium bugs instead
of Electron-specific bugs.
For us at [Figma](https://www.figma.com), the main issue is very slow
performance.
Despite the upstream improvements to `<webview>` through the OOPIF work, it is
probable that there will continue to be `<webview>`-specific bugs in the
future.
Therefore, this introduces a `<webview>` alternative to called `BrowserView`,
which...
- is a thin wrapper around `api::WebContents` (so bugs in `BrowserView` will
likely also be bugs in `BrowserWindow` web contents)
- is instantiated in the main process like `BrowserWindow` (and unlike
`<webview>`, which lives in the DOM of a `BrowserWindow` web contents)
- needs to be added to a `BrowserWindow` to display something on the screen
This implements the most basic API. The API is expected to evolve and change in
the near future and has consequently been marked as experimental. Please do not
use this API in production unless you are prepared to deal with breaking
changes.
In the future, we will want to change the API to support multiple
`BrowserView`s per window. We will also want to consider z-ordering
auto-resizing, and possibly even nested views.
2017-04-11 17:47:30 +00:00
|
|
|
} else if (type == "browserView") {
|
|
|
|
*out = Type::BROWSER_VIEW;
|
|
|
|
} else if (type == "webview") {
|
|
|
|
*out = Type::WEB_VIEW;
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2016-07-28 10:10:56 +00:00
|
|
|
} else if (type == "offscreen") {
|
|
|
|
*out = Type::OFF_SCREEN;
|
2017-06-26 09:13:05 +00:00
|
|
|
#endif
|
2016-06-14 16:23:03 +00:00
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
2016-06-14 16:09:54 +00:00
|
|
|
|
2015-05-29 03:12:55 +00:00
|
|
|
} // namespace mate
|
|
|
|
|
2014-04-24 08:45:25 +00:00
|
|
|
namespace atom {
|
|
|
|
|
|
|
|
namespace api {
|
|
|
|
|
2014-10-23 05:31:10 +00:00
|
|
|
namespace {
|
|
|
|
|
2015-04-30 13:15:19 +00:00
|
|
|
content::ServiceWorkerContext* GetServiceWorkerContext(
|
2019-01-09 20:12:44 +00:00
|
|
|
content::WebContents* web_contents) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* context = web_contents->GetBrowserContext();
|
|
|
|
auto* site_instance = web_contents->GetSiteInstance();
|
2015-04-30 13:15:19 +00:00
|
|
|
if (!context || !site_instance)
|
|
|
|
return nullptr;
|
|
|
|
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* storage_partition =
|
2015-06-05 07:41:03 +00:00
|
|
|
content::BrowserContext::GetStoragePartition(context, site_instance);
|
|
|
|
if (!storage_partition)
|
|
|
|
return nullptr;
|
2015-04-30 13:15:19 +00:00
|
|
|
|
|
|
|
return storage_partition->GetServiceWorkerContext();
|
|
|
|
}
|
|
|
|
|
2016-07-06 16:32:58 +00:00
|
|
|
// Called when CapturePage is done.
|
2018-11-28 04:50:53 +00:00
|
|
|
void OnCapturePageDone(scoped_refptr<util::Promise> promise,
|
2018-04-12 12:37:16 +00:00
|
|
|
const SkBitmap& bitmap) {
|
2018-04-10 10:07:15 +00:00
|
|
|
// Hack to enable transparency in captured image
|
|
|
|
// TODO(nitsakh) Remove hack once fixed in chromium
|
|
|
|
const_cast<SkBitmap&>(bitmap).setAlphaType(kPremul_SkAlphaType);
|
2018-11-28 04:50:53 +00:00
|
|
|
promise->Resolve(gfx::Image::CreateFrom1xBitmap(bitmap));
|
2016-07-06 16:32:58 +00:00
|
|
|
}
|
|
|
|
|
2014-10-23 05:31:10 +00:00
|
|
|
} // namespace
|
|
|
|
|
2018-03-09 09:31:09 +00:00
|
|
|
struct WebContents::FrameDispatchHelper {
|
|
|
|
WebContents* api_web_contents;
|
|
|
|
content::RenderFrameHost* rfh;
|
|
|
|
|
|
|
|
bool Send(IPC::Message* msg) { return rfh->Send(msg); }
|
|
|
|
|
|
|
|
void OnSetTemporaryZoomLevel(double level, IPC::Message* reply_msg) {
|
|
|
|
api_web_contents->OnSetTemporaryZoomLevel(rfh, level, reply_msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OnGetZoomLevel(IPC::Message* reply_msg) {
|
|
|
|
api_web_contents->OnGetZoomLevel(rfh, reply_msg);
|
|
|
|
}
|
|
|
|
|
2018-08-24 15:30:37 +00:00
|
|
|
void OnRendererMessageSync(const std::string& channel,
|
2018-03-09 09:31:09 +00:00
|
|
|
const base::ListValue& args,
|
|
|
|
IPC::Message* message) {
|
|
|
|
api_web_contents->OnRendererMessageSync(rfh, channel, args, message);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-10-19 08:52:07 +00:00
|
|
|
WebContents::WebContents(v8::Isolate* isolate,
|
|
|
|
content::WebContents* web_contents)
|
|
|
|
: content::WebContentsObserver(web_contents), type_(REMOTE) {
|
|
|
|
web_contents->SetUserAgentOverride(GetBrowserContext()->GetUserAgent(),
|
|
|
|
false);
|
|
|
|
Init(isolate);
|
|
|
|
AttachAsUserData(web_contents);
|
|
|
|
InitZoomController(web_contents, mate::Dictionary::CreateEmpty(isolate));
|
|
|
|
}
|
|
|
|
|
2016-04-25 01:17:54 +00:00
|
|
|
WebContents::WebContents(v8::Isolate* isolate,
|
2018-10-22 18:02:25 +00:00
|
|
|
std::unique_ptr<content::WebContents> web_contents,
|
2016-08-17 00:15:10 +00:00
|
|
|
Type type)
|
2018-10-22 18:02:25 +00:00
|
|
|
: content::WebContentsObserver(web_contents.get()), type_(type) {
|
2018-10-19 08:52:07 +00:00
|
|
|
DCHECK(type != REMOTE) << "Can't take ownership of a remote WebContents";
|
|
|
|
auto session = Session::CreateFrom(isolate, GetBrowserContext());
|
|
|
|
session_.Reset(isolate, session.ToV8());
|
2018-10-22 18:02:25 +00:00
|
|
|
InitWithSessionAndOptions(isolate, std::move(web_contents), session,
|
2018-10-19 08:52:07 +00:00
|
|
|
mate::Dictionary::CreateEmpty(isolate));
|
2014-10-23 06:04:13 +00:00
|
|
|
}
|
|
|
|
|
2018-05-21 22:18:38 +00:00
|
|
|
WebContents::WebContents(v8::Isolate* isolate,
|
|
|
|
const mate::Dictionary& options) {
|
2016-04-08 06:54:33 +00:00
|
|
|
// Read options.
|
|
|
|
options.Get("backgroundThrottling", &background_throttling_);
|
|
|
|
|
2016-06-16 01:54:28 +00:00
|
|
|
// FIXME(zcbenz): We should read "type" parameter for better design, but
|
|
|
|
// on Windows we have encountered a compiler bug that if we read "type"
|
|
|
|
// from |options| and then set |type_|, a memory corruption will happen
|
|
|
|
// and Electron will soon crash.
|
|
|
|
// Remvoe this after we upgraded to use VS 2015 Update 3.
|
|
|
|
bool b = false;
|
|
|
|
if (options.Get("isGuest", &b) && b)
|
|
|
|
type_ = WEB_VIEW;
|
|
|
|
else if (options.Get("isBackgroundPage", &b) && b)
|
|
|
|
type_ = BACKGROUND_PAGE;
|
Implement initial, experimental BrowserView API
Right now, `<webview>` is the only way to embed additional content in a
`BrowserWindow`. Unfortunately `<webview>` suffers from a [number of
problems](https://github.com/electron/electron/issues?utf8=%E2%9C%93&q=is%3Aissue%20is%3Aopen%20label%3Awebview%20).
To make matters worse, many of these are upstream Chromium bugs instead
of Electron-specific bugs.
For us at [Figma](https://www.figma.com), the main issue is very slow
performance.
Despite the upstream improvements to `<webview>` through the OOPIF work, it is
probable that there will continue to be `<webview>`-specific bugs in the
future.
Therefore, this introduces a `<webview>` alternative to called `BrowserView`,
which...
- is a thin wrapper around `api::WebContents` (so bugs in `BrowserView` will
likely also be bugs in `BrowserWindow` web contents)
- is instantiated in the main process like `BrowserWindow` (and unlike
`<webview>`, which lives in the DOM of a `BrowserWindow` web contents)
- needs to be added to a `BrowserWindow` to display something on the screen
This implements the most basic API. The API is expected to evolve and change in
the near future and has consequently been marked as experimental. Please do not
use this API in production unless you are prepared to deal with breaking
changes.
In the future, we will want to change the API to support multiple
`BrowserView`s per window. We will also want to consider z-ordering
auto-resizing, and possibly even nested views.
2017-04-11 17:47:30 +00:00
|
|
|
else if (options.Get("isBrowserView", &b) && b)
|
|
|
|
type_ = BROWSER_VIEW;
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2018-05-29 08:09:51 +00:00
|
|
|
else if (options.Get(options::kOffscreen, &b) && b)
|
2016-07-29 12:50:27 +00:00
|
|
|
type_ = OFF_SCREEN;
|
2017-06-26 09:13:05 +00:00
|
|
|
#endif
|
2015-06-24 15:29:32 +00:00
|
|
|
|
2017-03-04 02:09:16 +00:00
|
|
|
// Init embedder earlier
|
|
|
|
options.Get("embedder", &embedder_);
|
|
|
|
|
2016-09-07 02:16:52 +00:00
|
|
|
// Whether to enable DevTools.
|
|
|
|
options.Get("devTools", &enable_devtools_);
|
2016-09-05 08:27:56 +00:00
|
|
|
|
2015-09-05 14:39:48 +00:00
|
|
|
// Obtain the session.
|
|
|
|
std::string partition;
|
|
|
|
mate::Handle<api::Session> session;
|
2018-02-12 19:46:29 +00:00
|
|
|
if (options.Get("session", &session) && !session.IsEmpty()) {
|
2016-07-12 12:01:49 +00:00
|
|
|
} else if (options.Get("partition", &partition)) {
|
|
|
|
session = Session::FromPartition(isolate, partition);
|
2015-09-05 14:39:48 +00:00
|
|
|
} else {
|
|
|
|
// Use the default session if not specified.
|
2016-07-12 12:01:49 +00:00
|
|
|
session = Session::FromPartition(isolate, "");
|
2015-09-05 14:39:48 +00:00
|
|
|
}
|
|
|
|
session_.Reset(isolate, session.ToV8());
|
|
|
|
|
2018-09-15 00:16:22 +00:00
|
|
|
std::unique_ptr<content::WebContents> web_contents;
|
2016-06-14 16:23:03 +00:00
|
|
|
if (IsGuest()) {
|
2016-05-23 03:28:59 +00:00
|
|
|
scoped_refptr<content::SiteInstance> site_instance =
|
2018-04-18 01:55:30 +00:00
|
|
|
content::SiteInstance::CreateForURL(session->browser_context(),
|
|
|
|
GURL("chrome-guest://fake-host"));
|
|
|
|
content::WebContents::CreateParams params(session->browser_context(),
|
|
|
|
site_instance);
|
2018-08-16 22:57:40 +00:00
|
|
|
guest_delegate_.reset(
|
|
|
|
new WebViewGuestDelegate(embedder_->web_contents(), this));
|
2015-06-25 06:28:13 +00:00
|
|
|
params.guest_delegate = guest_delegate_.get();
|
2017-03-05 15:18:57 +00:00
|
|
|
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2017-03-04 02:09:16 +00:00
|
|
|
if (embedder_ && embedder_->IsOffScreen()) {
|
2018-04-18 01:55:30 +00:00
|
|
|
auto* view = new OffScreenWebContentsView(
|
|
|
|
false, base::Bind(&WebContents::OnPaint, base::Unretained(this)));
|
2017-03-04 02:09:16 +00:00
|
|
|
params.view = view;
|
|
|
|
params.delegate_view = view;
|
2017-03-05 15:18:57 +00:00
|
|
|
|
2017-03-04 02:09:16 +00:00
|
|
|
web_contents = content::WebContents::Create(params);
|
2018-10-05 18:03:35 +00:00
|
|
|
view->SetWebContents(web_contents.get());
|
2017-03-04 02:09:16 +00:00
|
|
|
} else {
|
2017-06-26 09:13:05 +00:00
|
|
|
#endif
|
2017-03-04 02:09:16 +00:00
|
|
|
web_contents = content::WebContents::Create(params);
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2017-03-04 02:09:16 +00:00
|
|
|
}
|
2016-07-31 10:19:56 +00:00
|
|
|
} else if (IsOffScreen()) {
|
2016-08-01 11:27:39 +00:00
|
|
|
bool transparent = false;
|
|
|
|
options.Get("transparent", &transparent);
|
|
|
|
|
2016-07-29 12:50:27 +00:00
|
|
|
content::WebContents::CreateParams params(session->browser_context());
|
2018-04-18 01:55:30 +00:00
|
|
|
auto* view = new OffScreenWebContentsView(
|
|
|
|
transparent, base::Bind(&WebContents::OnPaint, base::Unretained(this)));
|
2016-08-03 04:04:36 +00:00
|
|
|
params.view = view;
|
|
|
|
params.delegate_view = view;
|
2016-07-25 13:55:00 +00:00
|
|
|
|
2015-06-24 15:29:32 +00:00
|
|
|
web_contents = content::WebContents::Create(params);
|
2018-10-05 18:03:35 +00:00
|
|
|
view->SetWebContents(web_contents.get());
|
2017-06-26 09:13:05 +00:00
|
|
|
#endif
|
2016-07-29 12:50:27 +00:00
|
|
|
} else {
|
|
|
|
content::WebContents::CreateParams params(session->browser_context());
|
2016-07-27 23:58:23 +00:00
|
|
|
web_contents = content::WebContents::Create(params);
|
2015-06-24 14:14:46 +00:00
|
|
|
}
|
2015-06-09 07:50:20 +00:00
|
|
|
|
2018-10-22 18:02:25 +00:00
|
|
|
InitWithSessionAndOptions(isolate, std::move(web_contents), session, options);
|
2016-08-17 00:15:10 +00:00
|
|
|
}
|
|
|
|
|
2017-10-21 20:21:24 +00:00
|
|
|
void WebContents::InitZoomController(content::WebContents* web_contents,
|
|
|
|
const mate::Dictionary& options) {
|
|
|
|
WebContentsZoomController::CreateForWebContents(web_contents);
|
|
|
|
zoom_controller_ = WebContentsZoomController::FromWebContents(web_contents);
|
|
|
|
double zoom_factor;
|
|
|
|
if (options.Get(options::kZoomFactor, &zoom_factor))
|
|
|
|
zoom_controller_->SetDefaultZoomFactor(zoom_factor);
|
|
|
|
}
|
|
|
|
|
2018-10-22 18:02:25 +00:00
|
|
|
void WebContents::InitWithSessionAndOptions(
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
std::unique_ptr<content::WebContents> owned_web_contents,
|
|
|
|
mate::Handle<api::Session> session,
|
|
|
|
const mate::Dictionary& options) {
|
|
|
|
Observe(owned_web_contents.get());
|
|
|
|
// TODO(zcbenz): Make InitWithWebContents take unique_ptr.
|
|
|
|
// At the time of writing we are going through a refactoring and I don't want
|
|
|
|
// to make other people's work harder.
|
|
|
|
InitWithWebContents(owned_web_contents.release(), session->browser_context(),
|
|
|
|
IsGuest());
|
2015-06-05 07:01:51 +00:00
|
|
|
|
2015-10-01 03:14:19 +00:00
|
|
|
managed_web_contents()->GetView()->SetDelegate(this);
|
|
|
|
|
2018-10-22 18:02:25 +00:00
|
|
|
auto* prefs = web_contents()->GetMutableRendererPrefs();
|
2018-04-02 03:47:00 +00:00
|
|
|
prefs->accept_languages = g_browser_process->GetApplicationLocale();
|
|
|
|
|
2018-02-22 08:56:48 +00:00
|
|
|
#if defined(OS_LINUX) || defined(OS_WIN)
|
|
|
|
// Update font settings.
|
2018-04-18 01:55:30 +00:00
|
|
|
CR_DEFINE_STATIC_LOCAL(
|
|
|
|
const gfx::FontRenderParams, params,
|
2018-02-22 08:56:48 +00:00
|
|
|
(gfx::GetFontRenderParams(gfx::FontRenderParamsQuery(), nullptr)));
|
|
|
|
prefs->should_antialias_text = params.antialiasing;
|
|
|
|
prefs->use_subpixel_positioning = params.subpixel_positioning;
|
|
|
|
prefs->hinting = params.hinting;
|
|
|
|
prefs->use_autohinter = params.autohinter;
|
|
|
|
prefs->use_bitmaps = params.use_bitmaps;
|
|
|
|
prefs->subpixel_rendering = params.subpixel_rendering;
|
|
|
|
#endif
|
|
|
|
|
2015-09-22 13:56:56 +00:00
|
|
|
// Save the preferences in C++.
|
2018-10-22 18:02:25 +00:00
|
|
|
new WebContentsPreferences(web_contents(), options);
|
2015-09-04 16:44:22 +00:00
|
|
|
|
2017-01-30 11:18:40 +00:00
|
|
|
// Initialize permission helper.
|
2018-10-22 18:02:25 +00:00
|
|
|
WebContentsPermissionHelper::CreateForWebContents(web_contents());
|
2017-01-30 11:18:40 +00:00
|
|
|
// Initialize security state client.
|
2018-10-22 18:02:25 +00:00
|
|
|
SecurityStateTabHelper::CreateForWebContents(web_contents());
|
2017-01-30 11:18:40 +00:00
|
|
|
// Initialize zoom controller.
|
2018-10-22 18:02:25 +00:00
|
|
|
InitZoomController(web_contents(), options);
|
2016-01-23 13:29:47 +00:00
|
|
|
|
2018-10-22 18:02:25 +00:00
|
|
|
web_contents()->SetUserAgentOverride(GetBrowserContext()->GetUserAgent(),
|
|
|
|
false);
|
2015-07-14 19:13:25 +00:00
|
|
|
|
2016-06-14 16:23:03 +00:00
|
|
|
if (IsGuest()) {
|
2015-06-24 15:29:32 +00:00
|
|
|
NativeWindow* owner_window = nullptr;
|
2017-03-04 02:09:16 +00:00
|
|
|
if (embedder_) {
|
2015-06-24 15:29:32 +00:00
|
|
|
// New WebContents's owner_window is the embedder's owner_window.
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* relay =
|
2016-02-17 08:52:19 +00:00
|
|
|
NativeWindowRelay::FromWebContents(embedder_->web_contents());
|
2015-06-24 15:29:32 +00:00
|
|
|
if (relay)
|
2018-10-02 22:14:43 +00:00
|
|
|
owner_window = relay->GetNativeWindow();
|
2015-06-24 15:29:32 +00:00
|
|
|
}
|
|
|
|
if (owner_window)
|
|
|
|
SetOwnerWindow(owner_window);
|
|
|
|
}
|
2016-04-25 01:17:54 +00:00
|
|
|
|
|
|
|
Init(isolate);
|
2018-10-22 18:02:25 +00:00
|
|
|
AttachAsUserData(web_contents());
|
2014-10-23 06:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
WebContents::~WebContents() {
|
2016-01-13 09:59:59 +00:00
|
|
|
// The destroy() is called.
|
|
|
|
if (managed_web_contents()) {
|
2018-02-22 07:52:08 +00:00
|
|
|
managed_web_contents()->GetView()->SetDelegate(nullptr);
|
|
|
|
|
2016-02-22 07:20:23 +00:00
|
|
|
RenderViewDeleted(web_contents()->GetRenderViewHost());
|
2017-04-03 20:03:51 +00:00
|
|
|
|
2018-09-07 06:41:48 +00:00
|
|
|
if (type_ == WEB_VIEW) {
|
2018-11-08 15:57:28 +00:00
|
|
|
// For webview simply destroy the WebContents immediately.
|
|
|
|
// TODO(zcbenz): Add an internal API for webview instead of using
|
|
|
|
// destroy(), so we don't have to add a special branch here.
|
|
|
|
DestroyWebContents(false /* async */);
|
|
|
|
} else if (type_ == BROWSER_WINDOW && owner_window()) {
|
|
|
|
// For BrowserWindow we should close the window and clean up everything
|
|
|
|
// before WebContents is destroyed.
|
|
|
|
for (ExtendedWebContentsObserver& observer : observers_)
|
|
|
|
observer.OnCloseContents();
|
|
|
|
// BrowserWindow destroys WebContents asynchronously, manually emit the
|
|
|
|
// destroyed event here.
|
|
|
|
WebContentsDestroyed();
|
|
|
|
} else if (Browser::Get()->is_shutting_down()) {
|
|
|
|
// Destroy WebContents directly when app is shutting down.
|
2018-09-07 06:41:48 +00:00
|
|
|
DestroyWebContents(false /* async */);
|
2017-04-03 20:03:51 +00:00
|
|
|
} else {
|
2018-11-08 15:57:28 +00:00
|
|
|
// Destroy WebContents asynchronously unless app is shutting down,
|
|
|
|
// because destroy() might be called inside WebContents's event handler.
|
|
|
|
DestroyWebContents(true /* async */);
|
2018-09-07 06:41:48 +00:00
|
|
|
// The WebContentsDestroyed will not be called automatically because we
|
|
|
|
// destroy the webContents in the next tick. So we have to manually
|
|
|
|
// call it here to make sure "destroyed" event is emitted.
|
|
|
|
WebContentsDestroyed();
|
2017-04-03 20:03:51 +00:00
|
|
|
}
|
2016-01-13 09:59:59 +00:00
|
|
|
}
|
2014-04-24 08:45:25 +00:00
|
|
|
}
|
|
|
|
|
2017-04-03 20:03:51 +00:00
|
|
|
void WebContents::DestroyWebContents(bool async) {
|
2017-03-18 19:40:30 +00:00
|
|
|
// This event is only for internal use, which is emitted when WebContents is
|
|
|
|
// being destroyed.
|
|
|
|
Emit("will-destroy");
|
2017-04-03 20:03:51 +00:00
|
|
|
ResetManagedWebContents(async);
|
2017-03-18 19:40:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-23 09:59:40 +00:00
|
|
|
bool WebContents::DidAddMessageToConsole(content::WebContents* source,
|
|
|
|
int32_t level,
|
|
|
|
const base::string16& message,
|
|
|
|
int32_t line_no,
|
|
|
|
const base::string16& source_id) {
|
2018-02-14 14:28:21 +00:00
|
|
|
return Emit("console-message", level, message, line_no, source_id);
|
2014-10-25 02:01:19 +00:00
|
|
|
}
|
|
|
|
|
2016-10-09 23:30:38 +00:00
|
|
|
void WebContents::OnCreateWindow(
|
|
|
|
const GURL& target_url,
|
2018-04-05 23:13:24 +00:00
|
|
|
const content::Referrer& referrer,
|
2016-10-09 23:30:38 +00:00
|
|
|
const std::string& frame_name,
|
|
|
|
WindowOpenDisposition disposition,
|
2017-01-23 08:44:42 +00:00
|
|
|
const std::vector<std::string>& features,
|
2018-04-11 08:42:14 +00:00
|
|
|
const scoped_refptr<network::ResourceRequestBody>& body) {
|
2016-08-03 03:29:55 +00:00
|
|
|
if (type_ == BROWSER_WINDOW || type_ == OFF_SCREEN)
|
2018-04-05 23:13:24 +00:00
|
|
|
Emit("-new-window", target_url, frame_name, disposition, features, body,
|
|
|
|
referrer);
|
2015-06-25 03:07:23 +00:00
|
|
|
else
|
2016-09-15 13:37:35 +00:00
|
|
|
Emit("new-window", target_url, frame_name, disposition, features);
|
2014-10-25 05:21:09 +00:00
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
void WebContents::WebContentsCreated(content::WebContents* source_contents,
|
|
|
|
int opener_render_process_id,
|
|
|
|
int opener_render_frame_id,
|
|
|
|
const std::string& frame_name,
|
|
|
|
const GURL& target_url,
|
|
|
|
content::WebContents* new_contents) {
|
2018-10-22 18:02:25 +00:00
|
|
|
ChildWebContentsTracker::CreateForWebContents(new_contents);
|
|
|
|
auto* tracker = ChildWebContentsTracker::FromWebContents(new_contents);
|
|
|
|
tracker->url = target_url;
|
|
|
|
tracker->frame_name = frame_name;
|
2016-08-16 00:13:18 +00:00
|
|
|
}
|
|
|
|
|
2018-09-15 00:16:22 +00:00
|
|
|
void WebContents::AddNewContents(
|
|
|
|
content::WebContents* source,
|
|
|
|
std::unique_ptr<content::WebContents> new_contents,
|
|
|
|
WindowOpenDisposition disposition,
|
|
|
|
const gfx::Rect& initial_rect,
|
|
|
|
bool user_gesture,
|
|
|
|
bool* was_blocked) {
|
2018-10-22 18:02:25 +00:00
|
|
|
auto* tracker = ChildWebContentsTracker::FromWebContents(new_contents.get());
|
|
|
|
DCHECK(tracker);
|
|
|
|
|
2016-08-16 00:13:18 +00:00
|
|
|
v8::Locker locker(isolate());
|
|
|
|
v8::HandleScope handle_scope(isolate());
|
2018-10-22 18:02:25 +00:00
|
|
|
auto api_web_contents =
|
|
|
|
CreateAndTake(isolate(), std::move(new_contents), BROWSER_WINDOW);
|
2017-01-10 23:40:07 +00:00
|
|
|
if (Emit("-add-new-contents", api_web_contents, disposition, user_gesture,
|
2018-04-18 01:55:30 +00:00
|
|
|
initial_rect.x(), initial_rect.y(), initial_rect.width(),
|
2018-10-22 18:02:25 +00:00
|
|
|
initial_rect.height(), tracker->url, tracker->frame_name)) {
|
2018-11-08 15:57:28 +00:00
|
|
|
// TODO(zcbenz): Can we make this sync?
|
2017-04-19 16:24:04 +00:00
|
|
|
api_web_contents->DestroyWebContents(true /* async */);
|
2017-01-10 23:40:07 +00:00
|
|
|
}
|
2016-08-16 00:13:18 +00:00
|
|
|
}
|
|
|
|
|
2014-10-24 10:06:32 +00:00
|
|
|
content::WebContents* WebContents::OpenURLFromTab(
|
|
|
|
content::WebContents* source,
|
|
|
|
const content::OpenURLParams& params) {
|
2017-01-24 07:50:02 +00:00
|
|
|
if (params.disposition != WindowOpenDisposition::CURRENT_TAB) {
|
2016-08-03 03:29:55 +00:00
|
|
|
if (type_ == BROWSER_WINDOW || type_ == OFF_SCREEN)
|
2015-06-25 03:07:23 +00:00
|
|
|
Emit("-new-window", params.url, "", params.disposition);
|
|
|
|
else
|
|
|
|
Emit("new-window", params.url, "", params.disposition);
|
2014-12-17 22:55:22 +00:00
|
|
|
return nullptr;
|
2014-11-04 09:59:15 +00:00
|
|
|
}
|
2014-10-24 10:06:32 +00:00
|
|
|
|
2014-12-17 22:55:22 +00:00
|
|
|
// Give user a chance to cancel navigation.
|
2015-01-15 01:51:54 +00:00
|
|
|
if (Emit("will-navigate", params.url))
|
2014-12-17 22:55:22 +00:00
|
|
|
return nullptr;
|
|
|
|
|
2016-12-21 00:51:03 +00:00
|
|
|
// Don't load the URL if the web contents was marked as destroyed from a
|
|
|
|
// will-navigate event listener
|
|
|
|
if (IsDestroyed())
|
|
|
|
return nullptr;
|
|
|
|
|
2015-06-05 09:45:17 +00:00
|
|
|
return CommonWebContentsDelegate::OpenURLFromTab(source, params);
|
2014-10-24 10:06:32 +00:00
|
|
|
}
|
|
|
|
|
2015-06-25 03:07:23 +00:00
|
|
|
void WebContents::BeforeUnloadFired(content::WebContents* tab,
|
|
|
|
bool proceed,
|
|
|
|
bool* proceed_to_fire_unload) {
|
2016-08-03 03:29:55 +00:00
|
|
|
if (type_ == BROWSER_WINDOW || type_ == OFF_SCREEN)
|
2015-06-25 03:07:23 +00:00
|
|
|
*proceed_to_fire_unload = proceed;
|
|
|
|
else
|
|
|
|
*proceed_to_fire_unload = true;
|
|
|
|
}
|
|
|
|
|
2018-10-02 21:56:09 +00:00
|
|
|
void WebContents::SetContentsBounds(content::WebContents* source,
|
|
|
|
const gfx::Rect& pos) {
|
2015-06-25 03:07:23 +00:00
|
|
|
Emit("move", pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::CloseContents(content::WebContents* source) {
|
2015-07-10 02:38:15 +00:00
|
|
|
Emit("close");
|
2018-05-10 20:52:17 +00:00
|
|
|
#if defined(TOOLKIT_VIEWS) && !defined(OS_MACOSX)
|
2018-04-05 00:53:51 +00:00
|
|
|
HideAutofillPopup();
|
|
|
|
#endif
|
2018-02-22 07:52:08 +00:00
|
|
|
if (managed_web_contents())
|
|
|
|
managed_web_contents()->GetView()->SetDelegate(nullptr);
|
|
|
|
for (ExtendedWebContentsObserver& observer : observers_)
|
|
|
|
observer.OnCloseContents();
|
2015-06-25 03:07:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::ActivateContents(content::WebContents* source) {
|
2015-06-25 05:18:36 +00:00
|
|
|
Emit("activate");
|
2015-06-25 03:07:23 +00:00
|
|
|
}
|
|
|
|
|
2016-06-07 06:56:19 +00:00
|
|
|
void WebContents::UpdateTargetURL(content::WebContents* source,
|
|
|
|
const GURL& url) {
|
|
|
|
Emit("update-target-url", url);
|
|
|
|
}
|
|
|
|
|
2019-01-09 20:01:49 +00:00
|
|
|
bool WebContents::HandleKeyboardEvent(
|
2014-10-24 13:46:47 +00:00
|
|
|
content::WebContents* source,
|
|
|
|
const content::NativeWebKeyboardEvent& event) {
|
2016-05-18 04:47:50 +00:00
|
|
|
if (type_ == WEB_VIEW && embedder_) {
|
2015-06-25 03:07:23 +00:00
|
|
|
// Send the unhandled keyboard events back to the embedder.
|
2019-01-09 20:01:49 +00:00
|
|
|
return embedder_->HandleKeyboardEvent(source, event);
|
2016-05-18 04:47:50 +00:00
|
|
|
} else {
|
|
|
|
// Go to the default keyboard handling.
|
2019-01-09 20:01:49 +00:00
|
|
|
return CommonWebContentsDelegate::HandleKeyboardEvent(source, event);
|
2015-06-25 03:07:23 +00:00
|
|
|
}
|
2014-10-24 13:46:47 +00:00
|
|
|
}
|
|
|
|
|
2017-06-16 21:35:43 +00:00
|
|
|
content::KeyboardEventProcessingResult WebContents::PreHandleKeyboardEvent(
|
2016-12-06 22:41:18 +00:00
|
|
|
content::WebContents* source,
|
2017-06-16 21:35:43 +00:00
|
|
|
const content::NativeWebKeyboardEvent& event) {
|
2017-06-16 20:42:33 +00:00
|
|
|
if (event.GetType() == blink::WebInputEvent::Type::kRawKeyDown ||
|
|
|
|
event.GetType() == blink::WebInputEvent::Type::kKeyUp) {
|
2017-06-16 21:35:43 +00:00
|
|
|
bool prevent_default = Emit("before-input-event", event);
|
|
|
|
if (prevent_default) {
|
|
|
|
return content::KeyboardEventProcessingResult::HANDLED;
|
|
|
|
}
|
2017-06-25 19:01:05 +00:00
|
|
|
}
|
2017-06-16 21:35:43 +00:00
|
|
|
|
|
|
|
return content::KeyboardEventProcessingResult::NOT_HANDLED;
|
2016-12-06 22:41:18 +00:00
|
|
|
}
|
|
|
|
|
2018-09-15 00:17:50 +00:00
|
|
|
void WebContents::EnterFullscreenModeForTab(
|
|
|
|
content::WebContents* source,
|
|
|
|
const GURL& origin,
|
|
|
|
const blink::WebFullscreenOptions& options) {
|
2018-04-18 10:09:45 +00:00
|
|
|
auto* permission_helper =
|
|
|
|
WebContentsPermissionHelper::FromWebContents(source);
|
2016-02-01 10:03:38 +00:00
|
|
|
auto callback = base::Bind(&WebContents::OnEnterFullscreenModeForTab,
|
2018-09-15 00:17:50 +00:00
|
|
|
base::Unretained(this), source, origin, options);
|
2016-02-01 10:03:38 +00:00
|
|
|
permission_helper->RequestFullscreenPermission(callback);
|
|
|
|
}
|
|
|
|
|
2018-09-15 00:17:50 +00:00
|
|
|
void WebContents::OnEnterFullscreenModeForTab(
|
|
|
|
content::WebContents* source,
|
|
|
|
const GURL& origin,
|
|
|
|
const blink::WebFullscreenOptions& options,
|
|
|
|
bool allowed) {
|
2016-02-01 10:03:38 +00:00
|
|
|
if (!allowed)
|
|
|
|
return;
|
2018-09-15 00:17:50 +00:00
|
|
|
CommonWebContentsDelegate::EnterFullscreenModeForTab(source, origin, options);
|
2015-06-05 09:27:24 +00:00
|
|
|
Emit("enter-html-full-screen");
|
2015-05-08 06:18:15 +00:00
|
|
|
}
|
|
|
|
|
2015-05-09 12:17:40 +00:00
|
|
|
void WebContents::ExitFullscreenModeForTab(content::WebContents* source) {
|
2015-06-05 09:27:24 +00:00
|
|
|
CommonWebContentsDelegate::ExitFullscreenModeForTab(source);
|
|
|
|
Emit("leave-html-full-screen");
|
2015-05-08 06:18:15 +00:00
|
|
|
}
|
|
|
|
|
2017-01-23 09:59:40 +00:00
|
|
|
void WebContents::RendererUnresponsive(
|
|
|
|
content::WebContents* source,
|
2018-10-02 21:57:14 +00:00
|
|
|
content::RenderWidgetHost* render_widget_host,
|
|
|
|
base::RepeatingClosure hang_monitor_restarter) {
|
2015-06-25 03:07:23 +00:00
|
|
|
Emit("unresponsive");
|
|
|
|
}
|
|
|
|
|
2018-04-11 15:23:16 +00:00
|
|
|
void WebContents::RendererResponsive(
|
|
|
|
content::WebContents* source,
|
|
|
|
content::RenderWidgetHost* render_widget_host) {
|
2015-06-25 03:07:23 +00:00
|
|
|
Emit("responsive");
|
2018-02-22 06:57:03 +00:00
|
|
|
for (ExtendedWebContentsObserver& observer : observers_)
|
|
|
|
observer.OnRendererResponsive();
|
2015-06-25 03:07:23 +00:00
|
|
|
}
|
|
|
|
|
2015-10-31 13:39:07 +00:00
|
|
|
bool WebContents::HandleContextMenu(const content::ContextMenuParams& params) {
|
2016-05-03 00:02:33 +00:00
|
|
|
if (params.custom_context.is_pepper_menu) {
|
2018-04-18 01:55:30 +00:00
|
|
|
Emit("pepper-context-menu", std::make_pair(params, web_contents()),
|
2017-12-20 09:48:09 +00:00
|
|
|
base::Bind(&content::WebContents::NotifyContextMenuClosed,
|
|
|
|
base::Unretained(web_contents()), params.custom_context));
|
2016-05-03 00:02:33 +00:00
|
|
|
} else {
|
|
|
|
Emit("context-menu", std::make_pair(params, web_contents()));
|
|
|
|
}
|
2015-11-02 15:28:45 +00:00
|
|
|
|
2015-10-31 13:39:07 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-19 19:53:47 +00:00
|
|
|
bool WebContents::OnGoToEntryOffset(int offset) {
|
|
|
|
GoToOffset(offset);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-12-17 17:27:56 +00:00
|
|
|
void WebContents::FindReply(content::WebContents* web_contents,
|
|
|
|
int request_id,
|
|
|
|
int number_of_matches,
|
|
|
|
const gfx::Rect& selection_rect,
|
|
|
|
int active_match_ordinal,
|
|
|
|
bool final_update) {
|
2016-09-09 10:50:43 +00:00
|
|
|
if (!final_update)
|
|
|
|
return;
|
|
|
|
|
2015-12-17 17:27:56 +00:00
|
|
|
v8::Locker locker(isolate());
|
|
|
|
v8::HandleScope handle_scope(isolate());
|
|
|
|
mate::Dictionary result = mate::Dictionary::CreateEmpty(isolate());
|
2016-09-08 05:27:10 +00:00
|
|
|
result.Set("requestId", request_id);
|
|
|
|
result.Set("matches", number_of_matches);
|
|
|
|
result.Set("selectionArea", selection_rect);
|
|
|
|
result.Set("activeMatchOrdinal", active_match_ordinal);
|
2016-09-09 10:50:43 +00:00
|
|
|
result.Set("finalUpdate", final_update); // Deprecate after 2.0
|
2016-09-08 05:27:10 +00:00
|
|
|
Emit("found-in-page", result);
|
2015-12-17 17:27:56 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 14:30:28 +00:00
|
|
|
bool WebContents::CheckMediaAccessPermission(
|
|
|
|
content::RenderFrameHost* render_frame_host,
|
|
|
|
const GURL& security_origin,
|
|
|
|
content::MediaStreamType type) {
|
|
|
|
auto* web_contents =
|
|
|
|
content::WebContents::FromRenderFrameHost(render_frame_host);
|
2018-08-28 14:05:08 +00:00
|
|
|
auto* permission_helper =
|
|
|
|
WebContentsPermissionHelper::FromWebContents(web_contents);
|
|
|
|
return permission_helper->CheckMediaAccessPermission(security_origin, type);
|
2016-02-26 10:17:28 +00:00
|
|
|
}
|
|
|
|
|
2016-01-23 13:29:47 +00:00
|
|
|
void WebContents::RequestMediaAccessPermission(
|
|
|
|
content::WebContents* web_contents,
|
|
|
|
const content::MediaStreamRequest& request,
|
2018-10-02 18:09:25 +00:00
|
|
|
content::MediaResponseCallback callback) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* permission_helper =
|
2016-01-23 13:29:47 +00:00
|
|
|
WebContentsPermissionHelper::FromWebContents(web_contents);
|
2018-10-02 18:09:25 +00:00
|
|
|
permission_helper->RequestMediaAccessPermission(request, std::move(callback));
|
2016-01-23 13:29:47 +00:00
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
void WebContents::RequestToLockMouse(content::WebContents* web_contents,
|
|
|
|
bool user_gesture,
|
|
|
|
bool last_unlocked_by_target) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* permission_helper =
|
2016-02-01 09:43:49 +00:00
|
|
|
WebContentsPermissionHelper::FromWebContents(web_contents);
|
|
|
|
permission_helper->RequestPointerLockPermission(user_gesture);
|
|
|
|
}
|
|
|
|
|
2016-05-30 12:38:09 +00:00
|
|
|
std::unique_ptr<content::BluetoothChooser> WebContents::RunBluetoothChooser(
|
|
|
|
content::RenderFrameHost* frame,
|
|
|
|
const content::BluetoothChooser::EventHandler& event_handler) {
|
2018-06-18 07:32:55 +00:00
|
|
|
return std::make_unique<BluetoothChooser>(this, event_handler);
|
2016-05-30 12:38:09 +00:00
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
content::JavaScriptDialogManager* WebContents::GetJavaScriptDialogManager(
|
2017-04-28 00:28:48 +00:00
|
|
|
content::WebContents* source) {
|
|
|
|
if (!dialog_manager_)
|
|
|
|
dialog_manager_.reset(new AtomJavaScriptDialogManager(this));
|
|
|
|
|
|
|
|
return dialog_manager_.get();
|
|
|
|
}
|
|
|
|
|
2018-09-15 00:18:14 +00:00
|
|
|
void WebContents::OnAudioStateChanged(bool audible) {
|
2018-09-05 21:00:05 +00:00
|
|
|
Emit("-audio-state-changed", audible);
|
|
|
|
}
|
|
|
|
|
2019-01-12 01:00:43 +00:00
|
|
|
void WebContents::BeforeUnloadFired(bool proceed,
|
|
|
|
const base::TimeTicks& proceed_time) {
|
2015-06-25 03:07:23 +00:00
|
|
|
// Do nothing, we override this method just to avoid compilation error since
|
|
|
|
// there are two virtual functions named BeforeUnloadFired.
|
|
|
|
}
|
|
|
|
|
2016-12-28 23:44:44 +00:00
|
|
|
void WebContents::RenderViewCreated(content::RenderViewHost* render_view_host) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const impl = content::RenderWidgetHostImpl::FromID(
|
2016-12-28 23:44:44 +00:00
|
|
|
render_view_host->GetProcess()->GetID(),
|
|
|
|
render_view_host->GetRoutingID());
|
|
|
|
if (impl)
|
|
|
|
impl->disable_hidden_ = !background_throttling_;
|
|
|
|
}
|
|
|
|
|
2018-12-05 08:03:39 +00:00
|
|
|
void WebContents::RenderViewHostChanged(content::RenderViewHost* old_host,
|
|
|
|
content::RenderViewHost* new_host) {
|
|
|
|
currently_committed_process_id_ = new_host->GetProcess()->GetID();
|
|
|
|
}
|
|
|
|
|
2014-04-25 04:40:04 +00:00
|
|
|
void WebContents::RenderViewDeleted(content::RenderViewHost* render_view_host) {
|
2018-12-05 08:03:39 +00:00
|
|
|
// This event is necessary for tracking any states with respect to
|
|
|
|
// intermediate render view hosts aka speculative render view hosts. Currently
|
|
|
|
// used by object-registry.js to ref count remote objects.
|
2018-09-11 18:18:10 +00:00
|
|
|
Emit("render-view-deleted", render_view_host->GetProcess()->GetID());
|
2018-12-05 08:03:39 +00:00
|
|
|
|
|
|
|
if (-1 == currently_committed_process_id_ ||
|
|
|
|
render_view_host->GetProcess()->GetID() ==
|
|
|
|
currently_committed_process_id_) {
|
|
|
|
currently_committed_process_id_ = -1;
|
|
|
|
|
|
|
|
// When the RVH that has been deleted is the current RVH it means that the
|
|
|
|
// the web contents are being closed. This is communicated by this event.
|
|
|
|
// Currently tracked by guest-window-manager.js to destroy the
|
|
|
|
// BrowserWindow.
|
|
|
|
Emit("current-render-view-deleted",
|
|
|
|
render_view_host->GetProcess()->GetID());
|
|
|
|
}
|
2014-04-25 03:51:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::RenderProcessGone(base::TerminationStatus status) {
|
2016-09-18 00:00:45 +00:00
|
|
|
Emit("crashed", status == base::TERMINATION_STATUS_PROCESS_WAS_KILLED);
|
2014-04-25 03:51:05 +00:00
|
|
|
}
|
|
|
|
|
2015-05-12 19:35:56 +00:00
|
|
|
void WebContents::PluginCrashed(const base::FilePath& plugin_path,
|
|
|
|
base::ProcessId plugin_pid) {
|
|
|
|
content::WebPluginInfo info;
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* plugin_service = content::PluginService::GetInstance();
|
2015-05-12 19:35:56 +00:00
|
|
|
plugin_service->GetPluginInfoByPath(plugin_path, &info);
|
|
|
|
Emit("plugin-crashed", info.name, info.version);
|
|
|
|
}
|
|
|
|
|
2017-01-23 09:59:40 +00:00
|
|
|
void WebContents::MediaStartedPlaying(const MediaPlayerInfo& video_type,
|
|
|
|
const MediaPlayerId& id) {
|
2015-12-20 03:16:22 +00:00
|
|
|
Emit("media-started-playing");
|
|
|
|
}
|
|
|
|
|
2018-04-11 15:27:14 +00:00
|
|
|
void WebContents::MediaStoppedPlaying(
|
|
|
|
const MediaPlayerInfo& video_type,
|
|
|
|
const MediaPlayerId& id,
|
|
|
|
content::WebContentsObserver::MediaStoppedReason reason) {
|
2015-12-20 03:16:22 +00:00
|
|
|
Emit("media-paused");
|
|
|
|
}
|
|
|
|
|
2015-12-22 22:16:00 +00:00
|
|
|
void WebContents::DidChangeThemeColor(SkColor theme_color) {
|
2017-08-20 14:48:13 +00:00
|
|
|
if (theme_color != SK_ColorTRANSPARENT) {
|
|
|
|
Emit("did-change-theme-color", atom::ToRGBHex(theme_color));
|
|
|
|
} else {
|
2017-11-13 07:20:54 +00:00
|
|
|
Emit("did-change-theme-color", nullptr);
|
2017-08-20 14:48:13 +00:00
|
|
|
}
|
2015-12-22 22:16:00 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 13:49:31 +00:00
|
|
|
void WebContents::DocumentLoadedInFrame(
|
|
|
|
content::RenderFrameHost* render_frame_host) {
|
2016-08-04 04:03:24 +00:00
|
|
|
if (!render_frame_host->GetParent())
|
2015-04-29 13:49:31 +00:00
|
|
|
Emit("dom-ready");
|
|
|
|
}
|
|
|
|
|
2014-10-11 11:11:34 +00:00
|
|
|
void WebContents::DidFinishLoad(content::RenderFrameHost* render_frame_host,
|
|
|
|
const GURL& validated_url) {
|
|
|
|
bool is_main_frame = !render_frame_host->GetParent();
|
2018-04-26 10:17:55 +00:00
|
|
|
int frame_process_id = render_frame_host->GetProcess()->GetID();
|
|
|
|
int frame_routing_id = render_frame_host->GetRoutingID();
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-frame-finish-load", is_main_frame, frame_process_id,
|
2018-04-26 10:17:55 +00:00
|
|
|
frame_routing_id);
|
2014-06-09 04:03:19 +00:00
|
|
|
|
|
|
|
if (is_main_frame)
|
|
|
|
Emit("did-finish-load");
|
2014-04-25 04:22:16 +00:00
|
|
|
}
|
|
|
|
|
2014-10-24 15:05:25 +00:00
|
|
|
void WebContents::DidFailLoad(content::RenderFrameHost* render_frame_host,
|
2016-04-13 10:33:59 +00:00
|
|
|
const GURL& url,
|
2014-10-24 15:05:25 +00:00
|
|
|
int error_code,
|
2017-11-23 12:53:47 +00:00
|
|
|
const base::string16& error_description) {
|
2016-04-05 02:24:58 +00:00
|
|
|
bool is_main_frame = !render_frame_host->GetParent();
|
2018-04-26 10:17:55 +00:00
|
|
|
int frame_process_id = render_frame_host->GetProcess()->GetID();
|
|
|
|
int frame_routing_id = render_frame_host->GetRoutingID();
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-fail-load", error_code, error_description, url, is_main_frame,
|
|
|
|
frame_process_id, frame_routing_id);
|
2014-10-24 15:05:25 +00:00
|
|
|
}
|
|
|
|
|
2015-05-22 07:24:34 +00:00
|
|
|
void WebContents::DidStartLoading() {
|
2014-04-25 04:22:16 +00:00
|
|
|
Emit("did-start-loading");
|
|
|
|
}
|
|
|
|
|
2015-05-22 07:24:34 +00:00
|
|
|
void WebContents::DidStopLoading() {
|
2014-04-25 04:22:16 +00:00
|
|
|
Emit("did-stop-loading");
|
|
|
|
}
|
|
|
|
|
2018-09-15 15:42:43 +00:00
|
|
|
bool WebContents::EmitNavigationEvent(
|
|
|
|
const std::string& event,
|
2018-04-26 10:17:55 +00:00
|
|
|
content::NavigationHandle* navigation_handle) {
|
|
|
|
bool is_main_frame = navigation_handle->IsInMainFrame();
|
|
|
|
int frame_tree_node_id = navigation_handle->GetFrameTreeNodeId();
|
|
|
|
content::FrameTreeNode* frame_tree_node =
|
2018-05-10 20:52:42 +00:00
|
|
|
content::FrameTreeNode::GloballyFindByID(frame_tree_node_id);
|
2018-04-26 10:17:55 +00:00
|
|
|
content::RenderFrameHostManager* render_manager =
|
2018-05-10 20:52:42 +00:00
|
|
|
frame_tree_node->render_manager();
|
2018-04-26 10:17:55 +00:00
|
|
|
content::RenderFrameHost* frame_host = nullptr;
|
|
|
|
if (render_manager) {
|
|
|
|
frame_host = render_manager->speculative_frame_host();
|
|
|
|
if (!frame_host)
|
|
|
|
frame_host = render_manager->current_frame_host();
|
|
|
|
}
|
|
|
|
int frame_process_id = -1, frame_routing_id = -1;
|
|
|
|
if (frame_host) {
|
|
|
|
frame_process_id = frame_host->GetProcess()->GetID();
|
|
|
|
frame_routing_id = frame_host->GetRoutingID();
|
|
|
|
}
|
|
|
|
bool is_same_document = navigation_handle->IsSameDocument();
|
|
|
|
auto url = navigation_handle->GetURL();
|
2018-09-15 15:42:43 +00:00
|
|
|
return Emit(event, url, is_same_document, is_main_frame, frame_process_id,
|
|
|
|
frame_routing_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::DidStartNavigation(
|
|
|
|
content::NavigationHandle* navigation_handle) {
|
|
|
|
EmitNavigationEvent("did-start-navigation", navigation_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::DidRedirectNavigation(
|
|
|
|
content::NavigationHandle* navigation_handle) {
|
|
|
|
EmitNavigationEvent("did-redirect-navigation", navigation_handle);
|
2018-04-26 10:17:55 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 00:19:28 +00:00
|
|
|
void WebContents::DidFinishNavigation(
|
|
|
|
content::NavigationHandle* navigation_handle) {
|
2018-04-26 10:17:55 +00:00
|
|
|
if (!navigation_handle->HasCommitted())
|
|
|
|
return;
|
2016-07-14 00:19:28 +00:00
|
|
|
bool is_main_frame = navigation_handle->IsInMainFrame();
|
2018-04-26 10:17:55 +00:00
|
|
|
content::RenderFrameHost* frame_host =
|
2018-05-10 20:52:42 +00:00
|
|
|
navigation_handle->GetRenderFrameHost();
|
2018-04-26 10:17:55 +00:00
|
|
|
int frame_process_id = -1, frame_routing_id = -1;
|
|
|
|
if (frame_host) {
|
|
|
|
frame_process_id = frame_host->GetProcess()->GetID();
|
|
|
|
frame_routing_id = frame_host->GetRoutingID();
|
|
|
|
}
|
|
|
|
if (!navigation_handle->IsErrorPage()) {
|
2016-07-14 00:19:28 +00:00
|
|
|
auto url = navigation_handle->GetURL();
|
2017-08-08 00:49:20 +00:00
|
|
|
bool is_same_document = navigation_handle->IsSameDocument();
|
2018-05-01 04:34:41 +00:00
|
|
|
if (is_same_document) {
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-navigate-in-page", url, is_main_frame, frame_process_id,
|
|
|
|
frame_routing_id);
|
2018-05-01 04:34:41 +00:00
|
|
|
} else {
|
2018-05-10 20:52:42 +00:00
|
|
|
const net::HttpResponseHeaders* http_response =
|
|
|
|
navigation_handle->GetResponseHeaders();
|
2018-05-01 04:34:41 +00:00
|
|
|
std::string http_status_text;
|
|
|
|
int http_response_code = -1;
|
|
|
|
if (http_response) {
|
|
|
|
http_status_text = http_response->GetStatusText();
|
|
|
|
http_response_code = http_response->response_code();
|
|
|
|
}
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-frame-navigate", url, http_response_code, http_status_text,
|
|
|
|
is_main_frame, frame_process_id, frame_routing_id);
|
2018-05-01 04:34:41 +00:00
|
|
|
if (is_main_frame) {
|
|
|
|
Emit("did-navigate", url, http_response_code, http_status_text);
|
|
|
|
}
|
2016-07-14 00:19:28 +00:00
|
|
|
}
|
2018-08-16 22:57:40 +00:00
|
|
|
if (IsGuest())
|
|
|
|
Emit("load-commit", url, is_main_frame);
|
2016-07-14 00:19:28 +00:00
|
|
|
} else {
|
|
|
|
auto url = navigation_handle->GetURL();
|
|
|
|
int code = navigation_handle->GetNetErrorCode();
|
|
|
|
auto description = net::ErrorToShortString(code);
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-fail-provisional-load", code, description, url, is_main_frame,
|
|
|
|
frame_process_id, frame_routing_id);
|
2016-07-14 00:19:28 +00:00
|
|
|
|
|
|
|
// Do not emit "did-fail-load" for canceled requests.
|
|
|
|
if (code != net::ERR_ABORTED)
|
2018-05-10 20:52:42 +00:00
|
|
|
Emit("did-fail-load", code, description, url, is_main_frame,
|
|
|
|
frame_process_id, frame_routing_id);
|
2016-07-14 00:19:28 +00:00
|
|
|
}
|
2014-12-09 22:38:43 +00:00
|
|
|
}
|
|
|
|
|
2017-12-18 00:19:21 +00:00
|
|
|
void WebContents::TitleWasSet(content::NavigationEntry* entry) {
|
2017-12-18 07:54:44 +00:00
|
|
|
base::string16 final_title;
|
2017-12-24 12:42:23 +00:00
|
|
|
bool explicit_set = true;
|
|
|
|
if (entry) {
|
|
|
|
auto title = entry->GetTitle();
|
|
|
|
auto url = entry->GetURL();
|
|
|
|
if (url.SchemeIsFile() && title.empty()) {
|
|
|
|
final_title = base::UTF8ToUTF16(url.ExtractFileName());
|
|
|
|
explicit_set = false;
|
|
|
|
} else {
|
|
|
|
final_title = title;
|
|
|
|
}
|
2017-12-18 00:19:21 +00:00
|
|
|
}
|
2017-12-18 07:54:44 +00:00
|
|
|
Emit("page-title-updated", final_title, explicit_set);
|
2015-04-05 09:20:42 +00:00
|
|
|
}
|
|
|
|
|
2015-04-17 19:55:31 +00:00
|
|
|
void WebContents::DidUpdateFaviconURL(
|
|
|
|
const std::vector<content::FaviconURL>& urls) {
|
|
|
|
std::set<GURL> unique_urls;
|
2016-07-10 11:32:40 +00:00
|
|
|
for (const auto& iter : urls) {
|
2017-08-26 19:48:25 +00:00
|
|
|
if (iter.icon_type != content::FaviconURL::IconType::kFavicon)
|
2015-04-17 19:55:31 +00:00
|
|
|
continue;
|
2016-07-10 11:32:40 +00:00
|
|
|
const GURL& url = iter.icon_url;
|
2015-04-17 19:55:31 +00:00
|
|
|
if (url.is_valid())
|
|
|
|
unique_urls.insert(url);
|
|
|
|
}
|
2015-04-20 06:50:04 +00:00
|
|
|
Emit("page-favicon-updated", unique_urls);
|
2015-04-17 19:55:31 +00:00
|
|
|
}
|
|
|
|
|
2016-04-12 07:36:12 +00:00
|
|
|
void WebContents::DevToolsReloadPage() {
|
|
|
|
Emit("devtools-reload-page");
|
|
|
|
}
|
|
|
|
|
2015-10-01 03:14:19 +00:00
|
|
|
void WebContents::DevToolsFocused() {
|
|
|
|
Emit("devtools-focused");
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::DevToolsOpened() {
|
|
|
|
v8::Locker locker(isolate());
|
|
|
|
v8::HandleScope handle_scope(isolate());
|
2018-10-19 08:52:07 +00:00
|
|
|
auto handle =
|
|
|
|
FromOrCreate(isolate(), managed_web_contents()->GetDevToolsWebContents());
|
2015-10-01 03:14:19 +00:00
|
|
|
devtools_web_contents_.Reset(isolate(), handle.ToV8());
|
|
|
|
|
2016-05-26 10:29:39 +00:00
|
|
|
// Set inspected tabID.
|
2017-04-05 08:34:53 +00:00
|
|
|
base::Value tab_id(ID());
|
2018-04-18 01:55:30 +00:00
|
|
|
managed_web_contents()->CallClientFunction("DevToolsAPI.setInspectedTabId",
|
|
|
|
&tab_id, nullptr, nullptr);
|
2016-05-26 10:29:39 +00:00
|
|
|
|
2017-12-01 02:11:37 +00:00
|
|
|
// Inherit owner window in devtools when it doesn't have one.
|
2017-11-30 07:37:26 +00:00
|
|
|
auto* devtools = managed_web_contents()->GetDevToolsWebContents();
|
2018-10-02 22:14:43 +00:00
|
|
|
bool has_window =
|
|
|
|
devtools->GetUserData(NativeWindowRelay::kNativeWindowRelayUserDataKey);
|
2017-12-01 02:11:37 +00:00
|
|
|
if (owner_window() && !has_window)
|
2017-11-30 07:37:26 +00:00
|
|
|
handle->SetOwnerWindow(devtools, owner_window());
|
2015-10-01 06:41:01 +00:00
|
|
|
|
2015-10-01 03:14:19 +00:00
|
|
|
Emit("devtools-opened");
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::DevToolsClosed() {
|
|
|
|
v8::Locker locker(isolate());
|
|
|
|
v8::HandleScope handle_scope(isolate());
|
|
|
|
devtools_web_contents_.Reset();
|
|
|
|
|
|
|
|
Emit("devtools-closed");
|
|
|
|
}
|
|
|
|
|
2018-05-10 20:52:17 +00:00
|
|
|
#if defined(TOOLKIT_VIEWS) && !defined(OS_MACOSX)
|
2017-11-13 07:13:54 +00:00
|
|
|
void WebContents::ShowAutofillPopup(content::RenderFrameHost* frame_host,
|
|
|
|
const gfx::RectF& bounds,
|
|
|
|
const std::vector<base::string16>& values,
|
|
|
|
const std::vector<base::string16>& labels) {
|
2018-04-05 00:53:51 +00:00
|
|
|
bool offscreen = IsOffScreen() || (embedder_ && embedder_->IsOffScreen());
|
2018-06-18 15:15:37 +00:00
|
|
|
gfx::RectF popup_bounds(bounds);
|
|
|
|
content::RenderFrameHost* embedder_frame_host = nullptr;
|
|
|
|
if (embedder_) {
|
|
|
|
auto* embedder_view = embedder_->web_contents()->GetMainFrame()->GetView();
|
|
|
|
auto* view = web_contents()->GetMainFrame()->GetView();
|
|
|
|
auto offset = view->GetViewBounds().origin() -
|
|
|
|
embedder_view->GetViewBounds().origin();
|
|
|
|
popup_bounds.Offset(offset.x(), offset.y());
|
|
|
|
embedder_frame_host = embedder_->web_contents()->GetMainFrame();
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonWebContentsDelegate::ShowAutofillPopup(
|
|
|
|
frame_host, embedder_frame_host, offscreen, popup_bounds, values, labels);
|
2017-11-13 07:13:54 +00:00
|
|
|
}
|
2018-04-05 00:53:51 +00:00
|
|
|
#endif
|
2017-11-13 07:13:54 +00:00
|
|
|
|
2014-04-25 08:13:16 +00:00
|
|
|
bool WebContents::OnMessageReceived(const IPC::Message& message) {
|
|
|
|
bool handled = true;
|
|
|
|
IPC_BEGIN_MESSAGE_MAP(WebContents, message)
|
2019-01-12 01:00:43 +00:00
|
|
|
IPC_MESSAGE_HANDLER_CODE(WidgetHostMsg_SetCursor, OnCursorChange,
|
2017-12-19 00:57:10 +00:00
|
|
|
handled = false)
|
2017-05-26 01:38:27 +00:00
|
|
|
IPC_MESSAGE_UNHANDLED(handled = false)
|
|
|
|
IPC_END_MESSAGE_MAP()
|
|
|
|
|
|
|
|
return handled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WebContents::OnMessageReceived(const IPC::Message& message,
|
2018-03-09 09:31:09 +00:00
|
|
|
content::RenderFrameHost* frame_host) {
|
2017-05-26 01:38:27 +00:00
|
|
|
bool handled = true;
|
2018-03-09 09:31:09 +00:00
|
|
|
FrameDispatchHelper helper = {this, frame_host};
|
2017-11-03 02:01:26 +00:00
|
|
|
IPC_BEGIN_MESSAGE_MAP_WITH_PARAM(WebContents, message, frame_host)
|
2018-03-09 09:31:09 +00:00
|
|
|
IPC_MESSAGE_HANDLER(AtomFrameHostMsg_Message, OnRendererMessage)
|
|
|
|
IPC_MESSAGE_FORWARD_DELAY_REPLY(AtomFrameHostMsg_Message_Sync, &helper,
|
|
|
|
FrameDispatchHelper::OnRendererMessageSync)
|
2018-08-24 21:14:39 +00:00
|
|
|
IPC_MESSAGE_HANDLER(AtomFrameHostMsg_Message_To, OnRendererMessageTo)
|
2018-03-09 09:31:09 +00:00
|
|
|
IPC_MESSAGE_FORWARD_DELAY_REPLY(
|
|
|
|
AtomFrameHostMsg_SetTemporaryZoomLevel, &helper,
|
|
|
|
FrameDispatchHelper::OnSetTemporaryZoomLevel)
|
|
|
|
IPC_MESSAGE_FORWARD_DELAY_REPLY(AtomFrameHostMsg_GetZoomLevel, &helper,
|
|
|
|
FrameDispatchHelper::OnGetZoomLevel)
|
2018-05-10 20:52:17 +00:00
|
|
|
#if defined(TOOLKIT_VIEWS) && !defined(OS_MACOSX)
|
2017-11-03 02:01:26 +00:00
|
|
|
IPC_MESSAGE_HANDLER(AtomAutofillFrameHostMsg_ShowPopup, ShowAutofillPopup)
|
2018-04-05 00:53:51 +00:00
|
|
|
IPC_MESSAGE_HANDLER(AtomAutofillFrameHostMsg_HidePopup, HideAutofillPopup)
|
|
|
|
#endif
|
2014-04-25 08:13:16 +00:00
|
|
|
IPC_MESSAGE_UNHANDLED(handled = false)
|
|
|
|
IPC_END_MESSAGE_MAP()
|
|
|
|
|
|
|
|
return handled;
|
|
|
|
}
|
|
|
|
|
2016-01-13 09:59:59 +00:00
|
|
|
// There are three ways of destroying a webContents:
|
2016-07-28 22:48:01 +00:00
|
|
|
// 1. call webContents.destroy();
|
2016-01-13 09:59:59 +00:00
|
|
|
// 2. garbage collection;
|
|
|
|
// 3. user closes the window of webContents;
|
2018-08-16 22:57:40 +00:00
|
|
|
// 4. the embedder detaches the frame.
|
2018-09-07 06:41:48 +00:00
|
|
|
// For webview both #1 and #4 may happen, for BrowserWindow both #1 and #3 may
|
2016-01-13 09:59:59 +00:00
|
|
|
// happen. The #2 should never happen for webContents, because webview is
|
|
|
|
// managed by GuestViewManager, and BrowserWindow's webContents is managed
|
2018-02-22 03:49:17 +00:00
|
|
|
// by api::BrowserWindow.
|
2016-01-13 09:59:59 +00:00
|
|
|
// For #1, the destructor will do the cleanup work and we only need to make
|
|
|
|
// sure "destroyed" event is emitted. For #3, the content::WebContents will
|
|
|
|
// be destroyed on close, and WebContentsDestroyed would be called for it, so
|
|
|
|
// we need to make sure the api::WebContents is also deleted.
|
2018-08-16 22:57:40 +00:00
|
|
|
// For #4, the WebContents will be destroyed by embedder.
|
2014-07-28 07:29:51 +00:00
|
|
|
void WebContents::WebContentsDestroyed() {
|
2016-01-13 06:47:39 +00:00
|
|
|
// Cleanup relationships with other parts.
|
2015-06-24 09:58:12 +00:00
|
|
|
RemoveFromWeakMap();
|
2016-01-12 13:10:24 +00:00
|
|
|
|
|
|
|
// We can not call Destroy here because we need to call Emit first, but we
|
|
|
|
// also do not want any method to be used, so just mark as destroyed here.
|
|
|
|
MarkDestroyed();
|
|
|
|
|
|
|
|
Emit("destroyed");
|
|
|
|
|
2018-08-16 22:57:40 +00:00
|
|
|
// For guest view based on OOPIF, the WebContents is released by the embedder
|
|
|
|
// frame, and we need to clear the reference to the memory.
|
|
|
|
if (IsGuest() && managed_web_contents()) {
|
|
|
|
managed_web_contents()->ReleaseWebContents();
|
|
|
|
ResetManagedWebContents(false);
|
|
|
|
}
|
|
|
|
|
2016-01-12 13:10:24 +00:00
|
|
|
// Destroy the native class in next tick.
|
2018-04-18 01:55:30 +00:00
|
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, GetDestroyClosure());
|
2014-04-25 03:22:51 +00:00
|
|
|
}
|
|
|
|
|
2015-03-05 14:39:27 +00:00
|
|
|
void WebContents::NavigationEntryCommitted(
|
2015-05-11 05:51:52 +00:00
|
|
|
const content::LoadCommittedDetails& details) {
|
|
|
|
Emit("navigation-entry-commited", details.entry->GetURL(),
|
2017-08-08 00:49:20 +00:00
|
|
|
details.is_same_document, details.did_replace_entry);
|
2015-03-05 14:39:27 +00:00
|
|
|
}
|
|
|
|
|
2018-10-19 00:32:22 +00:00
|
|
|
void WebContents::SetBackgroundThrottling(bool allowed) {
|
|
|
|
background_throttling_ = allowed;
|
|
|
|
|
2019-01-09 20:12:44 +00:00
|
|
|
auto* contents = web_contents();
|
2018-10-19 00:32:22 +00:00
|
|
|
if (!contents) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto* render_view_host = contents->GetRenderViewHost();
|
|
|
|
if (!render_view_host) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto* render_process_host = render_view_host->GetProcess();
|
|
|
|
if (!render_process_host) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto* render_widget_host_impl = content::RenderWidgetHostImpl::FromID(
|
|
|
|
render_process_host->GetID(), render_view_host->GetRoutingID());
|
|
|
|
if (!render_widget_host_impl) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
render_widget_host_impl->disable_hidden_ = !background_throttling_;
|
|
|
|
|
|
|
|
if (render_widget_host_impl->is_hidden()) {
|
|
|
|
render_widget_host_impl->WasShown(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-06 01:28:40 +00:00
|
|
|
int WebContents::GetProcessID() const {
|
2017-12-18 02:46:23 +00:00
|
|
|
return web_contents()->GetMainFrame()->GetProcess()->GetID();
|
2015-06-23 08:16:10 +00:00
|
|
|
}
|
|
|
|
|
2017-05-15 20:29:34 +00:00
|
|
|
base::ProcessId WebContents::GetOSProcessID() const {
|
2018-06-25 20:30:00 +00:00
|
|
|
base::ProcessHandle process_handle =
|
2018-09-15 00:13:02 +00:00
|
|
|
web_contents()->GetMainFrame()->GetProcess()->GetProcess().Handle();
|
2017-04-18 10:31:20 +00:00
|
|
|
return base::GetProcId(process_handle);
|
|
|
|
}
|
|
|
|
|
2016-06-14 16:09:54 +00:00
|
|
|
WebContents::Type WebContents::GetType() const {
|
|
|
|
return type_;
|
2016-06-08 18:41:14 +00:00
|
|
|
}
|
|
|
|
|
2015-06-23 08:16:10 +00:00
|
|
|
bool WebContents::Equal(const WebContents* web_contents) const {
|
2018-07-23 09:08:36 +00:00
|
|
|
return ID() == web_contents->ID();
|
2015-06-23 08:16:10 +00:00
|
|
|
}
|
|
|
|
|
2014-11-06 19:29:41 +00:00
|
|
|
void WebContents::LoadURL(const GURL& url, const mate::Dictionary& options) {
|
2017-01-21 09:29:20 +00:00
|
|
|
if (!url.is_valid() || url.spec().size() > url::kMaxURLChars) {
|
2018-04-18 01:55:30 +00:00
|
|
|
Emit("did-fail-load", static_cast<int>(net::ERR_INVALID_URL),
|
2016-02-11 02:54:50 +00:00
|
|
|
net::ErrorToShortString(net::ERR_INVALID_URL),
|
2018-04-18 01:55:30 +00:00
|
|
|
url.possibly_invalid_spec(), true);
|
2016-02-11 02:54:50 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-04-25 04:52:30 +00:00
|
|
|
content::NavigationController::LoadURLParams params(url);
|
2014-11-06 19:29:41 +00:00
|
|
|
|
2018-04-05 23:13:24 +00:00
|
|
|
if (!options.Get("httpReferrer", ¶ms.referrer)) {
|
|
|
|
GURL http_referrer;
|
|
|
|
if (options.Get("httpReferrer", &http_referrer))
|
|
|
|
params.referrer = content::Referrer(http_referrer.GetAsReferrer(),
|
|
|
|
blink::kWebReferrerPolicyDefault);
|
|
|
|
}
|
2014-11-06 19:29:41 +00:00
|
|
|
|
2015-05-06 04:01:41 +00:00
|
|
|
std::string user_agent;
|
2015-06-09 01:27:57 +00:00
|
|
|
if (options.Get("userAgent", &user_agent))
|
2018-07-21 16:51:28 +00:00
|
|
|
web_contents()->SetUserAgentOverride(user_agent, false);
|
2015-05-06 04:01:41 +00:00
|
|
|
|
2015-10-26 21:18:16 +00:00
|
|
|
std::string extra_headers;
|
|
|
|
if (options.Get("extraHeaders", &extra_headers))
|
|
|
|
params.extra_headers = extra_headers;
|
|
|
|
|
2018-04-11 08:42:14 +00:00
|
|
|
scoped_refptr<network::ResourceRequestBody> body;
|
2016-10-09 23:30:38 +00:00
|
|
|
if (options.Get("postData", &body)) {
|
|
|
|
params.post_data = body;
|
|
|
|
params.load_type = content::NavigationController::LOAD_TYPE_HTTP_POST;
|
|
|
|
}
|
|
|
|
|
2017-03-01 00:49:03 +00:00
|
|
|
GURL base_url_for_data_url;
|
|
|
|
if (options.Get("baseURLForDataURL", &base_url_for_data_url)) {
|
|
|
|
params.base_url_for_data_url = base_url_for_data_url;
|
|
|
|
params.load_type = content::NavigationController::LOAD_TYPE_DATA;
|
|
|
|
}
|
2017-03-02 18:50:48 +00:00
|
|
|
|
2014-12-07 15:43:26 +00:00
|
|
|
params.transition_type = ui::PAGE_TRANSITION_TYPED;
|
2015-04-26 13:31:48 +00:00
|
|
|
params.should_clear_history_list = true;
|
2014-04-25 04:52:30 +00:00
|
|
|
params.override_user_agent = content::NavigationController::UA_OVERRIDE_TRUE;
|
2017-02-19 20:43:17 +00:00
|
|
|
web_contents()->GetController().LoadURLWithParams(params);
|
2017-04-07 17:34:14 +00:00
|
|
|
|
|
|
|
// Set the background color of RenderWidgetHostView.
|
2016-04-03 01:37:48 +00:00
|
|
|
// We have to call it right after LoadURL because the RenderViewHost is only
|
|
|
|
// created after loading a page.
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const view = web_contents()->GetRenderWidgetHostView();
|
2017-04-07 17:34:14 +00:00
|
|
|
if (view) {
|
2018-03-08 07:12:45 +00:00
|
|
|
auto* web_preferences = WebContentsPreferences::From(web_contents());
|
2017-04-07 17:34:14 +00:00
|
|
|
std::string color_name;
|
2018-08-03 21:23:07 +00:00
|
|
|
if (web_preferences->GetPreference(options::kBackgroundColor,
|
|
|
|
&color_name)) {
|
2017-04-07 17:34:14 +00:00
|
|
|
view->SetBackgroundColor(ParseHexColor(color_name));
|
|
|
|
} else {
|
|
|
|
view->SetBackgroundColor(SK_ColorTRANSPARENT);
|
|
|
|
}
|
|
|
|
}
|
2014-04-25 04:52:30 +00:00
|
|
|
}
|
|
|
|
|
2015-12-02 21:49:30 +00:00
|
|
|
void WebContents::DownloadURL(const GURL& url) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* browser_context = web_contents()->GetBrowserContext();
|
|
|
|
auto* download_manager =
|
2018-04-18 01:55:30 +00:00
|
|
|
content::BrowserContext::GetDownloadManager(browser_context);
|
2018-04-12 07:30:51 +00:00
|
|
|
std::unique_ptr<download::DownloadUrlParameters> download_params(
|
|
|
|
content::DownloadRequestUtils::CreateDownloadForWebContentsMainFrame(
|
2017-08-21 09:20:37 +00:00
|
|
|
web_contents(), url, NO_TRAFFIC_ANNOTATION_YET));
|
2018-04-12 07:30:51 +00:00
|
|
|
download_manager->DownloadUrl(std::move(download_params));
|
2015-12-02 21:49:30 +00:00
|
|
|
}
|
|
|
|
|
2015-09-15 01:17:45 +00:00
|
|
|
GURL WebContents::GetURL() const {
|
|
|
|
return web_contents()->GetURL();
|
|
|
|
}
|
|
|
|
|
2014-06-28 11:36:57 +00:00
|
|
|
base::string16 WebContents::GetTitle() const {
|
2014-04-25 05:10:16 +00:00
|
|
|
return web_contents()->GetTitle();
|
2014-04-24 09:00:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool WebContents::IsLoading() const {
|
2014-04-25 05:10:16 +00:00
|
|
|
return web_contents()->IsLoading();
|
2014-04-24 09:00:41 +00:00
|
|
|
}
|
|
|
|
|
2016-04-18 17:37:08 +00:00
|
|
|
bool WebContents::IsLoadingMainFrame() const {
|
2018-04-06 07:22:52 +00:00
|
|
|
return web_contents()->IsLoadingToDifferentDocument();
|
2016-04-18 17:37:08 +00:00
|
|
|
}
|
2014-04-24 09:00:41 +00:00
|
|
|
|
|
|
|
bool WebContents::IsWaitingForResponse() const {
|
2014-04-25 05:10:16 +00:00
|
|
|
return web_contents()->IsWaitingForResponse();
|
2014-04-24 09:00:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Stop() {
|
2014-04-25 05:10:16 +00:00
|
|
|
web_contents()->Stop();
|
2014-04-24 09:00:41 +00:00
|
|
|
}
|
|
|
|
|
2015-05-11 06:30:26 +00:00
|
|
|
void WebContents::GoBack() {
|
2015-05-11 06:40:40 +00:00
|
|
|
atom::AtomBrowserClient::SuppressRendererProcessRestartForOnce();
|
2015-05-11 06:30:26 +00:00
|
|
|
web_contents()->GetController().GoBack();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::GoForward() {
|
2015-05-11 06:40:40 +00:00
|
|
|
atom::AtomBrowserClient::SuppressRendererProcessRestartForOnce();
|
2015-05-11 06:30:26 +00:00
|
|
|
web_contents()->GetController().GoForward();
|
|
|
|
}
|
|
|
|
|
2015-05-11 08:44:01 +00:00
|
|
|
void WebContents::GoToOffset(int offset) {
|
|
|
|
atom::AtomBrowserClient::SuppressRendererProcessRestartForOnce();
|
|
|
|
web_contents()->GetController().GoToOffset(offset);
|
|
|
|
}
|
|
|
|
|
2017-03-09 03:06:26 +00:00
|
|
|
const std::string WebContents::GetWebRTCIPHandlingPolicy() const {
|
2018-04-18 01:55:30 +00:00
|
|
|
return web_contents()->GetMutableRendererPrefs()->webrtc_ip_handling_policy;
|
2017-03-08 14:55:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::SetWebRTCIPHandlingPolicy(
|
2017-03-09 03:06:26 +00:00
|
|
|
const std::string& webrtc_ip_handling_policy) {
|
2017-03-08 14:55:59 +00:00
|
|
|
if (GetWebRTCIPHandlingPolicy() == webrtc_ip_handling_policy)
|
|
|
|
return;
|
|
|
|
web_contents()->GetMutableRendererPrefs()->webrtc_ip_handling_policy =
|
2018-04-18 01:55:30 +00:00
|
|
|
webrtc_ip_handling_policy;
|
2017-03-08 14:55:59 +00:00
|
|
|
|
|
|
|
content::RenderViewHost* host = web_contents()->GetRenderViewHost();
|
|
|
|
if (host)
|
|
|
|
host->SyncRendererPrefs();
|
|
|
|
}
|
|
|
|
|
2014-04-24 09:00:41 +00:00
|
|
|
bool WebContents::IsCrashed() const {
|
2014-04-25 05:10:16 +00:00
|
|
|
return web_contents()->IsCrashed();
|
2014-04-24 09:00:41 +00:00
|
|
|
}
|
|
|
|
|
2016-06-22 06:57:51 +00:00
|
|
|
void WebContents::SetUserAgent(const std::string& user_agent,
|
|
|
|
mate::Arguments* args) {
|
2018-07-21 16:51:28 +00:00
|
|
|
web_contents()->SetUserAgentOverride(user_agent, false);
|
2014-10-24 12:57:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-22 04:25:10 +00:00
|
|
|
std::string WebContents::GetUserAgent() {
|
|
|
|
return web_contents()->GetUserAgentOverride();
|
|
|
|
}
|
|
|
|
|
2015-10-14 04:41:31 +00:00
|
|
|
bool WebContents::SavePage(const base::FilePath& full_file_path,
|
|
|
|
const content::SavePageType& save_type,
|
|
|
|
const SavePageHandler::SavePageCallback& callback) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* handler = new SavePageHandler(web_contents(), callback);
|
2015-10-14 04:41:31 +00:00
|
|
|
return handler->Handle(full_file_path, save_type);
|
|
|
|
}
|
|
|
|
|
2015-06-05 09:01:17 +00:00
|
|
|
void WebContents::OpenDevTools(mate::Arguments* args) {
|
2015-06-24 14:23:38 +00:00
|
|
|
if (type_ == REMOTE)
|
2015-06-05 07:41:03 +00:00
|
|
|
return;
|
2015-06-24 14:23:38 +00:00
|
|
|
|
2016-09-07 02:16:52 +00:00
|
|
|
if (!enable_devtools_)
|
2016-09-05 08:27:56 +00:00
|
|
|
return;
|
|
|
|
|
2016-04-19 12:42:05 +00:00
|
|
|
std::string state;
|
2016-05-17 13:39:18 +00:00
|
|
|
if (type_ == WEB_VIEW || !owner_window()) {
|
2016-04-19 12:42:05 +00:00
|
|
|
state = "detach";
|
2017-12-01 02:11:37 +00:00
|
|
|
}
|
2018-11-27 09:34:44 +00:00
|
|
|
bool activate = true;
|
2017-12-01 02:11:37 +00:00
|
|
|
if (args && args->Length() == 1) {
|
2015-06-05 09:01:17 +00:00
|
|
|
mate::Dictionary options;
|
2016-04-19 12:42:05 +00:00
|
|
|
if (args->GetNext(&options)) {
|
|
|
|
options.Get("mode", &state);
|
2018-11-27 09:34:44 +00:00
|
|
|
options.Get("activate", &activate);
|
2016-04-19 12:42:05 +00:00
|
|
|
}
|
2015-06-05 09:01:17 +00:00
|
|
|
}
|
2016-04-19 12:42:05 +00:00
|
|
|
managed_web_contents()->SetDockState(state);
|
2018-11-27 09:34:44 +00:00
|
|
|
managed_web_contents()->ShowDevTools(activate);
|
2014-11-02 14:34:22 +00:00
|
|
|
}
|
|
|
|
|
2014-11-03 14:12:56 +00:00
|
|
|
void WebContents::CloseDevTools() {
|
2015-06-24 14:23:38 +00:00
|
|
|
if (type_ == REMOTE)
|
2015-06-05 07:41:03 +00:00
|
|
|
return;
|
2015-06-24 14:23:38 +00:00
|
|
|
|
2015-06-25 06:32:38 +00:00
|
|
|
managed_web_contents()->CloseDevTools();
|
2014-11-03 14:12:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool WebContents::IsDevToolsOpened() {
|
2015-06-24 14:23:38 +00:00
|
|
|
if (type_ == REMOTE)
|
2015-06-05 07:41:03 +00:00
|
|
|
return false;
|
2015-06-25 06:32:38 +00:00
|
|
|
|
|
|
|
return managed_web_contents()->IsDevToolsViewShowing();
|
2014-11-03 14:12:56 +00:00
|
|
|
}
|
|
|
|
|
2016-01-01 05:11:21 +00:00
|
|
|
bool WebContents::IsDevToolsFocused() {
|
|
|
|
if (type_ == REMOTE)
|
|
|
|
return false;
|
2016-01-01 09:45:03 +00:00
|
|
|
|
2016-01-01 05:11:21 +00:00
|
|
|
return managed_web_contents()->GetView()->IsDevToolsViewFocused();
|
|
|
|
}
|
|
|
|
|
2015-09-18 03:06:38 +00:00
|
|
|
void WebContents::EnableDeviceEmulation(
|
|
|
|
const blink::WebDeviceEmulationParams& params) {
|
2015-08-31 09:19:19 +00:00
|
|
|
if (type_ == REMOTE)
|
|
|
|
return;
|
|
|
|
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* frame_host = web_contents()->GetMainFrame();
|
2017-12-30 16:25:09 +00:00
|
|
|
if (frame_host) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* widget_host =
|
2017-12-30 16:25:09 +00:00
|
|
|
frame_host ? frame_host->GetView()->GetRenderWidgetHost() : nullptr;
|
|
|
|
if (!widget_host)
|
|
|
|
return;
|
2019-01-12 01:00:43 +00:00
|
|
|
widget_host->Send(new WidgetMsg_EnableDeviceEmulation(
|
|
|
|
widget_host->GetRoutingID(), params));
|
2017-12-30 16:25:09 +00:00
|
|
|
}
|
2015-08-31 09:19:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::DisableDeviceEmulation() {
|
|
|
|
if (type_ == REMOTE)
|
|
|
|
return;
|
|
|
|
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* frame_host = web_contents()->GetMainFrame();
|
2017-12-30 16:25:09 +00:00
|
|
|
if (frame_host) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* widget_host =
|
2017-12-30 16:25:09 +00:00
|
|
|
frame_host ? frame_host->GetView()->GetRenderWidgetHost() : nullptr;
|
|
|
|
if (!widget_host)
|
|
|
|
return;
|
|
|
|
widget_host->Send(
|
2019-01-12 01:00:43 +00:00
|
|
|
new WidgetMsg_DisableDeviceEmulation(widget_host->GetRoutingID()));
|
2017-12-30 16:25:09 +00:00
|
|
|
}
|
2015-08-31 09:19:19 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 09:01:17 +00:00
|
|
|
void WebContents::ToggleDevTools() {
|
|
|
|
if (IsDevToolsOpened())
|
|
|
|
CloseDevTools();
|
|
|
|
else
|
|
|
|
OpenDevTools(nullptr);
|
|
|
|
}
|
|
|
|
|
2015-04-22 07:30:10 +00:00
|
|
|
void WebContents::InspectElement(int x, int y) {
|
2015-06-24 14:23:38 +00:00
|
|
|
if (type_ == REMOTE)
|
2015-06-05 07:41:03 +00:00
|
|
|
return;
|
2015-06-24 14:23:38 +00:00
|
|
|
|
2016-09-07 02:16:52 +00:00
|
|
|
if (!enable_devtools_)
|
2016-09-06 09:53:42 +00:00
|
|
|
return;
|
|
|
|
|
2016-08-23 22:26:13 +00:00
|
|
|
if (!managed_web_contents()->GetDevToolsWebContents())
|
|
|
|
OpenDevTools(nullptr);
|
2016-12-20 00:25:32 +00:00
|
|
|
managed_web_contents()->InspectElement(x, y);
|
2015-04-22 07:30:10 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 07:41:03 +00:00
|
|
|
void WebContents::InspectServiceWorker() {
|
2015-06-24 14:23:38 +00:00
|
|
|
if (type_ == REMOTE)
|
2015-06-05 07:41:03 +00:00
|
|
|
return;
|
2015-06-24 14:23:38 +00:00
|
|
|
|
2016-09-07 02:16:52 +00:00
|
|
|
if (!enable_devtools_)
|
2016-09-06 09:53:42 +00:00
|
|
|
return;
|
|
|
|
|
2015-06-05 07:41:03 +00:00
|
|
|
for (const auto& agent_host : content::DevToolsAgentHost::GetOrCreateAll()) {
|
|
|
|
if (agent_host->GetType() ==
|
2017-01-24 07:50:02 +00:00
|
|
|
content::DevToolsAgentHost::kTypeServiceWorker) {
|
2015-06-05 09:01:17 +00:00
|
|
|
OpenDevTools(nullptr);
|
2015-06-25 06:32:38 +00:00
|
|
|
managed_web_contents()->AttachTo(agent_host);
|
2015-06-05 07:41:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
void WebContents::HasServiceWorker(const base::Callback<void(bool)>& callback) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* context = GetServiceWorkerContext(web_contents());
|
2015-06-05 07:41:03 +00:00
|
|
|
if (!context)
|
|
|
|
return;
|
|
|
|
|
2017-06-21 10:34:14 +00:00
|
|
|
struct WrappedCallback {
|
|
|
|
base::Callback<void(bool)> callback_;
|
2017-06-29 23:50:55 +00:00
|
|
|
explicit WrappedCallback(const base::Callback<void(bool)>& callback)
|
|
|
|
: callback_(callback) {}
|
2017-06-21 10:34:14 +00:00
|
|
|
void Run(content::ServiceWorkerCapability capability) {
|
2017-06-29 23:50:55 +00:00
|
|
|
callback_.Run(capability !=
|
|
|
|
content::ServiceWorkerCapability::NO_SERVICE_WORKER);
|
2017-06-21 10:34:14 +00:00
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
};
|
2017-06-30 22:47:41 +00:00
|
|
|
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* wrapped_callback = new WrappedCallback(callback);
|
2017-06-21 10:34:14 +00:00
|
|
|
|
2017-06-29 23:50:55 +00:00
|
|
|
context->CheckHasServiceWorker(
|
|
|
|
web_contents()->GetLastCommittedURL(), GURL::EmptyGURL(),
|
2018-04-20 10:55:05 +00:00
|
|
|
base::BindOnce(&WrappedCallback::Run,
|
|
|
|
base::Unretained(wrapped_callback)));
|
2015-06-05 07:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::UnregisterServiceWorker(
|
|
|
|
const base::Callback<void(bool)>& callback) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* context = GetServiceWorkerContext(web_contents());
|
2015-06-05 07:41:03 +00:00
|
|
|
if (!context)
|
|
|
|
return;
|
|
|
|
|
|
|
|
context->UnregisterServiceWorker(web_contents()->GetLastCommittedURL(),
|
|
|
|
callback);
|
|
|
|
}
|
|
|
|
|
2017-06-25 19:01:05 +00:00
|
|
|
void WebContents::SetIgnoreMenuShortcuts(bool ignore) {
|
2018-09-10 10:06:16 +00:00
|
|
|
auto* web_preferences = WebContentsPreferences::From(web_contents());
|
|
|
|
DCHECK(web_preferences);
|
|
|
|
web_preferences->preference()->SetKey("ignoreMenuShortcuts",
|
|
|
|
base::Value(ignore));
|
2017-06-25 19:01:05 +00:00
|
|
|
}
|
|
|
|
|
2015-06-15 13:40:49 +00:00
|
|
|
void WebContents::SetAudioMuted(bool muted) {
|
|
|
|
web_contents()->SetAudioMuted(muted);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WebContents::IsAudioMuted() {
|
|
|
|
return web_contents()->IsAudioMuted();
|
|
|
|
}
|
|
|
|
|
2018-07-12 11:35:11 +00:00
|
|
|
bool WebContents::IsCurrentlyAudible() {
|
|
|
|
return web_contents()->IsCurrentlyAudible();
|
|
|
|
}
|
|
|
|
|
2018-10-13 01:57:04 +00:00
|
|
|
#if BUILDFLAG(ENABLE_PRINTING)
|
2018-11-09 03:42:34 +00:00
|
|
|
void WebContents::Print(mate::Arguments* args) {
|
|
|
|
bool silent, print_background = false;
|
|
|
|
base::string16 device_name;
|
|
|
|
mate::Dictionary options = mate::Dictionary::CreateEmpty(args->isolate());
|
|
|
|
base::DictionaryValue settings;
|
|
|
|
if (args->Length() >= 1 && !args->GetNext(&options)) {
|
|
|
|
args->ThrowError("Invalid print settings specified");
|
2015-06-13 13:23:45 +00:00
|
|
|
return;
|
|
|
|
}
|
2018-11-09 03:42:34 +00:00
|
|
|
printing::CompletionCallback callback;
|
|
|
|
if (args->Length() == 2 && !args->GetNext(&callback)) {
|
|
|
|
args->ThrowError("Invalid optional callback provided");
|
|
|
|
return;
|
2017-07-21 07:16:27 +00:00
|
|
|
}
|
2018-11-09 03:42:34 +00:00
|
|
|
options.Get("silent", &silent);
|
|
|
|
options.Get("printBackground", &print_background);
|
|
|
|
if (options.Get("deviceName", &device_name) && !device_name.empty()) {
|
|
|
|
settings.SetString(printing::kSettingDeviceName, device_name);
|
|
|
|
}
|
|
|
|
auto* print_view_manager =
|
|
|
|
printing::PrintViewManagerBasic::FromWebContents(web_contents());
|
|
|
|
auto* focused_frame = web_contents()->GetFocusedFrame();
|
|
|
|
auto* rfh = focused_frame && focused_frame->HasSelection()
|
|
|
|
? focused_frame
|
|
|
|
: web_contents()->GetMainFrame();
|
|
|
|
print_view_manager->PrintNow(
|
|
|
|
rfh,
|
|
|
|
std::make_unique<PrintMsg_PrintPages>(rfh->GetRoutingID(), silent,
|
|
|
|
print_background, settings),
|
|
|
|
std::move(callback));
|
2017-02-06 09:18:04 +00:00
|
|
|
}
|
|
|
|
|
2017-05-18 17:26:22 +00:00
|
|
|
std::vector<printing::PrinterBasicInfo> WebContents::GetPrinterList() {
|
2017-05-09 02:38:03 +00:00
|
|
|
std::vector<printing::PrinterBasicInfo> printers;
|
|
|
|
auto print_backend = printing::PrintBackend::CreateInstance(nullptr);
|
2018-11-09 03:42:34 +00:00
|
|
|
{
|
|
|
|
// TODO(deepak1556): Deprecate this api in favor of an
|
|
|
|
// async version and post a non blocing task call.
|
|
|
|
base::ThreadRestrictions::ScopedAllowIO allow_io;
|
|
|
|
print_backend->EnumeratePrinters(&printers);
|
|
|
|
}
|
2017-05-09 02:38:03 +00:00
|
|
|
return printers;
|
2015-06-13 13:23:45 +00:00
|
|
|
}
|
|
|
|
|
2018-11-09 03:42:34 +00:00
|
|
|
void WebContents::PrintToPDF(
|
|
|
|
const base::DictionaryValue& settings,
|
|
|
|
const PrintPreviewMessageHandler::PrintToPDFCallback& callback) {
|
|
|
|
PrintPreviewMessageHandler::FromWebContents(web_contents())
|
|
|
|
->PrintToPDF(settings, callback);
|
2015-06-09 05:07:40 +00:00
|
|
|
}
|
2018-11-09 03:42:34 +00:00
|
|
|
#endif
|
2015-06-09 05:07:40 +00:00
|
|
|
|
2015-09-07 08:12:31 +00:00
|
|
|
void WebContents::AddWorkSpace(mate::Arguments* args,
|
|
|
|
const base::FilePath& path) {
|
2015-07-24 09:39:11 +00:00
|
|
|
if (path.empty()) {
|
2015-09-07 08:12:31 +00:00
|
|
|
args->ThrowError("path cannot be empty");
|
2015-07-24 09:39:11 +00:00
|
|
|
return;
|
|
|
|
}
|
2018-09-12 13:45:08 +00:00
|
|
|
DevToolsAddFileSystem(std::string(), path);
|
2015-07-24 09:39:11 +00:00
|
|
|
}
|
|
|
|
|
2015-09-07 08:12:31 +00:00
|
|
|
void WebContents::RemoveWorkSpace(mate::Arguments* args,
|
|
|
|
const base::FilePath& path) {
|
2015-07-24 09:39:11 +00:00
|
|
|
if (path.empty()) {
|
2015-09-07 08:12:31 +00:00
|
|
|
args->ThrowError("path cannot be empty");
|
2015-07-24 09:39:11 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
DevToolsRemoveFileSystem(path);
|
|
|
|
}
|
|
|
|
|
2015-01-19 20:09:47 +00:00
|
|
|
void WebContents::Undo() {
|
|
|
|
web_contents()->Undo();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Redo() {
|
|
|
|
web_contents()->Redo();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Cut() {
|
|
|
|
web_contents()->Cut();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Copy() {
|
|
|
|
web_contents()->Copy();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Paste() {
|
|
|
|
web_contents()->Paste();
|
|
|
|
}
|
|
|
|
|
2015-05-15 07:15:19 +00:00
|
|
|
void WebContents::PasteAndMatchStyle() {
|
|
|
|
web_contents()->PasteAndMatchStyle();
|
|
|
|
}
|
|
|
|
|
2015-01-19 20:09:47 +00:00
|
|
|
void WebContents::Delete() {
|
|
|
|
web_contents()->Delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::SelectAll() {
|
|
|
|
web_contents()->SelectAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Unselect() {
|
2017-04-13 10:21:30 +00:00
|
|
|
web_contents()->CollapseSelection();
|
2015-01-19 20:09:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::Replace(const base::string16& word) {
|
|
|
|
web_contents()->Replace(word);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::ReplaceMisspelling(const base::string16& word) {
|
|
|
|
web_contents()->ReplaceMisspelling(word);
|
|
|
|
}
|
|
|
|
|
2016-03-08 04:40:10 +00:00
|
|
|
uint32_t WebContents::FindInPage(mate::Arguments* args) {
|
2015-12-17 17:27:56 +00:00
|
|
|
base::string16 search_text;
|
2015-12-17 23:10:42 +00:00
|
|
|
if (!args->GetNext(&search_text) || search_text.empty()) {
|
2015-12-17 17:27:56 +00:00
|
|
|
args->ThrowError("Must provide a non-empty search content");
|
2015-12-17 23:10:42 +00:00
|
|
|
return 0;
|
2015-12-17 17:27:56 +00:00
|
|
|
}
|
|
|
|
|
2019-01-12 01:00:43 +00:00
|
|
|
uint32_t request_id = GetNextRequestId();
|
|
|
|
mate::Dictionary dict;
|
|
|
|
auto options = blink::mojom::FindOptions::New();
|
|
|
|
if (args->GetNext(&dict)) {
|
|
|
|
dict.Get("forward", &options->forward);
|
|
|
|
dict.Get("matchCase", &options->match_case);
|
|
|
|
dict.Get("findNext", &options->find_next);
|
|
|
|
}
|
2015-12-17 17:27:56 +00:00
|
|
|
|
2019-01-12 01:00:43 +00:00
|
|
|
web_contents()->Find(request_id, search_text, std::move(options));
|
2015-12-17 23:10:42 +00:00
|
|
|
return request_id;
|
2015-12-17 17:27:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::StopFindInPage(content::StopFindAction action) {
|
|
|
|
web_contents()->StopFinding(action);
|
|
|
|
}
|
|
|
|
|
2016-06-02 17:12:38 +00:00
|
|
|
void WebContents::ShowDefinitionForSelection() {
|
2016-06-07 20:20:06 +00:00
|
|
|
#if defined(OS_MACOSX)
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const view = web_contents()->GetRenderWidgetHostView();
|
2016-06-02 17:12:38 +00:00
|
|
|
if (view)
|
|
|
|
view->ShowDefinitionForSelection();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-07-26 19:06:11 +00:00
|
|
|
void WebContents::CopyImageAt(int x, int y) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const host = web_contents()->GetMainFrame();
|
2016-07-26 19:06:11 +00:00
|
|
|
if (host)
|
|
|
|
host->CopyImageAt(x, y);
|
|
|
|
}
|
|
|
|
|
2015-07-24 04:58:28 +00:00
|
|
|
void WebContents::Focus() {
|
|
|
|
web_contents()->Focus();
|
|
|
|
}
|
|
|
|
|
2016-08-03 03:29:55 +00:00
|
|
|
#if !defined(OS_MACOSX)
|
|
|
|
bool WebContents::IsFocused() const {
|
2018-05-16 19:12:45 +00:00
|
|
|
auto* view = web_contents()->GetRenderWidgetHostView();
|
2018-04-18 01:55:30 +00:00
|
|
|
if (!view)
|
|
|
|
return false;
|
2016-08-15 21:13:24 +00:00
|
|
|
|
|
|
|
if (GetType() != BACKGROUND_PAGE) {
|
2018-05-16 19:12:45 +00:00
|
|
|
auto* window = web_contents()->GetNativeView()->GetToplevelWindow();
|
2016-08-15 21:13:24 +00:00
|
|
|
if (window && !window->IsVisible())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return view->HasFocus();
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-07-24 04:58:28 +00:00
|
|
|
void WebContents::TabTraverse(bool reverse) {
|
|
|
|
web_contents()->FocusThroughTabTraversal(reverse);
|
|
|
|
}
|
|
|
|
|
2018-10-06 11:48:00 +00:00
|
|
|
bool WebContents::SendIPCMessage(bool internal,
|
|
|
|
bool send_to_all,
|
2018-08-24 15:30:37 +00:00
|
|
|
const std::string& channel,
|
2014-04-25 05:10:16 +00:00
|
|
|
const base::ListValue& args) {
|
2018-10-06 11:48:00 +00:00
|
|
|
return SendIPCMessageWithSender(internal, send_to_all, channel, args);
|
2018-09-01 01:13:51 +00:00
|
|
|
}
|
|
|
|
|
2018-10-06 11:48:00 +00:00
|
|
|
bool WebContents::SendIPCMessageWithSender(bool internal,
|
|
|
|
bool send_to_all,
|
2018-09-01 01:13:51 +00:00
|
|
|
const std::string& channel,
|
|
|
|
const base::ListValue& args,
|
|
|
|
int32_t sender_id) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* frame_host = web_contents()->GetMainFrame();
|
2017-12-29 20:36:11 +00:00
|
|
|
if (frame_host) {
|
2018-10-06 11:48:00 +00:00
|
|
|
return frame_host->Send(new AtomFrameMsg_Message(frame_host->GetRoutingID(),
|
|
|
|
internal, send_to_all,
|
|
|
|
channel, args, sender_id));
|
2018-03-09 09:31:09 +00:00
|
|
|
}
|
|
|
|
return false;
|
2014-04-25 05:10:16 +00:00
|
|
|
}
|
|
|
|
|
2018-12-10 00:37:42 +00:00
|
|
|
bool WebContents::SendIPCMessageToFrame(bool internal,
|
|
|
|
bool send_to_all,
|
|
|
|
int32_t frame_id,
|
|
|
|
const std::string& channel,
|
|
|
|
const base::ListValue& args) {
|
|
|
|
auto frames = web_contents()->GetAllFrames();
|
|
|
|
auto iter = std::find_if(frames.begin(), frames.end(), [frame_id](auto* f) {
|
|
|
|
return f->GetRoutingID() == frame_id;
|
|
|
|
});
|
|
|
|
if (iter == frames.end())
|
|
|
|
return false;
|
|
|
|
if (!(*iter)->IsRenderFrameLive())
|
|
|
|
return false;
|
|
|
|
return (*iter)->Send(new AtomFrameMsg_Message(
|
|
|
|
frame_id, internal, send_to_all, channel, args, 0 /* sender_id */));
|
|
|
|
}
|
|
|
|
|
2015-09-18 06:20:31 +00:00
|
|
|
void WebContents::SendInputEvent(v8::Isolate* isolate,
|
|
|
|
v8::Local<v8::Value> input_event) {
|
2018-04-12 12:54:04 +00:00
|
|
|
content::RenderWidgetHostView* view =
|
|
|
|
web_contents()->GetRenderWidgetHostView();
|
2015-09-18 06:20:31 +00:00
|
|
|
if (!view)
|
|
|
|
return;
|
|
|
|
|
2018-04-12 12:54:04 +00:00
|
|
|
content::RenderWidgetHost* rwh = view->GetRenderWidgetHost();
|
2018-04-18 01:55:30 +00:00
|
|
|
blink::WebInputEvent::Type type =
|
|
|
|
mate::GetWebInputEventType(isolate, input_event);
|
2017-06-16 20:42:33 +00:00
|
|
|
if (blink::WebInputEvent::IsMouseEventType(type)) {
|
2015-09-18 06:20:31 +00:00
|
|
|
blink::WebMouseEvent mouse_event;
|
|
|
|
if (mate::ConvertFromV8(isolate, input_event, &mouse_event)) {
|
2018-05-15 01:59:22 +00:00
|
|
|
if (IsOffScreen()) {
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2018-05-15 01:59:22 +00:00
|
|
|
GetOffScreenRenderWidgetHostView()->SendMouseEvent(mouse_event);
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
rwh->ForwardMouseEvent(mouse_event);
|
|
|
|
}
|
2015-09-18 06:20:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-06-16 20:42:33 +00:00
|
|
|
} else if (blink::WebInputEvent::IsKeyboardEventType(type)) {
|
2017-04-13 10:21:30 +00:00
|
|
|
content::NativeWebKeyboardEvent keyboard_event(
|
2017-06-16 20:42:33 +00:00
|
|
|
blink::WebKeyboardEvent::kRawKeyDown,
|
2018-04-18 01:55:30 +00:00
|
|
|
blink::WebInputEvent::kNoModifiers, ui::EventTimeForNow());
|
2015-09-18 06:20:31 +00:00
|
|
|
if (mate::ConvertFromV8(isolate, input_event, &keyboard_event)) {
|
2018-04-12 12:54:04 +00:00
|
|
|
rwh->ForwardKeyboardEvent(keyboard_event);
|
2015-09-18 06:20:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-06-16 20:42:33 +00:00
|
|
|
} else if (type == blink::WebInputEvent::kMouseWheel) {
|
2015-09-18 06:20:31 +00:00
|
|
|
blink::WebMouseWheelEvent mouse_wheel_event;
|
|
|
|
if (mate::ConvertFromV8(isolate, input_event, &mouse_wheel_event)) {
|
2018-05-15 01:59:22 +00:00
|
|
|
if (IsOffScreen()) {
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2018-05-15 01:59:22 +00:00
|
|
|
GetOffScreenRenderWidgetHostView()->SendMouseWheelEvent(
|
|
|
|
mouse_wheel_event);
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
rwh->ForwardWheelEvent(mouse_wheel_event);
|
|
|
|
}
|
2015-09-18 06:20:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
isolate->ThrowException(
|
|
|
|
v8::Exception::Error(mate::StringToV8(isolate, "Invalid event object")));
|
2015-09-18 06:20:31 +00:00
|
|
|
}
|
|
|
|
|
2016-06-26 02:46:40 +00:00
|
|
|
void WebContents::BeginFrameSubscription(mate::Arguments* args) {
|
2018-05-14 17:09:05 +00:00
|
|
|
bool only_dirty = false;
|
2016-06-26 02:46:40 +00:00
|
|
|
FrameSubscriber::FrameCaptureCallback callback;
|
2016-06-21 00:15:39 +00:00
|
|
|
|
2018-05-14 17:09:05 +00:00
|
|
|
args->GetNext(&only_dirty);
|
2016-06-25 16:23:40 +00:00
|
|
|
if (!args->GetNext(&callback)) {
|
2016-06-26 02:46:40 +00:00
|
|
|
args->ThrowError();
|
|
|
|
return;
|
2016-06-25 16:23:40 +00:00
|
|
|
}
|
2016-06-21 00:15:39 +00:00
|
|
|
|
2018-05-14 15:55:39 +00:00
|
|
|
frame_subscriber_.reset(
|
2018-05-14 17:09:05 +00:00
|
|
|
new FrameSubscriber(isolate(), web_contents(), callback, only_dirty));
|
2015-09-18 07:57:43 +00:00
|
|
|
}
|
2015-09-18 06:20:31 +00:00
|
|
|
|
2015-09-18 07:57:43 +00:00
|
|
|
void WebContents::EndFrameSubscription() {
|
2018-04-18 00:45:15 +00:00
|
|
|
frame_subscriber_.reset();
|
2015-09-18 07:57:43 +00:00
|
|
|
}
|
2015-09-18 06:20:31 +00:00
|
|
|
|
2016-07-03 04:58:31 +00:00
|
|
|
void WebContents::StartDrag(const mate::Dictionary& item,
|
|
|
|
mate::Arguments* args) {
|
|
|
|
base::FilePath file;
|
|
|
|
std::vector<base::FilePath> files;
|
|
|
|
if (!item.Get("files", &files) && item.Get("file", &file)) {
|
|
|
|
files.push_back(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
mate::Handle<NativeImage> icon;
|
|
|
|
if (!item.Get("icon", &icon) && !file.empty()) {
|
|
|
|
// TODO(zcbenz): Set default icon from file.
|
|
|
|
}
|
|
|
|
|
|
|
|
// Error checking.
|
|
|
|
if (icon.IsEmpty()) {
|
2017-01-26 18:38:13 +00:00
|
|
|
args->ThrowError("Must specify 'icon' option");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(OS_MACOSX)
|
|
|
|
// NSWindow.dragImage requires a non-empty NSImage
|
|
|
|
if (icon->image().IsEmpty()) {
|
2017-01-26 18:34:27 +00:00
|
|
|
args->ThrowError("Must specify non-empty 'icon' option");
|
2016-07-03 04:58:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-01-26 18:38:13 +00:00
|
|
|
#endif
|
2016-07-03 04:58:31 +00:00
|
|
|
|
|
|
|
// Start dragging.
|
|
|
|
if (!files.empty()) {
|
2019-01-09 20:10:26 +00:00
|
|
|
base::MessageLoopCurrent::ScopedNestableTaskAllower allow;
|
2016-07-03 04:58:31 +00:00
|
|
|
DragFileItems(files, icon->image(), web_contents()->GetNativeView());
|
|
|
|
} else {
|
2017-01-26 18:34:27 +00:00
|
|
|
args->ThrowError("Must specify either 'file' or 'files' option");
|
2016-07-03 04:58:31 +00:00
|
|
|
}
|
2016-07-03 03:26:43 +00:00
|
|
|
}
|
|
|
|
|
2018-11-28 04:50:53 +00:00
|
|
|
v8::Local<v8::Promise> WebContents::CapturePage(mate::Arguments* args) {
|
2016-07-05 22:43:57 +00:00
|
|
|
gfx::Rect rect;
|
2018-11-28 04:50:53 +00:00
|
|
|
scoped_refptr<util::Promise> promise = new util::Promise(isolate());
|
2016-07-05 22:43:57 +00:00
|
|
|
|
2018-11-28 04:50:53 +00:00
|
|
|
// get rect arguments if they exist
|
|
|
|
args->GetNext(&rect);
|
2016-07-05 22:43:57 +00:00
|
|
|
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const view = web_contents()->GetRenderWidgetHostView();
|
2017-04-13 10:21:30 +00:00
|
|
|
if (!view) {
|
2018-11-28 04:50:53 +00:00
|
|
|
promise->Resolve(gfx::Image());
|
|
|
|
return promise->GetHandle();
|
2016-07-05 22:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Capture full page if user doesn't specify a |rect|.
|
2018-04-18 01:55:30 +00:00
|
|
|
const gfx::Size view_size =
|
|
|
|
rect.IsEmpty() ? view->GetViewBounds().size() : rect.size();
|
2016-07-05 22:43:57 +00:00
|
|
|
|
|
|
|
// By default, the requested bitmap size is the view size in screen
|
|
|
|
// coordinates. However, if there's more pixel detail available on the
|
|
|
|
// current system, increase the requested bitmap size to capture it all.
|
|
|
|
gfx::Size bitmap_size = view_size;
|
|
|
|
const gfx::NativeView native_view = view->GetNativeView();
|
2018-04-18 01:55:30 +00:00
|
|
|
const float scale = display::Screen::GetScreen()
|
|
|
|
->GetDisplayNearestView(native_view)
|
|
|
|
.device_scale_factor();
|
2016-07-05 22:43:57 +00:00
|
|
|
if (scale > 1.0f)
|
|
|
|
bitmap_size = gfx::ScaleToCeiledSize(view_size, scale);
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
view->CopyFromSurface(gfx::Rect(rect.origin(), view_size), bitmap_size,
|
2018-11-28 04:50:53 +00:00
|
|
|
base::BindOnce(&OnCapturePageDone, promise));
|
|
|
|
return promise->GetHandle();
|
2016-07-05 22:43:57 +00:00
|
|
|
}
|
|
|
|
|
2016-01-30 05:33:55 +00:00
|
|
|
void WebContents::OnCursorChange(const content::WebCursor& cursor) {
|
2017-06-16 20:59:57 +00:00
|
|
|
content::CursorInfo info;
|
2016-02-01 06:17:58 +00:00
|
|
|
cursor.GetCursorInfo(&info);
|
2016-01-31 16:00:14 +00:00
|
|
|
|
|
|
|
if (cursor.IsCustom()) {
|
|
|
|
Emit("cursor-changed", CursorTypeToString(info),
|
2018-04-18 01:55:30 +00:00
|
|
|
gfx::Image::CreateFrom1xBitmap(info.custom_image),
|
|
|
|
info.image_scale_factor,
|
|
|
|
gfx::Size(info.custom_image.width(), info.custom_image.height()),
|
|
|
|
info.hotspot);
|
2016-01-31 16:00:14 +00:00
|
|
|
} else {
|
|
|
|
Emit("cursor-changed", CursorTypeToString(info));
|
|
|
|
}
|
2016-01-30 05:33:55 +00:00
|
|
|
}
|
2016-07-18 06:54:52 +00:00
|
|
|
|
2015-06-05 07:18:15 +00:00
|
|
|
bool WebContents::IsGuest() const {
|
2015-06-24 14:23:38 +00:00
|
|
|
return type_ == WEB_VIEW;
|
2015-06-05 07:18:15 +00:00
|
|
|
}
|
|
|
|
|
2018-08-16 22:57:40 +00:00
|
|
|
void WebContents::AttachToIframe(content::WebContents* embedder_web_contents,
|
|
|
|
int embedder_frame_id) {
|
|
|
|
if (guest_delegate_)
|
|
|
|
guest_delegate_->AttachToIframe(embedder_web_contents, embedder_frame_id);
|
|
|
|
}
|
|
|
|
|
2016-07-28 10:10:56 +00:00
|
|
|
bool WebContents::IsOffScreen() const {
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2016-07-28 10:10:56 +00:00
|
|
|
return type_ == OFF_SCREEN;
|
2017-06-26 09:13:05 +00:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2016-07-28 10:10:56 +00:00
|
|
|
}
|
|
|
|
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2016-08-04 04:47:52 +00:00
|
|
|
void WebContents::OnPaint(const gfx::Rect& dirty_rect, const SkBitmap& bitmap) {
|
2017-04-12 14:16:27 +00:00
|
|
|
Emit("paint", dirty_rect, gfx::Image::CreateFrom1xBitmap(bitmap));
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::StartPainting() {
|
2018-03-20 01:25:51 +00:00
|
|
|
auto* osr_wcv = GetOffScreenWebContentsView();
|
2018-01-25 13:46:30 +00:00
|
|
|
if (osr_wcv)
|
|
|
|
osr_wcv->SetPainting(true);
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::StopPainting() {
|
2018-03-20 01:25:51 +00:00
|
|
|
auto* osr_wcv = GetOffScreenWebContentsView();
|
2018-01-25 13:46:30 +00:00
|
|
|
if (osr_wcv)
|
|
|
|
osr_wcv->SetPainting(false);
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool WebContents::IsPainting() const {
|
2018-03-20 01:25:51 +00:00
|
|
|
auto* osr_wcv = GetOffScreenWebContentsView();
|
2018-01-25 13:46:30 +00:00
|
|
|
return osr_wcv && osr_wcv->IsPainting();
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::SetFrameRate(int frame_rate) {
|
2018-03-20 01:25:51 +00:00
|
|
|
auto* osr_wcv = GetOffScreenWebContentsView();
|
2018-01-25 13:46:30 +00:00
|
|
|
if (osr_wcv)
|
|
|
|
osr_wcv->SetFrameRate(frame_rate);
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int WebContents::GetFrameRate() const {
|
2018-03-20 01:25:51 +00:00
|
|
|
auto* osr_wcv = GetOffScreenWebContentsView();
|
2018-01-25 13:46:30 +00:00
|
|
|
return osr_wcv ? osr_wcv->GetFrameRate() : 0;
|
2016-08-03 03:29:55 +00:00
|
|
|
}
|
2018-10-01 20:00:53 +00:00
|
|
|
#endif
|
2016-08-03 03:29:55 +00:00
|
|
|
|
2016-09-01 17:25:12 +00:00
|
|
|
void WebContents::Invalidate() {
|
2017-02-08 07:08:03 +00:00
|
|
|
if (IsOffScreen()) {
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2018-05-15 01:59:22 +00:00
|
|
|
auto* osr_rwhv = GetOffScreenRenderWidgetHostView();
|
2017-02-08 07:08:03 +00:00
|
|
|
if (osr_rwhv)
|
|
|
|
osr_rwhv->Invalidate();
|
2017-06-26 09:13:05 +00:00
|
|
|
#endif
|
2017-02-08 08:05:16 +00:00
|
|
|
} else {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* const window = owner_window();
|
2017-02-14 20:30:23 +00:00
|
|
|
if (window)
|
|
|
|
window->Invalidate();
|
2017-02-08 07:08:03 +00:00
|
|
|
}
|
2016-09-01 17:25:12 +00:00
|
|
|
}
|
2016-08-03 03:29:55 +00:00
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
gfx::Size WebContents::GetSizeForNewRenderView(content::WebContents* wc) const {
|
2017-05-11 21:48:14 +00:00
|
|
|
if (IsOffScreen() && wc == web_contents()) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* relay = NativeWindowRelay::FromWebContents(web_contents());
|
2017-05-11 21:48:14 +00:00
|
|
|
if (relay) {
|
2018-10-02 22:14:43 +00:00
|
|
|
auto* owner_window = relay->GetNativeWindow();
|
|
|
|
return owner_window ? owner_window->GetSize() : gfx::Size();
|
2017-05-11 21:48:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return gfx::Size();
|
|
|
|
}
|
|
|
|
|
2017-01-29 14:13:20 +00:00
|
|
|
void WebContents::SetZoomLevel(double level) {
|
2017-01-30 11:18:40 +00:00
|
|
|
zoom_controller_->SetZoomLevel(level);
|
2017-01-29 14:13:20 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 17:15:23 +00:00
|
|
|
double WebContents::GetZoomLevel() const {
|
2017-01-30 11:18:40 +00:00
|
|
|
return zoom_controller_->GetZoomLevel();
|
2017-01-29 14:13:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebContents::SetZoomFactor(double factor) {
|
|
|
|
auto level = content::ZoomFactorToZoomLevel(factor);
|
|
|
|
SetZoomLevel(level);
|
|
|
|
}
|
|
|
|
|
2018-08-09 17:15:23 +00:00
|
|
|
double WebContents::GetZoomFactor() const {
|
2017-01-29 14:13:20 +00:00
|
|
|
auto level = GetZoomLevel();
|
|
|
|
return content::ZoomLevelToZoomFactor(level);
|
|
|
|
}
|
|
|
|
|
2018-03-09 09:31:09 +00:00
|
|
|
void WebContents::OnSetTemporaryZoomLevel(content::RenderFrameHost* rfh,
|
|
|
|
double level,
|
2017-01-30 17:06:50 +00:00
|
|
|
IPC::Message* reply_msg) {
|
|
|
|
zoom_controller_->SetTemporaryZoomLevel(level);
|
2017-01-31 09:15:45 +00:00
|
|
|
double new_level = zoom_controller_->GetZoomLevel();
|
2017-12-29 20:36:11 +00:00
|
|
|
AtomFrameHostMsg_SetTemporaryZoomLevel::WriteReplyParams(reply_msg,
|
|
|
|
new_level);
|
|
|
|
rfh->Send(reply_msg);
|
2017-01-30 17:06:50 +00:00
|
|
|
}
|
|
|
|
|
2017-12-29 20:36:11 +00:00
|
|
|
void WebContents::OnGetZoomLevel(content::RenderFrameHost* rfh,
|
|
|
|
IPC::Message* reply_msg) {
|
|
|
|
AtomFrameHostMsg_GetZoomLevel::WriteReplyParams(reply_msg, GetZoomLevel());
|
|
|
|
rfh->Send(reply_msg);
|
2016-09-01 17:25:12 +00:00
|
|
|
}
|
2016-08-03 03:29:55 +00:00
|
|
|
|
2018-08-10 22:19:49 +00:00
|
|
|
v8::Local<v8::Value> WebContents::GetPreloadPath(v8::Isolate* isolate) const {
|
|
|
|
if (auto* web_preferences = WebContentsPreferences::From(web_contents())) {
|
|
|
|
base::FilePath::StringType preload;
|
|
|
|
if (web_preferences->GetPreloadPath(&preload)) {
|
|
|
|
return mate::ConvertToV8(isolate, preload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return v8::Null(isolate);
|
|
|
|
}
|
|
|
|
|
2018-08-09 17:15:23 +00:00
|
|
|
v8::Local<v8::Value> WebContents::GetWebPreferences(
|
|
|
|
v8::Isolate* isolate) const {
|
2018-03-08 07:12:45 +00:00
|
|
|
auto* web_preferences = WebContentsPreferences::From(web_contents());
|
2018-03-14 02:19:22 +00:00
|
|
|
if (!web_preferences)
|
|
|
|
return v8::Null(isolate);
|
2018-08-03 21:23:07 +00:00
|
|
|
return mate::ConvertToV8(isolate, *web_preferences->preference());
|
2015-09-22 13:56:56 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 17:15:23 +00:00
|
|
|
v8::Local<v8::Value> WebContents::GetLastWebPreferences(
|
|
|
|
v8::Isolate* isolate) const {
|
|
|
|
auto* web_preferences = WebContentsPreferences::From(web_contents());
|
2018-03-15 04:56:46 +00:00
|
|
|
if (!web_preferences)
|
|
|
|
return v8::Null(isolate);
|
2018-08-03 21:23:07 +00:00
|
|
|
return mate::ConvertToV8(isolate, *web_preferences->last_preference());
|
2018-03-15 04:56:46 +00:00
|
|
|
}
|
|
|
|
|
2018-10-13 17:50:07 +00:00
|
|
|
bool WebContents::IsRemoteModuleEnabled() const {
|
|
|
|
if (auto* web_preferences = WebContentsPreferences::From(web_contents())) {
|
|
|
|
return web_preferences->IsRemoteModuleEnabled();
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-09 17:15:23 +00:00
|
|
|
v8::Local<v8::Value> WebContents::GetOwnerBrowserWindow() const {
|
2015-10-01 05:45:59 +00:00
|
|
|
if (owner_window())
|
2018-02-22 03:49:17 +00:00
|
|
|
return BrowserWindow::From(isolate(), owner_window());
|
2015-10-01 05:45:59 +00:00
|
|
|
else
|
|
|
|
return v8::Null(isolate());
|
|
|
|
}
|
|
|
|
|
2016-05-17 12:56:47 +00:00
|
|
|
int32_t WebContents::ID() const {
|
|
|
|
return weak_map_id();
|
|
|
|
}
|
|
|
|
|
2015-10-01 03:14:19 +00:00
|
|
|
v8::Local<v8::Value> WebContents::Session(v8::Isolate* isolate) {
|
|
|
|
return v8::Local<v8::Value>::New(isolate, session_);
|
|
|
|
}
|
|
|
|
|
2016-02-17 08:52:19 +00:00
|
|
|
content::WebContents* WebContents::HostWebContents() {
|
|
|
|
if (!embedder_)
|
|
|
|
return nullptr;
|
|
|
|
return embedder_->web_contents();
|
|
|
|
}
|
|
|
|
|
2016-09-08 17:01:01 +00:00
|
|
|
void WebContents::SetEmbedder(const WebContents* embedder) {
|
|
|
|
if (embedder) {
|
|
|
|
NativeWindow* owner_window = nullptr;
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* relay = NativeWindowRelay::FromWebContents(embedder->web_contents());
|
2016-09-08 17:01:01 +00:00
|
|
|
if (relay) {
|
2018-10-02 22:14:43 +00:00
|
|
|
owner_window = relay->GetNativeWindow();
|
2016-09-08 17:01:01 +00:00
|
|
|
}
|
|
|
|
if (owner_window)
|
|
|
|
SetOwnerWindow(owner_window);
|
|
|
|
|
|
|
|
content::RenderWidgetHostView* rwhv =
|
|
|
|
web_contents()->GetRenderWidgetHostView();
|
|
|
|
if (rwhv) {
|
|
|
|
rwhv->Hide();
|
|
|
|
rwhv->Show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-30 07:37:26 +00:00
|
|
|
void WebContents::SetDevToolsWebContents(const WebContents* devtools) {
|
|
|
|
if (managed_web_contents())
|
|
|
|
managed_web_contents()->SetDevToolsWebContents(devtools->web_contents());
|
|
|
|
}
|
|
|
|
|
2017-08-21 10:50:12 +00:00
|
|
|
v8::Local<v8::Value> WebContents::GetNativeView() const {
|
|
|
|
gfx::NativeView ptr = web_contents()->GetNativeView();
|
2018-04-18 01:55:30 +00:00
|
|
|
auto buffer = node::Buffer::Copy(isolate(), reinterpret_cast<char*>(&ptr),
|
|
|
|
sizeof(gfx::NativeView));
|
2017-08-21 10:50:12 +00:00
|
|
|
if (buffer.IsEmpty())
|
|
|
|
return v8::Null(isolate());
|
|
|
|
else
|
|
|
|
return buffer.ToLocalChecked();
|
|
|
|
}
|
|
|
|
|
2015-10-01 03:14:19 +00:00
|
|
|
v8::Local<v8::Value> WebContents::DevToolsWebContents(v8::Isolate* isolate) {
|
|
|
|
if (devtools_web_contents_.IsEmpty())
|
|
|
|
return v8::Null(isolate);
|
|
|
|
else
|
|
|
|
return v8::Local<v8::Value>::New(isolate, devtools_web_contents_);
|
|
|
|
}
|
|
|
|
|
2016-01-21 18:22:23 +00:00
|
|
|
v8::Local<v8::Value> WebContents::Debugger(v8::Isolate* isolate) {
|
|
|
|
if (debugger_.IsEmpty()) {
|
|
|
|
auto handle = atom::api::Debugger::Create(isolate, web_contents());
|
|
|
|
debugger_.Reset(isolate, handle.ToV8());
|
|
|
|
}
|
|
|
|
return v8::Local<v8::Value>::New(isolate, debugger_);
|
|
|
|
}
|
|
|
|
|
2017-06-30 22:47:41 +00:00
|
|
|
void WebContents::GrantOriginAccess(const GURL& url) {
|
2018-10-02 19:01:04 +00:00
|
|
|
content::ChildProcessSecurityPolicy::GetInstance()->GrantCommitOrigin(
|
2018-04-12 07:37:54 +00:00
|
|
|
web_contents()->GetMainFrame()->GetProcess()->GetID(),
|
|
|
|
url::Origin::Create(url));
|
2017-06-30 22:47:41 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 18:00:31 +00:00
|
|
|
bool WebContents::TakeHeapSnapshot(const base::FilePath& file_path,
|
|
|
|
const std::string& channel) {
|
|
|
|
base::ThreadRestrictions::ScopedAllowIO allow_io;
|
|
|
|
|
|
|
|
base::File file(file_path,
|
|
|
|
base::File::FLAG_CREATE_ALWAYS | base::File::FLAG_WRITE);
|
|
|
|
if (!file.IsValid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
auto* frame_host = web_contents()->GetMainFrame();
|
|
|
|
if (!frame_host)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return frame_host->Send(new AtomFrameMsg_TakeHeapSnapshot(
|
|
|
|
frame_host->GetRoutingID(),
|
|
|
|
IPC::TakePlatformFileForTransit(std::move(file)), channel));
|
|
|
|
}
|
|
|
|
|
2015-12-03 08:04:46 +00:00
|
|
|
// static
|
|
|
|
void WebContents::BuildPrototype(v8::Isolate* isolate,
|
2016-08-02 09:08:12 +00:00
|
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
2016-08-02 10:28:12 +00:00
|
|
|
prototype->SetClassName(mate::StringToV8(isolate, "WebContents"));
|
2016-08-02 09:08:12 +00:00
|
|
|
mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
|
2015-12-03 08:04:46 +00:00
|
|
|
.MakeDestroyable()
|
2018-10-19 00:32:22 +00:00
|
|
|
.SetMethod("setBackgroundThrottling",
|
|
|
|
&WebContents::SetBackgroundThrottling)
|
2016-09-06 01:28:40 +00:00
|
|
|
.SetMethod("getProcessId", &WebContents::GetProcessID)
|
2017-04-18 10:31:20 +00:00
|
|
|
.SetMethod("getOSProcessId", &WebContents::GetOSProcessID)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("equal", &WebContents::Equal)
|
|
|
|
.SetMethod("_loadURL", &WebContents::LoadURL)
|
2015-12-03 18:31:51 +00:00
|
|
|
.SetMethod("downloadURL", &WebContents::DownloadURL)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("_getURL", &WebContents::GetURL)
|
|
|
|
.SetMethod("getTitle", &WebContents::GetTitle)
|
|
|
|
.SetMethod("isLoading", &WebContents::IsLoading)
|
2016-04-18 17:37:08 +00:00
|
|
|
.SetMethod("isLoadingMainFrame", &WebContents::IsLoadingMainFrame)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("isWaitingForResponse", &WebContents::IsWaitingForResponse)
|
|
|
|
.SetMethod("_stop", &WebContents::Stop)
|
|
|
|
.SetMethod("_goBack", &WebContents::GoBack)
|
|
|
|
.SetMethod("_goForward", &WebContents::GoForward)
|
|
|
|
.SetMethod("_goToOffset", &WebContents::GoToOffset)
|
|
|
|
.SetMethod("isCrashed", &WebContents::IsCrashed)
|
|
|
|
.SetMethod("setUserAgent", &WebContents::SetUserAgent)
|
|
|
|
.SetMethod("getUserAgent", &WebContents::GetUserAgent)
|
|
|
|
.SetMethod("savePage", &WebContents::SavePage)
|
|
|
|
.SetMethod("openDevTools", &WebContents::OpenDevTools)
|
|
|
|
.SetMethod("closeDevTools", &WebContents::CloseDevTools)
|
|
|
|
.SetMethod("isDevToolsOpened", &WebContents::IsDevToolsOpened)
|
2016-01-01 05:11:21 +00:00
|
|
|
.SetMethod("isDevToolsFocused", &WebContents::IsDevToolsFocused)
|
2017-06-25 19:01:05 +00:00
|
|
|
.SetMethod("enableDeviceEmulation", &WebContents::EnableDeviceEmulation)
|
|
|
|
.SetMethod("disableDeviceEmulation", &WebContents::DisableDeviceEmulation)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("toggleDevTools", &WebContents::ToggleDevTools)
|
|
|
|
.SetMethod("inspectElement", &WebContents::InspectElement)
|
2018-04-18 01:55:30 +00:00
|
|
|
.SetMethod("setIgnoreMenuShortcuts", &WebContents::SetIgnoreMenuShortcuts)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("setAudioMuted", &WebContents::SetAudioMuted)
|
|
|
|
.SetMethod("isAudioMuted", &WebContents::IsAudioMuted)
|
2018-07-12 11:35:11 +00:00
|
|
|
.SetMethod("isCurrentlyAudible", &WebContents::IsCurrentlyAudible)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("undo", &WebContents::Undo)
|
|
|
|
.SetMethod("redo", &WebContents::Redo)
|
|
|
|
.SetMethod("cut", &WebContents::Cut)
|
|
|
|
.SetMethod("copy", &WebContents::Copy)
|
|
|
|
.SetMethod("paste", &WebContents::Paste)
|
|
|
|
.SetMethod("pasteAndMatchStyle", &WebContents::PasteAndMatchStyle)
|
|
|
|
.SetMethod("delete", &WebContents::Delete)
|
|
|
|
.SetMethod("selectAll", &WebContents::SelectAll)
|
|
|
|
.SetMethod("unselect", &WebContents::Unselect)
|
|
|
|
.SetMethod("replace", &WebContents::Replace)
|
|
|
|
.SetMethod("replaceMisspelling", &WebContents::ReplaceMisspelling)
|
2015-12-17 17:27:56 +00:00
|
|
|
.SetMethod("findInPage", &WebContents::FindInPage)
|
|
|
|
.SetMethod("stopFindInPage", &WebContents::StopFindInPage)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("focus", &WebContents::Focus)
|
2016-08-03 03:29:55 +00:00
|
|
|
.SetMethod("isFocused", &WebContents::IsFocused)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("tabTraverse", &WebContents::TabTraverse)
|
|
|
|
.SetMethod("_send", &WebContents::SendIPCMessage)
|
2018-12-10 00:37:42 +00:00
|
|
|
.SetMethod("_sendToFrame", &WebContents::SendIPCMessageToFrame)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("sendInputEvent", &WebContents::SendInputEvent)
|
2017-06-25 19:01:05 +00:00
|
|
|
.SetMethod("beginFrameSubscription", &WebContents::BeginFrameSubscription)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("endFrameSubscription", &WebContents::EndFrameSubscription)
|
2016-07-03 03:26:43 +00:00
|
|
|
.SetMethod("startDrag", &WebContents::StartDrag)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("isGuest", &WebContents::IsGuest)
|
2018-08-16 22:57:40 +00:00
|
|
|
.SetMethod("attachToIframe", &WebContents::AttachToIframe)
|
2016-08-03 03:29:55 +00:00
|
|
|
.SetMethod("isOffscreen", &WebContents::IsOffScreen)
|
2018-10-01 20:00:53 +00:00
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
2016-08-03 03:29:55 +00:00
|
|
|
.SetMethod("startPainting", &WebContents::StartPainting)
|
|
|
|
.SetMethod("stopPainting", &WebContents::StopPainting)
|
|
|
|
.SetMethod("isPainting", &WebContents::IsPainting)
|
|
|
|
.SetMethod("setFrameRate", &WebContents::SetFrameRate)
|
|
|
|
.SetMethod("getFrameRate", &WebContents::GetFrameRate)
|
2018-10-01 20:00:53 +00:00
|
|
|
#endif
|
2016-09-01 17:25:12 +00:00
|
|
|
.SetMethod("invalidate", &WebContents::Invalidate)
|
2017-01-29 14:13:20 +00:00
|
|
|
.SetMethod("setZoomLevel", &WebContents::SetZoomLevel)
|
2019-01-21 07:40:27 +00:00
|
|
|
.SetMethod("getZoomLevel", &WebContents::GetZoomLevel)
|
2017-01-29 14:13:20 +00:00
|
|
|
.SetMethod("setZoomFactor", &WebContents::SetZoomFactor)
|
2019-01-21 07:40:27 +00:00
|
|
|
.SetMethod("getZoomFactor", &WebContents::GetZoomFactor)
|
2016-06-08 18:41:14 +00:00
|
|
|
.SetMethod("getType", &WebContents::GetType)
|
2018-08-10 22:19:49 +00:00
|
|
|
.SetMethod("_getPreloadPath", &WebContents::GetPreloadPath)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("getWebPreferences", &WebContents::GetWebPreferences)
|
2018-03-15 04:56:46 +00:00
|
|
|
.SetMethod("getLastWebPreferences", &WebContents::GetLastWebPreferences)
|
2018-10-13 17:50:07 +00:00
|
|
|
.SetMethod("_isRemoteModuleEnabled", &WebContents::IsRemoteModuleEnabled)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("getOwnerBrowserWindow", &WebContents::GetOwnerBrowserWindow)
|
|
|
|
.SetMethod("hasServiceWorker", &WebContents::HasServiceWorker)
|
|
|
|
.SetMethod("unregisterServiceWorker",
|
|
|
|
&WebContents::UnregisterServiceWorker)
|
|
|
|
.SetMethod("inspectServiceWorker", &WebContents::InspectServiceWorker)
|
2018-11-09 03:42:34 +00:00
|
|
|
#if BUILDFLAG(ENABLE_PRINTING)
|
|
|
|
.SetMethod("_print", &WebContents::Print)
|
|
|
|
.SetMethod("_getPrinters", &WebContents::GetPrinterList)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("_printToPDF", &WebContents::PrintToPDF)
|
2018-11-09 03:42:34 +00:00
|
|
|
#endif
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetMethod("addWorkSpace", &WebContents::AddWorkSpace)
|
|
|
|
.SetMethod("removeWorkSpace", &WebContents::RemoveWorkSpace)
|
2016-06-07 20:13:49 +00:00
|
|
|
.SetMethod("showDefinitionForSelection",
|
|
|
|
&WebContents::ShowDefinitionForSelection)
|
2016-07-26 19:06:11 +00:00
|
|
|
.SetMethod("copyImageAt", &WebContents::CopyImageAt)
|
2016-07-05 22:43:57 +00:00
|
|
|
.SetMethod("capturePage", &WebContents::CapturePage)
|
2016-09-08 17:01:01 +00:00
|
|
|
.SetMethod("setEmbedder", &WebContents::SetEmbedder)
|
2017-11-30 07:37:26 +00:00
|
|
|
.SetMethod("setDevToolsWebContents", &WebContents::SetDevToolsWebContents)
|
2017-08-21 10:50:12 +00:00
|
|
|
.SetMethod("getNativeView", &WebContents::GetNativeView)
|
2017-03-08 14:55:59 +00:00
|
|
|
.SetMethod("setWebRTCIPHandlingPolicy",
|
|
|
|
&WebContents::SetWebRTCIPHandlingPolicy)
|
|
|
|
.SetMethod("getWebRTCIPHandlingPolicy",
|
|
|
|
&WebContents::GetWebRTCIPHandlingPolicy)
|
2017-06-30 22:47:41 +00:00
|
|
|
.SetMethod("_grantOriginAccess", &WebContents::GrantOriginAccess)
|
2018-09-18 18:00:31 +00:00
|
|
|
.SetMethod("_takeHeapSnapshot", &WebContents::TakeHeapSnapshot)
|
2016-05-17 12:56:47 +00:00
|
|
|
.SetProperty("id", &WebContents::ID)
|
2015-12-03 08:04:46 +00:00
|
|
|
.SetProperty("session", &WebContents::Session)
|
2016-02-17 08:52:19 +00:00
|
|
|
.SetProperty("hostWebContents", &WebContents::HostWebContents)
|
2016-01-21 18:22:23 +00:00
|
|
|
.SetProperty("devToolsWebContents", &WebContents::DevToolsWebContents)
|
|
|
|
.SetProperty("debugger", &WebContents::Debugger);
|
2015-07-06 10:21:04 +00:00
|
|
|
}
|
|
|
|
|
2015-07-14 19:13:25 +00:00
|
|
|
AtomBrowserContext* WebContents::GetBrowserContext() const {
|
|
|
|
return static_cast<AtomBrowserContext*>(web_contents()->GetBrowserContext());
|
|
|
|
}
|
|
|
|
|
2018-03-09 09:31:09 +00:00
|
|
|
void WebContents::OnRendererMessage(content::RenderFrameHost* frame_host,
|
2018-08-24 15:30:37 +00:00
|
|
|
const std::string& channel,
|
2014-04-25 08:13:16 +00:00
|
|
|
const base::ListValue& args) {
|
|
|
|
// webContents.emit(channel, new Event(), args...);
|
2018-12-10 00:37:42 +00:00
|
|
|
EmitWithSender(channel, frame_host, nullptr, args);
|
2014-04-25 08:13:16 +00:00
|
|
|
}
|
|
|
|
|
2018-03-09 09:31:09 +00:00
|
|
|
void WebContents::OnRendererMessageSync(content::RenderFrameHost* frame_host,
|
2018-08-24 15:30:37 +00:00
|
|
|
const std::string& channel,
|
2014-04-25 08:13:16 +00:00
|
|
|
const base::ListValue& args,
|
|
|
|
IPC::Message* message) {
|
|
|
|
// webContents.emit(channel, new Event(sender, message), args...);
|
2018-08-24 15:30:37 +00:00
|
|
|
EmitWithSender(channel, frame_host, message, args);
|
2014-04-24 08:45:25 +00:00
|
|
|
}
|
|
|
|
|
2018-08-24 21:14:39 +00:00
|
|
|
void WebContents::OnRendererMessageTo(content::RenderFrameHost* frame_host,
|
2018-10-06 11:48:00 +00:00
|
|
|
bool internal,
|
2018-08-24 21:14:39 +00:00
|
|
|
bool send_to_all,
|
|
|
|
int32_t web_contents_id,
|
2018-08-24 22:30:10 +00:00
|
|
|
const std::string& channel,
|
2018-08-24 21:14:39 +00:00
|
|
|
const base::ListValue& args) {
|
|
|
|
auto* web_contents = mate::TrackableObject<WebContents>::FromWeakMapID(
|
|
|
|
isolate(), web_contents_id);
|
|
|
|
|
|
|
|
if (web_contents) {
|
2018-10-06 11:48:00 +00:00
|
|
|
web_contents->SendIPCMessageWithSender(internal, send_to_all, channel, args,
|
|
|
|
ID());
|
2018-08-24 21:14:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-24 08:45:25 +00:00
|
|
|
// static
|
2018-10-19 08:52:07 +00:00
|
|
|
mate::Handle<WebContents> WebContents::Create(v8::Isolate* isolate,
|
|
|
|
const mate::Dictionary& options) {
|
|
|
|
return mate::CreateHandle(isolate, new WebContents(isolate, options));
|
2016-08-17 00:15:10 +00:00
|
|
|
}
|
|
|
|
|
2018-10-19 08:52:07 +00:00
|
|
|
// static
|
|
|
|
mate::Handle<WebContents> WebContents::CreateAndTake(
|
2018-04-18 01:55:30 +00:00
|
|
|
v8::Isolate* isolate,
|
2018-10-22 18:02:25 +00:00
|
|
|
std::unique_ptr<content::WebContents> web_contents,
|
2018-04-18 01:55:30 +00:00
|
|
|
Type type) {
|
2018-10-22 18:02:25 +00:00
|
|
|
return mate::CreateHandle(
|
|
|
|
isolate, new WebContents(isolate, std::move(web_contents), type));
|
2014-04-24 08:45:25 +00:00
|
|
|
}
|
|
|
|
|
2014-10-23 06:04:13 +00:00
|
|
|
// static
|
2018-10-19 08:52:07 +00:00
|
|
|
mate::Handle<WebContents> WebContents::From(
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
content::WebContents* web_contents) {
|
|
|
|
auto* existing = TrackableObject::FromWrappedClass(isolate, web_contents);
|
|
|
|
if (existing)
|
|
|
|
return mate::CreateHandle(isolate, static_cast<WebContents*>(existing));
|
|
|
|
else
|
|
|
|
return mate::Handle<WebContents>();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
mate::Handle<WebContents> WebContents::FromOrCreate(
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
content::WebContents* web_contents) {
|
|
|
|
auto existing = From(isolate, web_contents);
|
|
|
|
if (!existing.IsEmpty())
|
|
|
|
return existing;
|
|
|
|
else
|
|
|
|
return mate::CreateHandle(isolate, new WebContents(isolate, web_contents));
|
2014-10-23 06:04:13 +00:00
|
|
|
}
|
|
|
|
|
2014-04-24 08:45:25 +00:00
|
|
|
} // namespace api
|
|
|
|
|
|
|
|
} // namespace atom
|
2014-10-23 06:04:13 +00:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2016-08-02 11:38:35 +00:00
|
|
|
using atom::api::WebContents;
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
void Initialize(v8::Local<v8::Object> exports,
|
|
|
|
v8::Local<v8::Value> unused,
|
|
|
|
v8::Local<v8::Context> context,
|
|
|
|
void* priv) {
|
2014-10-23 06:04:13 +00:00
|
|
|
v8::Isolate* isolate = context->GetIsolate();
|
|
|
|
mate::Dictionary dict(isolate, exports);
|
2019-01-09 19:17:05 +00:00
|
|
|
dict.Set("WebContents", WebContents::GetConstructor(isolate)
|
|
|
|
->GetFunction(context)
|
|
|
|
.ToLocalChecked());
|
2016-08-02 11:38:35 +00:00
|
|
|
dict.SetMethod("create", &WebContents::Create);
|
|
|
|
dict.SetMethod("fromId", &mate::TrackableObject<WebContents>::FromWeakMapID);
|
2016-06-07 17:52:03 +00:00
|
|
|
dict.SetMethod("getAllWebContents",
|
2016-08-02 11:38:35 +00:00
|
|
|
&mate::TrackableObject<WebContents>::GetAll);
|
2014-10-23 06:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2018-01-06 15:58:24 +00:00
|
|
|
NODE_BUILTIN_MODULE_CONTEXT_AWARE(atom_browser_web_contents, Initialize)
|