dc72f74020
* chore: bump chromium in DEPS to 5ce64b91b4d6a78c97480059f15ff6469fc0918e * chore: bump chromium in DEPS to e74c73d0000f81b3f40a513176c8d024bba57d28 * chore: bump chromium in DEPS to 501640e650d4657ba63db65fa257e4a899168de7 * chore: bump chromium in DEPS to 00db20e1bc3d77706723a87ada3c1c647a1c37b7 * chore: update patches * refactor: AddNewContents now takes a target_url Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2167732 * chore: SetHostCleanupFinalizationGroupCallback has been removed from V8 * refactor: use WebInputEvent::Namespace types directly Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2160523 * refactor: FollowRedirect takes in cors exempt headers now Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2129787 * refactor: printing::DuplexMode moved to mojo Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2162388 * refactor: use MessagePortDescriptor instead of raw mojo::MessagePipeHandles Refs: https://chromium-review.googlesource.com/c/chromium/src/+/1952124 * chore: update patches * chore: bump chromium in DEPS to f1537676d613f3567cfb43adf577b3847fba4bc3 * chore: update patches * refactor: service_manager::BinderMapWithContext merged into mojo::BinderMap Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2174654 * chore: unused argument removed from ReadAvailableTypes in ui::Clipboard Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2173666 * chore: bump chromium in DEPS to 949888433ab935dd6125c107226a4c9d6da9bf48 * chore: update patches * update patches * chore: update sysroots * chore: bump chromium in DEPS to eaac5b5035fe189b6706e1637122e37134206059 * chore: bump chromium in DEPS to 258b54b903d33dab963adf59016691e6537f8b70 * build: update patches * refactor: cursor.mojom and cursor_types.mojom moved to //ui/base/cursor/mojom Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2172874 * chore: DesktopWindowTreeHostLinux becomes DesktopWindowTreeHostPlatform Refs: * refactor: LogErrorEventDescription moved from ui to x11 Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2164245 * chore: update patches * chore: bump chromium in DEPS to bd06abcfe807d4461683479237cdd920dafa52ca * chore: bump chromium in DEPS to 1afb0891e56f1e79d204db43ca053a46d0974511 * chore: bump chromium in DEPS to 5cb0f794bf7f155bf8c0a241b94e01c9d90c2744 * chore: bump chromium in DEPS to 37327ba3303234e1a3cd3310ca11a68e81b95123 * update patches * remove ClientSideDetectionService from browser_process Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2175320 * refactor: shuttle cursor changed event to WebContentsObserver Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2172779 * chore: bump chromium in DEPS to 1d97904bb6936e106df13705208b73e47367c2b9 * avoid IPC crash introduced earlier in the roll Refs:b67334e781 (diff-607f4e8f7c5686aa09af98c783925babR128)
* remove WebContentsView::SizeContents Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2188931 * chore: bump chromium in DEPS to 87066d1db0546a9de33952ba964e1e42f76f1dae * update patches * adapt to //content creating NetworkContexts https://chromium-review.googlesource.com/c/chromium/src/+/2195196 * WIP temporarily ignore mojo-ified PrintHostMsg_PrintPreviewFailed Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2195343 * add checkout_pgo_profiles to DEPS * chore: bump chromium in DEPS to a095318bec1dedf580d676f408eaeefdbd9956b1 * add blink's execution_context to electron_lib deps Refs:b67334e781
#, https://chromium-review.googlesource.com/c/chromium/src/+/1952124 * update patches * update patches * refactor: use newly mojo-ified PrintPreviewFailed Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2195343 Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> * chore: update patches * chore: remove NOTIFICATION_EXTENSIONS_READY_DEPRECATED Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2199874 * avoid a call to Profile::FromBrowserContext in printing Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2195757 * chore: bump chromium in DEPS to 8f5c6a46861a991e12ffd80554b6bd41a11b0b65 * chore: bump chromium in DEPS to ab66134739ff3b6bdb8de53bbc58a97d1b7b5d8a * chore: bump chromium in DEPS to dd08df9f6c8d6198f0a7b931ca9e845e27dae033 * chore: update patches * [printing] Mojofy PrintHostMsg_PrintPreviewCancelled https://chromium-review.googlesource.com/c/chromium/src/+/2198331 * [printing] Mojofy PrintHostMsg_PrintPreviewInvalidPrinterSettings https://chromium-review.googlesource.com/c/chromium/src/+/2200556 * [printing] Mojofy PrintHostMsg_DidPrepareDocumentForPreview https://chromium-review.googlesource.com/c/chromium/src/+/2201496 * fix: inherit base network context params * fix: use message handle api specific to embedders * update patches * chore: update v8 headers * fixup! avoid a call to Profile::FromBrowserContext in printing * chore: bump chromium in DEPS to f198bc3276828d992b6202d75a9da6a8363164ea * chore: update patches * Remove cors_exempt_headers.h https://chromium-review.googlesource.com/c/chromium/src/+/2203759 * Supply speech recognition sandbox from service_sandbox_type.h https://chromium-review.googlesource.com/c/chromium/src/+/2206918 * [content] Remove unused methods from WebContents https://chromium-review.googlesource.com/c/chromium/src/+/2199113 * [XProto] Replace XAtom with x11::Atom https://chromium-review.googlesource.com/c/chromium/src/+/2202789 * chore: bump chromium in DEPS to 5df37fabaaed09f3ea511f4ef49203d8ebee7b68 * [UI] Support secondary labels in dropdown menus https://chromium-review.googlesource.com/c/chromium/src/+/2208976 * chore: update patches * refactor: GetSublabelAt => GetSecondaryLabelAt * Revert "[printing] Mojofy PrintHostMsg_DidPrepareDocumentForPreview" This reverts commit 44ed0892a0c81716229638f17253bbb2cf8f037d. https://chromium-review.googlesource.com/c/chromium/src/+/2208778 * Unnest CanonicalCookie::CookieInclusionStatus https://chromium-review.googlesource.com/c/chromium/src/+/2203171 * update print-to-pdf.html with latest html data https://bugs.chromium.org/p/chromium/issues/detail?id=1085667 * chore: bump chromium in DEPS to 5dfa8d5f706580fe983e138952553cd6af11ee8b * fix crash in x server when setting window icon * chore: update patches * Use a normal return value for WebContentsView::GetContainerBounds. https://chromium-review.googlesource.com/c/chromium/src/+/2212481 * fix build on windows * chore: bump chromium in DEPS to 9d15054f4cba060901e43deecf74002f11f47be3 * chore: update patches * [printing] Mojofy PrintHostMsg_CheckForCancel https://chromium-review.googlesource.com/c/chromium/src/+/2212181 Co-authored-by: Samuel Attard <samuel.r.attard@gmail.com> Co-authored-by: Electron Bot <anonymous@electronjs.org> Co-authored-by: Andy Locascio <andy@slack-corp.com> Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> Co-authored-by: deepak1556 <hop2deep@gmail.com>
539 lines
18 KiB
C++
539 lines
18 KiB
C++
// Copyright (c) 2013 GitHub, Inc.
|
|
// Use of this source code is governed by the MIT license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "shell/browser/api/electron_api_browser_window.h"
|
|
|
|
#include <memory>
|
|
|
|
#include "base/threading/thread_task_runner_handle.h"
|
|
#include "content/browser/renderer_host/render_widget_host_impl.h" // nogncheck
|
|
#include "content/browser/renderer_host/render_widget_host_owner_delegate.h" // nogncheck
|
|
#include "content/browser/web_contents/web_contents_impl.h" // nogncheck
|
|
#include "content/public/browser/render_process_host.h"
|
|
#include "content/public/browser/render_view_host.h"
|
|
#include "shell/browser/api/electron_api_web_contents_view.h"
|
|
#include "shell/browser/browser.h"
|
|
#include "shell/browser/unresponsive_suppressor.h"
|
|
#include "shell/browser/web_contents_preferences.h"
|
|
#include "shell/browser/window_list.h"
|
|
#include "shell/common/color_util.h"
|
|
#include "shell/common/gin_helper/constructor.h"
|
|
#include "shell/common/gin_helper/dictionary.h"
|
|
#include "shell/common/gin_helper/object_template_builder.h"
|
|
#include "shell/common/node_includes.h"
|
|
#include "shell/common/options_switches.h"
|
|
#include "ui/gl/gpu_switching_manager.h"
|
|
|
|
namespace electron {
|
|
|
|
namespace api {
|
|
|
|
BrowserWindow::BrowserWindow(gin::Arguments* args,
|
|
const gin_helper::Dictionary& options)
|
|
: TopLevelWindow(args->isolate(), options), weak_factory_(this) {
|
|
// Use options.webPreferences in WebContents.
|
|
v8::Isolate* isolate = args->isolate();
|
|
gin_helper::Dictionary web_preferences =
|
|
gin::Dictionary::CreateEmpty(isolate);
|
|
options.Get(options::kWebPreferences, &web_preferences);
|
|
|
|
// Copy the backgroundColor to webContents.
|
|
v8::Local<v8::Value> value;
|
|
if (options.Get(options::kBackgroundColor, &value))
|
|
web_preferences.Set(options::kBackgroundColor, value);
|
|
|
|
// Copy the transparent setting to webContents
|
|
v8::Local<v8::Value> transparent;
|
|
if (options.Get("transparent", &transparent))
|
|
web_preferences.Set("transparent", transparent);
|
|
|
|
// Copy the show setting to webContents, but only if we don't want to paint
|
|
// when initially hidden
|
|
bool paint_when_initially_hidden = true;
|
|
options.Get("paintWhenInitiallyHidden", &paint_when_initially_hidden);
|
|
if (!paint_when_initially_hidden) {
|
|
bool show = true;
|
|
options.Get(options::kShow, &show);
|
|
web_preferences.Set(options::kShow, show);
|
|
}
|
|
|
|
// Copy the webContents option to webPreferences. This is only used internally
|
|
// to implement nativeWindowOpen option.
|
|
if (options.Get("webContents", &value)) {
|
|
web_preferences.SetHidden("webContents", value);
|
|
}
|
|
|
|
// Creates the WebContentsView.
|
|
gin::Handle<WebContentsView> web_contents_view =
|
|
WebContentsView::Create(isolate, web_preferences);
|
|
DCHECK(web_contents_view.get());
|
|
|
|
// Save a reference of the WebContents.
|
|
gin::Handle<WebContents> web_contents =
|
|
web_contents_view->GetWebContents(isolate);
|
|
web_contents_.Reset(isolate, web_contents.ToV8());
|
|
api_web_contents_ = web_contents->GetWeakPtr();
|
|
api_web_contents_->AddObserver(this);
|
|
Observe(api_web_contents_->web_contents());
|
|
|
|
// Keep a copy of the options for later use.
|
|
gin_helper::Dictionary(isolate, web_contents->GetWrapper())
|
|
.Set("browserWindowOptions", options);
|
|
|
|
// Associate with BrowserWindow.
|
|
web_contents->SetOwnerWindow(window());
|
|
|
|
auto* host = web_contents->web_contents()->GetRenderViewHost();
|
|
if (host)
|
|
host->GetWidget()->AddInputEventObserver(this);
|
|
|
|
InitWithArgs(args);
|
|
|
|
// Install the content view after TopLevelWindow's JS code is initialized.
|
|
SetContentView(gin::CreateHandle<View>(isolate, web_contents_view.get()));
|
|
|
|
#if defined(OS_MACOSX)
|
|
OverrideNSWindowContentView(web_contents->managed_web_contents());
|
|
#endif
|
|
|
|
// Init window after everything has been setup.
|
|
window()->InitFromOptions(options);
|
|
}
|
|
|
|
BrowserWindow::~BrowserWindow() {
|
|
// FIXME This is a hack rather than a proper fix preventing shutdown crashes.
|
|
if (api_web_contents_)
|
|
api_web_contents_->RemoveObserver(this);
|
|
// Note that the OnWindowClosed will not be called after the destructor runs,
|
|
// since the window object is managed by the TopLevelWindow class.
|
|
if (web_contents())
|
|
Cleanup();
|
|
}
|
|
|
|
void BrowserWindow::OnInputEvent(const blink::WebInputEvent& event) {
|
|
switch (event.GetType()) {
|
|
case blink::WebInputEvent::Type::kGestureScrollBegin:
|
|
case blink::WebInputEvent::Type::kGestureScrollUpdate:
|
|
case blink::WebInputEvent::Type::kGestureScrollEnd:
|
|
Emit("scroll-touch-edge");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void BrowserWindow::RenderViewHostChanged(content::RenderViewHost* old_host,
|
|
content::RenderViewHost* new_host) {
|
|
if (old_host)
|
|
old_host->GetWidget()->RemoveInputEventObserver(this);
|
|
if (new_host)
|
|
new_host->GetWidget()->AddInputEventObserver(this);
|
|
}
|
|
|
|
void BrowserWindow::RenderViewCreated(
|
|
content::RenderViewHost* render_view_host) {
|
|
if (!window()->transparent())
|
|
return;
|
|
|
|
content::RenderWidgetHostImpl* impl = content::RenderWidgetHostImpl::FromID(
|
|
render_view_host->GetProcess()->GetID(),
|
|
render_view_host->GetRoutingID());
|
|
if (impl)
|
|
impl->owner_delegate()->SetBackgroundOpaque(false);
|
|
}
|
|
|
|
void BrowserWindow::DidFirstVisuallyNonEmptyPaint() {
|
|
if (window()->IsVisible())
|
|
return;
|
|
|
|
// When there is a non-empty first paint, resize the RenderWidget to force
|
|
// Chromium to draw.
|
|
auto* const view = web_contents()->GetRenderWidgetHostView();
|
|
view->Show();
|
|
view->SetSize(window()->GetContentSize());
|
|
|
|
// Emit the ReadyToShow event in next tick in case of pending drawing work.
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::BindOnce(
|
|
[](base::WeakPtr<BrowserWindow> self) {
|
|
if (self)
|
|
self->Emit("ready-to-show");
|
|
},
|
|
GetWeakPtr()));
|
|
}
|
|
|
|
void BrowserWindow::BeforeUnloadDialogCancelled() {
|
|
WindowList::WindowCloseCancelled(window());
|
|
// Cancel unresponsive event when window close is cancelled.
|
|
window_unresponsive_closure_.Cancel();
|
|
}
|
|
|
|
void BrowserWindow::OnRendererUnresponsive(content::RenderProcessHost*) {
|
|
// Schedule the unresponsive shortly later, since we may receive the
|
|
// responsive event soon. This could happen after the whole application had
|
|
// blocked for a while.
|
|
// Also notice that when closing this event would be ignored because we have
|
|
// explicitly started a close timeout counter. This is on purpose because we
|
|
// don't want the unresponsive event to be sent too early when user is closing
|
|
// the window.
|
|
ScheduleUnresponsiveEvent(50);
|
|
}
|
|
|
|
void BrowserWindow::OnCloseContents() {
|
|
// On some machines it may happen that the window gets destroyed for twice,
|
|
// checking web_contents() can effectively guard against that.
|
|
// https://github.com/electron/electron/issues/16202.
|
|
//
|
|
// TODO(zcbenz): We should find out the root cause and improve the closing
|
|
// procedure of BrowserWindow.
|
|
if (!web_contents())
|
|
return;
|
|
|
|
// Close all child windows before closing current window.
|
|
v8::Locker locker(isolate());
|
|
v8::HandleScope handle_scope(isolate());
|
|
for (v8::Local<v8::Value> value : child_windows_.Values(isolate())) {
|
|
gin::Handle<BrowserWindow> child;
|
|
if (gin::ConvertFromV8(isolate(), value, &child) && !child.IsEmpty())
|
|
child->window()->CloseImmediately();
|
|
}
|
|
|
|
// When the web contents is gone, close the window immediately, but the
|
|
// memory will not be freed until you call delete.
|
|
// In this way, it would be safe to manage windows via smart pointers. If you
|
|
// want to free memory when the window is closed, you can do deleting by
|
|
// overriding the OnWindowClosed method in the observer.
|
|
window()->CloseImmediately();
|
|
|
|
// Do not sent "unresponsive" event after window is closed.
|
|
window_unresponsive_closure_.Cancel();
|
|
}
|
|
|
|
void BrowserWindow::OnRendererResponsive() {
|
|
window_unresponsive_closure_.Cancel();
|
|
Emit("responsive");
|
|
}
|
|
|
|
void BrowserWindow::OnDraggableRegionsUpdated(
|
|
const std::vector<mojom::DraggableRegionPtr>& regions) {
|
|
UpdateDraggableRegions(regions);
|
|
}
|
|
|
|
void BrowserWindow::OnSetContentBounds(const gfx::Rect& rect) {
|
|
// window.resizeTo(...)
|
|
// window.moveTo(...)
|
|
window()->SetBounds(rect, false);
|
|
}
|
|
|
|
void BrowserWindow::OnActivateContents() {
|
|
// Hide the auto-hide menu when webContents is focused.
|
|
#if !defined(OS_MACOSX)
|
|
if (IsMenuBarAutoHide() && IsMenuBarVisible())
|
|
window()->SetMenuBarVisibility(false);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::OnPageTitleUpdated(const base::string16& title,
|
|
bool explicit_set) {
|
|
// Change window title to page title.
|
|
auto self = GetWeakPtr();
|
|
if (!Emit("page-title-updated", title, explicit_set)) {
|
|
// |this| might be deleted, or marked as destroyed by close().
|
|
if (self && !IsDestroyed())
|
|
SetTitle(base::UTF16ToUTF8(title));
|
|
}
|
|
}
|
|
|
|
void BrowserWindow::RequestPreferredWidth(int* width) {
|
|
*width = web_contents()->GetPreferredSize().width();
|
|
}
|
|
|
|
void BrowserWindow::OnCloseButtonClicked(bool* prevent_default) {
|
|
// When user tries to close the window by clicking the close button, we do
|
|
// not close the window immediately, instead we try to close the web page
|
|
// first, and when the web page is closed the window will also be closed.
|
|
*prevent_default = true;
|
|
|
|
// Assume the window is not responding if it doesn't cancel the close and is
|
|
// not closed in 5s, in this way we can quickly show the unresponsive
|
|
// dialog when the window is busy executing some script withouth waiting for
|
|
// the unresponsive timeout.
|
|
if (window_unresponsive_closure_.IsCancelled())
|
|
ScheduleUnresponsiveEvent(5000);
|
|
|
|
if (!web_contents())
|
|
// Already closed by renderer
|
|
return;
|
|
|
|
// Required to make beforeunload handler work.
|
|
api_web_contents_->NotifyUserActivation();
|
|
|
|
if (web_contents()->NeedToFireBeforeUnloadOrUnload())
|
|
web_contents()->DispatchBeforeUnload(false /* auto_cancel */);
|
|
else
|
|
web_contents()->Close();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowClosed() {
|
|
Cleanup();
|
|
// See TopLevelWindow::OnWindowClosed on why calling InvalidateWeakPtrs.
|
|
weak_factory_.InvalidateWeakPtrs();
|
|
TopLevelWindow::OnWindowClosed();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowBlur() {
|
|
web_contents()->StoreFocus();
|
|
#if defined(OS_MACOSX)
|
|
auto* rwhv = web_contents()->GetRenderWidgetHostView();
|
|
if (rwhv)
|
|
rwhv->SetActive(false);
|
|
#endif
|
|
|
|
TopLevelWindow::OnWindowBlur();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowFocus() {
|
|
web_contents()->RestoreFocus();
|
|
#if defined(OS_MACOSX)
|
|
auto* rwhv = web_contents()->GetRenderWidgetHostView();
|
|
if (rwhv)
|
|
rwhv->SetActive(true);
|
|
#else
|
|
if (!api_web_contents_->IsDevToolsOpened())
|
|
web_contents()->Focus();
|
|
#endif
|
|
|
|
TopLevelWindow::OnWindowFocus();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowResize() {
|
|
#if defined(OS_MACOSX)
|
|
if (!draggable_regions_.empty())
|
|
UpdateDraggableRegions(draggable_regions_);
|
|
#endif
|
|
TopLevelWindow::OnWindowResize();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowLeaveFullScreen() {
|
|
TopLevelWindow::OnWindowLeaveFullScreen();
|
|
#if defined(OS_MACOSX)
|
|
if (web_contents()->IsFullscreenForCurrentTab())
|
|
web_contents()->ExitFullscreen(true);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::Focus() {
|
|
if (api_web_contents_->IsOffScreen())
|
|
FocusOnWebView();
|
|
else
|
|
TopLevelWindow::Focus();
|
|
}
|
|
|
|
void BrowserWindow::Blur() {
|
|
if (api_web_contents_->IsOffScreen())
|
|
BlurWebView();
|
|
else
|
|
TopLevelWindow::Blur();
|
|
}
|
|
|
|
void BrowserWindow::SetBackgroundColor(const std::string& color_name) {
|
|
TopLevelWindow::SetBackgroundColor(color_name);
|
|
auto* view = web_contents()->GetRenderWidgetHostView();
|
|
if (view)
|
|
view->SetBackgroundColor(ParseHexColor(color_name));
|
|
// Also update the web preferences object otherwise the view will be reset on
|
|
// the next load URL call
|
|
if (api_web_contents_) {
|
|
auto* web_preferences =
|
|
WebContentsPreferences::From(api_web_contents_->web_contents());
|
|
if (web_preferences) {
|
|
web_preferences->preference()->SetStringKey(options::kBackgroundColor,
|
|
color_name);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BrowserWindow::SetBrowserView(v8::Local<v8::Value> value) {
|
|
TopLevelWindow::ResetBrowserViews();
|
|
TopLevelWindow::AddBrowserView(value);
|
|
#if defined(OS_MACOSX)
|
|
UpdateDraggableRegions(draggable_regions_);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::AddBrowserView(v8::Local<v8::Value> value) {
|
|
TopLevelWindow::AddBrowserView(value);
|
|
#if defined(OS_MACOSX)
|
|
UpdateDraggableRegions(draggable_regions_);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::RemoveBrowserView(v8::Local<v8::Value> value) {
|
|
TopLevelWindow::RemoveBrowserView(value);
|
|
#if defined(OS_MACOSX)
|
|
UpdateDraggableRegions(draggable_regions_);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::ResetBrowserViews() {
|
|
TopLevelWindow::ResetBrowserViews();
|
|
#if defined(OS_MACOSX)
|
|
UpdateDraggableRegions(draggable_regions_);
|
|
#endif
|
|
}
|
|
|
|
void BrowserWindow::SetVibrancy(v8::Isolate* isolate,
|
|
v8::Local<v8::Value> value) {
|
|
std::string type = gin::V8ToString(isolate, value);
|
|
|
|
auto* render_view_host = web_contents()->GetRenderViewHost();
|
|
if (render_view_host) {
|
|
auto* impl = content::RenderWidgetHostImpl::FromID(
|
|
render_view_host->GetProcess()->GetID(),
|
|
render_view_host->GetRoutingID());
|
|
if (impl)
|
|
impl->owner_delegate()->SetBackgroundOpaque(
|
|
type.empty() ? !window_->transparent() : false);
|
|
}
|
|
|
|
TopLevelWindow::SetVibrancy(isolate, value);
|
|
}
|
|
|
|
void BrowserWindow::FocusOnWebView() {
|
|
web_contents()->GetRenderViewHost()->GetWidget()->Focus();
|
|
}
|
|
|
|
void BrowserWindow::BlurWebView() {
|
|
web_contents()->GetRenderViewHost()->GetWidget()->Blur();
|
|
}
|
|
|
|
bool BrowserWindow::IsWebViewFocused() {
|
|
auto* host_view = web_contents()->GetRenderViewHost()->GetWidget()->GetView();
|
|
return host_view && host_view->HasFocus();
|
|
}
|
|
|
|
v8::Local<v8::Value> BrowserWindow::GetWebContents(v8::Isolate* isolate) {
|
|
if (web_contents_.IsEmpty())
|
|
return v8::Null(isolate);
|
|
return v8::Local<v8::Value>::New(isolate, web_contents_);
|
|
}
|
|
|
|
// Convert draggable regions in raw format to SkRegion format.
|
|
std::unique_ptr<SkRegion> BrowserWindow::DraggableRegionsToSkRegion(
|
|
const std::vector<mojom::DraggableRegionPtr>& regions) {
|
|
auto sk_region = std::make_unique<SkRegion>();
|
|
for (const auto& region : regions) {
|
|
sk_region->op(
|
|
{region->bounds.x(), region->bounds.y(), region->bounds.right(),
|
|
region->bounds.bottom()},
|
|
region->draggable ? SkRegion::kUnion_Op : SkRegion::kDifference_Op);
|
|
}
|
|
return sk_region;
|
|
}
|
|
|
|
void BrowserWindow::ScheduleUnresponsiveEvent(int ms) {
|
|
if (!window_unresponsive_closure_.IsCancelled())
|
|
return;
|
|
|
|
window_unresponsive_closure_.Reset(base::BindRepeating(
|
|
&BrowserWindow::NotifyWindowUnresponsive, GetWeakPtr()));
|
|
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
|
|
FROM_HERE, window_unresponsive_closure_.callback(),
|
|
base::TimeDelta::FromMilliseconds(ms));
|
|
}
|
|
|
|
void BrowserWindow::NotifyWindowUnresponsive() {
|
|
window_unresponsive_closure_.Cancel();
|
|
if (!window_->IsClosed() && window_->IsEnabled() &&
|
|
!IsUnresponsiveEventSuppressed()) {
|
|
Emit("unresponsive");
|
|
}
|
|
}
|
|
|
|
void BrowserWindow::Cleanup() {
|
|
auto* host = web_contents()->GetRenderViewHost();
|
|
if (host)
|
|
host->GetWidget()->RemoveInputEventObserver(this);
|
|
|
|
// Destroy WebContents asynchronously unless app is shutting down,
|
|
// because destroy() might be called inside WebContents's event handler.
|
|
api_web_contents_->DestroyWebContents(!Browser::Get()->is_shutting_down());
|
|
Observe(nullptr);
|
|
}
|
|
|
|
void BrowserWindow::OnWindowShow() {
|
|
web_contents()->WasShown();
|
|
TopLevelWindow::OnWindowShow();
|
|
}
|
|
|
|
void BrowserWindow::OnWindowHide() {
|
|
web_contents()->WasOccluded();
|
|
TopLevelWindow::OnWindowHide();
|
|
}
|
|
|
|
// static
|
|
gin_helper::WrappableBase* BrowserWindow::New(gin_helper::ErrorThrower thrower,
|
|
gin::Arguments* args) {
|
|
if (!Browser::Get()->is_ready()) {
|
|
thrower.ThrowError("Cannot create BrowserWindow before app is ready");
|
|
return nullptr;
|
|
}
|
|
|
|
if (args->Length() > 1) {
|
|
args->ThrowError();
|
|
return nullptr;
|
|
}
|
|
|
|
gin_helper::Dictionary options;
|
|
if (!(args->Length() == 1 && args->GetNext(&options))) {
|
|
options = gin::Dictionary::CreateEmpty(args->isolate());
|
|
}
|
|
|
|
return new BrowserWindow(args, options);
|
|
}
|
|
|
|
// static
|
|
void BrowserWindow::BuildPrototype(v8::Isolate* isolate,
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
|
prototype->SetClassName(gin::StringToV8(isolate, "BrowserWindow"));
|
|
gin_helper::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
|
|
.SetMethod("focusOnWebView", &BrowserWindow::FocusOnWebView)
|
|
.SetMethod("blurWebView", &BrowserWindow::BlurWebView)
|
|
.SetMethod("isWebViewFocused", &BrowserWindow::IsWebViewFocused)
|
|
.SetProperty("webContents", &BrowserWindow::GetWebContents);
|
|
}
|
|
|
|
// static
|
|
v8::Local<v8::Value> BrowserWindow::From(v8::Isolate* isolate,
|
|
NativeWindow* native_window) {
|
|
auto* existing = TrackableObject::FromWrappedClass(isolate, native_window);
|
|
if (existing)
|
|
return existing->GetWrapper();
|
|
else
|
|
return v8::Null(isolate);
|
|
}
|
|
|
|
} // namespace api
|
|
|
|
} // namespace electron
|
|
|
|
namespace {
|
|
|
|
using electron::api::BrowserWindow;
|
|
using electron::api::TopLevelWindow;
|
|
|
|
void Initialize(v8::Local<v8::Object> exports,
|
|
v8::Local<v8::Value> unused,
|
|
v8::Local<v8::Context> context,
|
|
void* priv) {
|
|
v8::Isolate* isolate = context->GetIsolate();
|
|
gin_helper::Dictionary dict(isolate, exports);
|
|
dict.Set("BrowserWindow",
|
|
gin_helper::CreateConstructor<BrowserWindow>(
|
|
isolate, base::BindRepeating(&BrowserWindow::New)));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
NODE_LINKED_MODULE_CONTEXT_AWARE(electron_browser_window, Initialize)
|