28ada6ea8b
* chore: bump chromium in DEPS to 99.0.4819.0
* chore: update patches
* chore: bump chromium in DEPS to 99.0.4824.0
* chore: update patches
* chore: bump chromium in DEPS to 99.0.4827.0
* chore: update patches
* 3352511: PiP: Add inkdrop and pointer cursor to PiP window buttons
https://chromium-review.googlesource.com/c/chromium/src/+/3352511
* 3309164: webhid: Show FIDO devices in the chooser if allowed
https://chromium-review.googlesource.com/c/chromium/src/+/3309164
* 3297868: hid: Add experimental HIDDevice.forget()
https://chromium-review.googlesource.com/c/chromium/src/+/3297868
* 3362491: [Extensions] Move i18n API to //extensions
https://chromium-review.googlesource.com/c/chromium/src/+/3362491
* MCC Refactor step0: Allow embedders to register associated_interface binders with RenderFrameHostImpl::associated_registry_.
https://chromium-review.googlesource.com/c/chromium/src/+/3281481
* 3352616: [Gtk] Remove libgtk from the link-line
https://chromium-review.googlesource.com/c/chromium/src/+/3352616
* 3249211: Clear-Site-Data support for partitioned cookies
https://chromium-review.googlesource.com/c/chromium/src/+/3249211
* [Extensions][COIL] Use [allow|block]list in //extensions/common
https://chromium-review.googlesource.com/c/chromium/src/+/3372668
* Begin ScopedUserPrefUpdate migration to modern base::Value
https://chromium-review.googlesource.com/c/chromium/src/+/3376154
* [Code Health] Refactor PrefService GetDict + GetList to use base::Value
https://chromium-review.googlesource.com/c/chromium/src/+/3343526
* 3354997: [CodeHealth] Remove deprecated SetDictionary method
https://chromium-review.googlesource.com/c/chromium/src/+/3354997
* 3287323: Add LacrosPrefStore for lacros settings
https://chromium-review.googlesource.com/c/chromium/src/+/3287323
* 3365916: [PA] Clean up remaining lazy commit code
https://chromium-review.googlesource.com/c/chromium/src/+/3365916
* [MPArch] Target the external protocol error at the responsible frame.
https://chromium-review.googlesource.com/c/chromium/src/+/3011560
* Pass origin to RegisterNonNetworkSubresourceURLLoaderFactories
https://chromium-review.googlesource.com/c/chromium/src/+/3350608
* Linux: Send OSCrypt raw encryption key to the Network Service
https://chromium-review.googlesource.com/c/chromium/src/+/3320484
* [PlzServiceWorker] Remove remaining references to PlzServiceWorker.
https://chromium-review.googlesource.com/c/chromium/src/+/3359441
* chore: fixup for lint
* 3327621: Fix tablet mode detection for Win 11.
https://chromium-review.googlesource.com/c/chromium/src/+/3327621
* 3342428: ax_mac: move AXTextMarker conversion utils under ui umbrella
https://chromium-review.googlesource.com/c/chromium/src/+/3342428
* 3353974: Mac: Use base::Feature for overlay features
https://chromium-review.googlesource.com/c/chromium/src/+/3353974
* chore: bump chromium in DEPS to 99.0.4828.0
* chore: update patches
* chore: bump chromium in DEPS to 99.0.4837.0
* chore: update patches
* chore: update patches
* 3379142: Drop FALLTHROUGH macro
Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3379142
* 3381749: C++17: Allow use of std::map::try_emplace and std::map::insert_or_assign
Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3381749
* chore: bump chromium in DEPS to 99.0.4839.0
* chore: update patches
* chore: bump chromium in DEPS to 99.0.4840.0
* chore: bump chromium in DEPS to 99.0.4844.0
* 3395881: [api] Deprecate Local<v8::Context> v8::Object::CreationContext()
Ref: https://chromium-review.googlesource.com/c/v8/v8/+/3395881
* chore: update patches
* chore: bump chromium in DEPS to 100.0.4845.0
* chore: update patches
* chore: bump chromium in DEPS to 100.0.4847.0
* chore: update patches
* chore: bump chromium in DEPS to 100.0.4849.0
* chore: update patches
* chore: bump chromium in DEPS to 100.0.4851.0
* chore: bump chromium in DEPS to 100.0.4853.0
* update patches
* chore: update patches
* update patches
* 3383599: Fonts Access: Remove prototype that uses a font picker.
https://chromium-review.googlesource.com/c/chromium/src/+/3383599
* 3404768: Remove ALLOW_UNUSED macros
https://chromium-review.googlesource.com/c/chromium/src/+/3404768
* 3374762: Remove ignore_result.h
https://chromium-review.googlesource.com/c/chromium/src/+/3374762
* 3399305: [unseasoned-pdf] Apply proper frame offsets for touch selections
https://chromium-review.googlesource.com/c/chromium/src/+/3399305
* 3402210: [Extensions] Don't trigger unload event for already unloaded extension
https://chromium-review.googlesource.com/c/chromium/src/+/3402210
* 3410912: Combine URLLoaderClient OnReceiveResponse and OnStartLoadingResponseBody.
https://chromium-review.googlesource.com/c/chromium/src/+/3410912
* 3370428: Make the AuthSchemes policy support dynamic refresh
https://chromium-review.googlesource.com/c/chromium/src/+/3370428
* 3407603: Finish ScopedUserPrefUpdate migration to modern base::Value
https://chromium-review.googlesource.com/c/chromium/src/+/3407603
* 3378352: ozone/x11: move code from //ui/p/x11 to //ui/ozone/p/x11
https://chromium-review.googlesource.com/c/chromium/src/+/3378352
* 3370810: Delete chrome/service, AKA the Cloud Print service process.
https://chromium-review.googlesource.com/c/chromium/src/+/3370810
* chore: bump chromium in DEPS to 100.0.4855.0
* chore: update patches
* fixup! 3370810: Delete chrome/service, AKA the Cloud Print service process.
* revert 3348007 to fix windows build
* 3318572: [Code health] Fix gn check errors in //extensions/browser:*
https://chromium-review.googlesource.com/c/chromium/src/+/3318572
* fix printing.patch
* fix iwyu issue
* 3408515: win: Make ShorcutOperation an enum class and modernize names
https://chromium-review.googlesource.com/c/chromium/src/+/3408515
* 3388333: [UIA] Remove dead code accessibility_misc_utils.h/cc
https://chromium-review.googlesource.com/c/chromium/src/+/3388333
* fix windows build? i hope
* patch gn visibility of //ui/ozone/platform/x11
* missing include base/logging.h
* use BUILDFLAG for USE_NSS_CERTS
https://chromium-review.googlesource.com/c/chromium/src/+/3379123
* defined(OS_*) ==> BUILDFLAG(IS_*)
https://bugs.chromium.org/p/chromium/issues/detail?id=1234043
* fixup! 3404768: Remove ALLOW_UNUSED macros
* another attempt to fix windows build
* temporarily disable the custom scheme service worker test
https://github.com/electron/electron/issues/32664
* fix loading mv3 extensions
not sure what cl broke this unfort.
* fixup! 3404768: Remove ALLOW_UNUSED macros
* patch nan
https://chromium-review.googlesource.com/c/v8/v8/+/3395880
* fix node test
* fix nullptr in FindPdfFrame
* patch perfetto to fix build issue on win-ia32
bc44c3c753
* fix build for linux-x64-testing-no-run-as-node
* fix patch
* skip <webview>.capturePage() test
https://github.com/electron/electron/issues/32705
* test: fix failing tests of focus/blur events of WebContents (#32711)
* inherit stdio from app module test child processes
this prevents them from timing out due to full stdout buffers
* test to see if we can get better logs on windows ci
* try again for appveyor log things
* skip contentTracing tests on ia32
* ci: disable gpu compositing
* drop applied patch
* fix merge fail
* Revert "ci: disable gpu compositing"
This reverts commit 0344129fcb19ea3e87e06c1110d751f22eba3fec.
Co-authored-by: electron-roller[bot] <84116207+electron-roller[bot]@users.noreply.github.com>
Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org>
Co-authored-by: PatchUp <73610968+patchup[bot]@users.noreply.github.com>
Co-authored-by: John Kleinschmidt <jkleinsc@github.com>
Co-authored-by: VerteDinde <khammond@slack-corp.com>
Co-authored-by: VerteDinde <vertedinde@electronjs.org>
Co-authored-by: Jeremy Rose <jeremya@chromium.org>
Co-authored-by: Jeremy Rose <nornagon@nornagon.net>
Co-authored-by: Cheng Zhao <zcbenz@gmail.com>
Co-authored-by: deepak1556 <hop2deep@gmail.com>
1356 lines
41 KiB
C++
1356 lines
41 KiB
C++
// Copyright (c) 2018 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_base_window.h"
|
|
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "electron/buildflags/buildflags.h"
|
|
#include "gin/dictionary.h"
|
|
#include "shell/browser/api/electron_api_browser_view.h"
|
|
#include "shell/browser/api/electron_api_menu.h"
|
|
#include "shell/browser/api/electron_api_view.h"
|
|
#include "shell/browser/api/electron_api_web_contents.h"
|
|
#include "shell/browser/javascript_environment.h"
|
|
#include "shell/common/color_util.h"
|
|
#include "shell/common/gin_converters/callback_converter.h"
|
|
#include "shell/common/gin_converters/file_path_converter.h"
|
|
#include "shell/common/gin_converters/gfx_converter.h"
|
|
#include "shell/common/gin_converters/image_converter.h"
|
|
#include "shell/common/gin_converters/native_window_converter.h"
|
|
#include "shell/common/gin_converters/value_converter.h"
|
|
#include "shell/common/gin_helper/dictionary.h"
|
|
#include "shell/common/gin_helper/object_template_builder.h"
|
|
#include "shell/common/gin_helper/persistent_dictionary.h"
|
|
#include "shell/common/node_includes.h"
|
|
#include "shell/common/options_switches.h"
|
|
|
|
#if defined(TOOLKIT_VIEWS)
|
|
#include "shell/browser/native_window_views.h"
|
|
#endif
|
|
|
|
#if BUILDFLAG(IS_WIN)
|
|
#include "shell/browser/ui/win/taskbar_host.h"
|
|
#include "ui/base/win/shell.h"
|
|
#endif
|
|
|
|
#if BUILDFLAG(IS_WIN)
|
|
namespace gin {
|
|
|
|
template <>
|
|
struct Converter<electron::TaskbarHost::ThumbarButton> {
|
|
static bool FromV8(v8::Isolate* isolate,
|
|
v8::Handle<v8::Value> val,
|
|
electron::TaskbarHost::ThumbarButton* out) {
|
|
gin::Dictionary dict(isolate);
|
|
if (!gin::ConvertFromV8(isolate, val, &dict))
|
|
return false;
|
|
dict.Get("click", &(out->clicked_callback));
|
|
dict.Get("tooltip", &(out->tooltip));
|
|
dict.Get("flags", &out->flags);
|
|
return dict.Get("icon", &(out->icon));
|
|
}
|
|
};
|
|
|
|
} // namespace gin
|
|
#endif
|
|
|
|
namespace electron {
|
|
|
|
namespace api {
|
|
|
|
namespace {
|
|
|
|
// Converts binary data to Buffer.
|
|
v8::Local<v8::Value> ToBuffer(v8::Isolate* isolate, void* val, int size) {
|
|
auto buffer = node::Buffer::Copy(isolate, static_cast<char*>(val), size);
|
|
if (buffer.IsEmpty())
|
|
return v8::Null(isolate);
|
|
else
|
|
return buffer.ToLocalChecked();
|
|
}
|
|
|
|
} // namespace
|
|
|
|
BaseWindow::BaseWindow(v8::Isolate* isolate,
|
|
const gin_helper::Dictionary& options) {
|
|
// The parent window.
|
|
gin::Handle<BaseWindow> parent;
|
|
if (options.Get("parent", &parent) && !parent.IsEmpty())
|
|
parent_window_.Reset(isolate, parent.ToV8());
|
|
|
|
#if BUILDFLAG(ENABLE_OSR)
|
|
// Offscreen windows are always created frameless.
|
|
gin_helper::Dictionary web_preferences;
|
|
bool offscreen;
|
|
if (options.Get(options::kWebPreferences, &web_preferences) &&
|
|
web_preferences.Get(options::kOffscreen, &offscreen) && offscreen) {
|
|
const_cast<gin_helper::Dictionary&>(options).Set(options::kFrame, false);
|
|
}
|
|
#endif
|
|
|
|
// Creates NativeWindow.
|
|
window_.reset(NativeWindow::Create(
|
|
options, parent.IsEmpty() ? nullptr : parent->window_.get()));
|
|
window_->AddObserver(this);
|
|
|
|
#if defined(TOOLKIT_VIEWS)
|
|
v8::Local<v8::Value> icon;
|
|
if (options.Get(options::kIcon, &icon)) {
|
|
SetIconImpl(isolate, icon, NativeImage::OnConvertError::kWarn);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
BaseWindow::BaseWindow(gin_helper::Arguments* args,
|
|
const gin_helper::Dictionary& options)
|
|
: BaseWindow(args->isolate(), options) {
|
|
InitWithArgs(args);
|
|
// Init window after everything has been setup.
|
|
window()->InitFromOptions(options);
|
|
}
|
|
|
|
BaseWindow::~BaseWindow() {
|
|
CloseImmediately();
|
|
|
|
// Destroy the native window in next tick because the native code might be
|
|
// iterating all windows.
|
|
base::ThreadTaskRunnerHandle::Get()->DeleteSoon(FROM_HERE, window_.release());
|
|
|
|
// Remove global reference so the JS object can be garbage collected.
|
|
self_ref_.Reset();
|
|
}
|
|
|
|
void BaseWindow::InitWith(v8::Isolate* isolate, v8::Local<v8::Object> wrapper) {
|
|
AttachAsUserData(window_.get());
|
|
gin_helper::TrackableObject<BaseWindow>::InitWith(isolate, wrapper);
|
|
|
|
// We can only append this window to parent window's child windows after this
|
|
// window's JS wrapper gets initialized.
|
|
if (!parent_window_.IsEmpty()) {
|
|
gin::Handle<BaseWindow> parent;
|
|
gin::ConvertFromV8(isolate, GetParentWindow(), &parent);
|
|
DCHECK(!parent.IsEmpty());
|
|
parent->child_windows_.Set(isolate, weak_map_id(), wrapper);
|
|
}
|
|
|
|
// Reference this object in case it got garbage collected.
|
|
self_ref_.Reset(isolate, wrapper);
|
|
}
|
|
|
|
void BaseWindow::WillCloseWindow(bool* prevent_default) {
|
|
if (Emit("close")) {
|
|
*prevent_default = true;
|
|
}
|
|
}
|
|
|
|
void BaseWindow::OnWindowClosed() {
|
|
// Invalidate weak ptrs before the Javascript object is destroyed,
|
|
// there might be some delayed emit events which shouldn't be
|
|
// triggered after this.
|
|
weak_factory_.InvalidateWeakPtrs();
|
|
|
|
RemoveFromWeakMap();
|
|
window_->RemoveObserver(this);
|
|
|
|
// 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("closed");
|
|
|
|
RemoveFromParentChildWindows();
|
|
BaseWindow::ResetBrowserViews();
|
|
|
|
// Destroy the native class when window is closed.
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, GetDestroyClosure());
|
|
}
|
|
|
|
void BaseWindow::OnWindowEndSession() {
|
|
Emit("session-end");
|
|
}
|
|
|
|
void BaseWindow::OnWindowBlur() {
|
|
EmitEventSoon("blur");
|
|
}
|
|
|
|
void BaseWindow::OnWindowFocus() {
|
|
EmitEventSoon("focus");
|
|
}
|
|
|
|
void BaseWindow::OnWindowShow() {
|
|
Emit("show");
|
|
}
|
|
|
|
void BaseWindow::OnWindowHide() {
|
|
Emit("hide");
|
|
}
|
|
|
|
void BaseWindow::OnWindowMaximize() {
|
|
Emit("maximize");
|
|
}
|
|
|
|
void BaseWindow::OnWindowUnmaximize() {
|
|
Emit("unmaximize");
|
|
}
|
|
|
|
void BaseWindow::OnWindowMinimize() {
|
|
Emit("minimize");
|
|
}
|
|
|
|
void BaseWindow::OnWindowRestore() {
|
|
Emit("restore");
|
|
}
|
|
|
|
void BaseWindow::OnWindowWillResize(const gfx::Rect& new_bounds,
|
|
const gfx::ResizeEdge& edge,
|
|
bool* prevent_default) {
|
|
v8::Isolate* isolate = JavascriptEnvironment::GetIsolate();
|
|
v8::Locker locker(isolate);
|
|
v8::HandleScope handle_scope(isolate);
|
|
gin_helper::Dictionary info = gin::Dictionary::CreateEmpty(isolate);
|
|
info.Set("edge", edge);
|
|
|
|
if (Emit("will-resize", new_bounds, info)) {
|
|
*prevent_default = true;
|
|
}
|
|
}
|
|
|
|
void BaseWindow::OnWindowResize() {
|
|
Emit("resize");
|
|
}
|
|
|
|
void BaseWindow::OnWindowResized() {
|
|
Emit("resized");
|
|
}
|
|
|
|
void BaseWindow::OnWindowWillMove(const gfx::Rect& new_bounds,
|
|
bool* prevent_default) {
|
|
if (Emit("will-move", new_bounds)) {
|
|
*prevent_default = true;
|
|
}
|
|
}
|
|
|
|
void BaseWindow::OnWindowMove() {
|
|
Emit("move");
|
|
}
|
|
|
|
void BaseWindow::OnWindowMoved() {
|
|
Emit("moved");
|
|
}
|
|
|
|
void BaseWindow::OnWindowEnterFullScreen() {
|
|
Emit("enter-full-screen");
|
|
}
|
|
|
|
void BaseWindow::OnWindowLeaveFullScreen() {
|
|
Emit("leave-full-screen");
|
|
}
|
|
|
|
void BaseWindow::OnWindowScrollTouchBegin() {
|
|
Emit("scroll-touch-begin");
|
|
}
|
|
|
|
void BaseWindow::OnWindowScrollTouchEnd() {
|
|
Emit("scroll-touch-end");
|
|
}
|
|
|
|
void BaseWindow::OnWindowSwipe(const std::string& direction) {
|
|
Emit("swipe", direction);
|
|
}
|
|
|
|
void BaseWindow::OnWindowRotateGesture(float rotation) {
|
|
Emit("rotate-gesture", rotation);
|
|
}
|
|
|
|
void BaseWindow::OnWindowSheetBegin() {
|
|
Emit("sheet-begin");
|
|
}
|
|
|
|
void BaseWindow::OnWindowSheetEnd() {
|
|
Emit("sheet-end");
|
|
}
|
|
|
|
void BaseWindow::OnWindowEnterHtmlFullScreen() {
|
|
Emit("enter-html-full-screen");
|
|
}
|
|
|
|
void BaseWindow::OnWindowLeaveHtmlFullScreen() {
|
|
Emit("leave-html-full-screen");
|
|
}
|
|
|
|
void BaseWindow::OnWindowAlwaysOnTopChanged() {
|
|
Emit("always-on-top-changed", IsAlwaysOnTop());
|
|
}
|
|
|
|
void BaseWindow::OnExecuteAppCommand(const std::string& command_name) {
|
|
Emit("app-command", command_name);
|
|
}
|
|
|
|
void BaseWindow::OnTouchBarItemResult(const std::string& item_id,
|
|
const base::DictionaryValue& details) {
|
|
Emit("-touch-bar-interaction", item_id, details);
|
|
}
|
|
|
|
void BaseWindow::OnNewWindowForTab() {
|
|
Emit("new-window-for-tab");
|
|
}
|
|
|
|
void BaseWindow::OnSystemContextMenu(int x, int y, bool* prevent_default) {
|
|
if (Emit("system-context-menu", gfx::Point(x, y))) {
|
|
*prevent_default = true;
|
|
}
|
|
}
|
|
|
|
#if BUILDFLAG(IS_WIN)
|
|
void BaseWindow::OnWindowMessage(UINT message, WPARAM w_param, LPARAM l_param) {
|
|
if (IsWindowMessageHooked(message)) {
|
|
v8::Isolate* isolate = JavascriptEnvironment::GetIsolate();
|
|
v8::Locker locker(isolate);
|
|
v8::HandleScope scope(isolate);
|
|
messages_callback_map_[message].Run(
|
|
ToBuffer(isolate, static_cast<void*>(&w_param), sizeof(WPARAM)),
|
|
ToBuffer(isolate, static_cast<void*>(&l_param), sizeof(LPARAM)));
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void BaseWindow::SetContentView(gin::Handle<View> view) {
|
|
ResetBrowserViews();
|
|
content_view_.Reset(isolate(), view.ToV8());
|
|
window_->SetContentView(view->view());
|
|
}
|
|
|
|
void BaseWindow::CloseImmediately() {
|
|
if (!window_->IsClosed())
|
|
window_->CloseImmediately();
|
|
}
|
|
|
|
void BaseWindow::Close() {
|
|
window_->Close();
|
|
}
|
|
|
|
void BaseWindow::Focus() {
|
|
window_->Focus(true);
|
|
}
|
|
|
|
void BaseWindow::Blur() {
|
|
window_->Focus(false);
|
|
}
|
|
|
|
bool BaseWindow::IsFocused() {
|
|
return window_->IsFocused();
|
|
}
|
|
|
|
void BaseWindow::Show() {
|
|
window_->Show();
|
|
}
|
|
|
|
void BaseWindow::ShowInactive() {
|
|
// This method doesn't make sense for modal window.
|
|
if (IsModal())
|
|
return;
|
|
window_->ShowInactive();
|
|
}
|
|
|
|
void BaseWindow::Hide() {
|
|
window_->Hide();
|
|
}
|
|
|
|
bool BaseWindow::IsVisible() {
|
|
return window_->IsVisible();
|
|
}
|
|
|
|
bool BaseWindow::IsEnabled() {
|
|
return window_->IsEnabled();
|
|
}
|
|
|
|
void BaseWindow::SetEnabled(bool enable) {
|
|
window_->SetEnabled(enable);
|
|
}
|
|
|
|
void BaseWindow::Maximize() {
|
|
window_->Maximize();
|
|
}
|
|
|
|
void BaseWindow::Unmaximize() {
|
|
window_->Unmaximize();
|
|
}
|
|
|
|
bool BaseWindow::IsMaximized() {
|
|
return window_->IsMaximized();
|
|
}
|
|
|
|
void BaseWindow::Minimize() {
|
|
window_->Minimize();
|
|
}
|
|
|
|
void BaseWindow::Restore() {
|
|
window_->Restore();
|
|
}
|
|
|
|
bool BaseWindow::IsMinimized() {
|
|
return window_->IsMinimized();
|
|
}
|
|
|
|
void BaseWindow::SetFullScreen(bool fullscreen) {
|
|
window_->SetFullScreen(fullscreen);
|
|
}
|
|
|
|
bool BaseWindow::IsFullscreen() {
|
|
return window_->IsFullscreen();
|
|
}
|
|
|
|
void BaseWindow::SetBounds(const gfx::Rect& bounds,
|
|
gin_helper::Arguments* args) {
|
|
bool animate = false;
|
|
args->GetNext(&animate);
|
|
window_->SetBounds(bounds, animate);
|
|
}
|
|
|
|
gfx::Rect BaseWindow::GetBounds() {
|
|
return window_->GetBounds();
|
|
}
|
|
|
|
bool BaseWindow::IsNormal() {
|
|
return window_->IsNormal();
|
|
}
|
|
|
|
gfx::Rect BaseWindow::GetNormalBounds() {
|
|
return window_->GetNormalBounds();
|
|
}
|
|
|
|
void BaseWindow::SetContentBounds(const gfx::Rect& bounds,
|
|
gin_helper::Arguments* args) {
|
|
bool animate = false;
|
|
args->GetNext(&animate);
|
|
window_->SetContentBounds(bounds, animate);
|
|
}
|
|
|
|
gfx::Rect BaseWindow::GetContentBounds() {
|
|
return window_->GetContentBounds();
|
|
}
|
|
|
|
void BaseWindow::SetSize(int width, int height, gin_helper::Arguments* args) {
|
|
bool animate = false;
|
|
gfx::Size size = window_->GetMinimumSize();
|
|
size.SetToMax(gfx::Size(width, height));
|
|
args->GetNext(&animate);
|
|
window_->SetSize(size, animate);
|
|
}
|
|
|
|
std::vector<int> BaseWindow::GetSize() {
|
|
std::vector<int> result(2);
|
|
gfx::Size size = window_->GetSize();
|
|
result[0] = size.width();
|
|
result[1] = size.height();
|
|
return result;
|
|
}
|
|
|
|
void BaseWindow::SetContentSize(int width,
|
|
int height,
|
|
gin_helper::Arguments* args) {
|
|
bool animate = false;
|
|
args->GetNext(&animate);
|
|
window_->SetContentSize(gfx::Size(width, height), animate);
|
|
}
|
|
|
|
std::vector<int> BaseWindow::GetContentSize() {
|
|
std::vector<int> result(2);
|
|
gfx::Size size = window_->GetContentSize();
|
|
result[0] = size.width();
|
|
result[1] = size.height();
|
|
return result;
|
|
}
|
|
|
|
void BaseWindow::SetMinimumSize(int width, int height) {
|
|
window_->SetMinimumSize(gfx::Size(width, height));
|
|
}
|
|
|
|
std::vector<int> BaseWindow::GetMinimumSize() {
|
|
std::vector<int> result(2);
|
|
gfx::Size size = window_->GetMinimumSize();
|
|
result[0] = size.width();
|
|
result[1] = size.height();
|
|
return result;
|
|
}
|
|
|
|
void BaseWindow::SetMaximumSize(int width, int height) {
|
|
window_->SetMaximumSize(gfx::Size(width, height));
|
|
}
|
|
|
|
std::vector<int> BaseWindow::GetMaximumSize() {
|
|
std::vector<int> result(2);
|
|
gfx::Size size = window_->GetMaximumSize();
|
|
result[0] = size.width();
|
|
result[1] = size.height();
|
|
return result;
|
|
}
|
|
|
|
void BaseWindow::SetSheetOffset(double offsetY, gin_helper::Arguments* args) {
|
|
double offsetX = 0.0;
|
|
args->GetNext(&offsetX);
|
|
window_->SetSheetOffset(offsetX, offsetY);
|
|
}
|
|
|
|
void BaseWindow::SetResizable(bool resizable) {
|
|
window_->SetResizable(resizable);
|
|
}
|
|
|
|
bool BaseWindow::IsResizable() {
|
|
return window_->IsResizable();
|
|
}
|
|
|
|
void BaseWindow::SetMovable(bool movable) {
|
|
window_->SetMovable(movable);
|
|
}
|
|
|
|
bool BaseWindow::IsMovable() {
|
|
return window_->IsMovable();
|
|
}
|
|
|
|
void BaseWindow::SetMinimizable(bool minimizable) {
|
|
window_->SetMinimizable(minimizable);
|
|
}
|
|
|
|
bool BaseWindow::IsMinimizable() {
|
|
return window_->IsMinimizable();
|
|
}
|
|
|
|
void BaseWindow::SetMaximizable(bool maximizable) {
|
|
window_->SetMaximizable(maximizable);
|
|
}
|
|
|
|
bool BaseWindow::IsMaximizable() {
|
|
return window_->IsMaximizable();
|
|
}
|
|
|
|
void BaseWindow::SetFullScreenable(bool fullscreenable) {
|
|
window_->SetFullScreenable(fullscreenable);
|
|
}
|
|
|
|
bool BaseWindow::IsFullScreenable() {
|
|
return window_->IsFullScreenable();
|
|
}
|
|
|
|
void BaseWindow::SetClosable(bool closable) {
|
|
window_->SetClosable(closable);
|
|
}
|
|
|
|
bool BaseWindow::IsClosable() {
|
|
return window_->IsClosable();
|
|
}
|
|
|
|
void BaseWindow::SetAlwaysOnTop(bool top, gin_helper::Arguments* args) {
|
|
std::string level = "floating";
|
|
int relative_level = 0;
|
|
args->GetNext(&level);
|
|
args->GetNext(&relative_level);
|
|
|
|
ui::ZOrderLevel z_order =
|
|
top ? ui::ZOrderLevel::kFloatingWindow : ui::ZOrderLevel::kNormal;
|
|
window_->SetAlwaysOnTop(z_order, level, relative_level);
|
|
}
|
|
|
|
bool BaseWindow::IsAlwaysOnTop() {
|
|
return window_->GetZOrderLevel() != ui::ZOrderLevel::kNormal;
|
|
}
|
|
|
|
void BaseWindow::Center() {
|
|
window_->Center();
|
|
}
|
|
|
|
void BaseWindow::SetPosition(int x, int y, gin_helper::Arguments* args) {
|
|
bool animate = false;
|
|
args->GetNext(&animate);
|
|
window_->SetPosition(gfx::Point(x, y), animate);
|
|
}
|
|
|
|
std::vector<int> BaseWindow::GetPosition() {
|
|
std::vector<int> result(2);
|
|
gfx::Point pos = window_->GetPosition();
|
|
result[0] = pos.x();
|
|
result[1] = pos.y();
|
|
return result;
|
|
}
|
|
void BaseWindow::MoveAbove(const std::string& sourceId,
|
|
gin_helper::Arguments* args) {
|
|
#if BUILDFLAG(ENABLE_DESKTOP_CAPTURER)
|
|
if (!window_->MoveAbove(sourceId))
|
|
args->ThrowError("Invalid media source id");
|
|
#else
|
|
args->ThrowError("enable_desktop_capturer=true to use this feature");
|
|
#endif
|
|
}
|
|
|
|
void BaseWindow::MoveTop() {
|
|
window_->MoveTop();
|
|
}
|
|
|
|
void BaseWindow::SetTitle(const std::string& title) {
|
|
window_->SetTitle(title);
|
|
}
|
|
|
|
std::string BaseWindow::GetTitle() {
|
|
return window_->GetTitle();
|
|
}
|
|
|
|
void BaseWindow::SetAccessibleTitle(const std::string& title) {
|
|
window_->SetAccessibleTitle(title);
|
|
}
|
|
|
|
std::string BaseWindow::GetAccessibleTitle() {
|
|
return window_->GetAccessibleTitle();
|
|
}
|
|
|
|
void BaseWindow::FlashFrame(bool flash) {
|
|
window_->FlashFrame(flash);
|
|
}
|
|
|
|
void BaseWindow::SetSkipTaskbar(bool skip) {
|
|
window_->SetSkipTaskbar(skip);
|
|
}
|
|
|
|
void BaseWindow::SetExcludedFromShownWindowsMenu(bool excluded) {
|
|
window_->SetExcludedFromShownWindowsMenu(excluded);
|
|
}
|
|
|
|
bool BaseWindow::IsExcludedFromShownWindowsMenu() {
|
|
return window_->IsExcludedFromShownWindowsMenu();
|
|
}
|
|
|
|
void BaseWindow::SetSimpleFullScreen(bool simple_fullscreen) {
|
|
window_->SetSimpleFullScreen(simple_fullscreen);
|
|
}
|
|
|
|
bool BaseWindow::IsSimpleFullScreen() {
|
|
return window_->IsSimpleFullScreen();
|
|
}
|
|
|
|
void BaseWindow::SetKiosk(bool kiosk) {
|
|
window_->SetKiosk(kiosk);
|
|
}
|
|
|
|
bool BaseWindow::IsKiosk() {
|
|
return window_->IsKiosk();
|
|
}
|
|
|
|
bool BaseWindow::IsTabletMode() const {
|
|
return window_->IsTabletMode();
|
|
}
|
|
|
|
void BaseWindow::SetBackgroundColor(const std::string& color_name) {
|
|
SkColor color = ParseHexColor(color_name);
|
|
window_->SetBackgroundColor(color);
|
|
}
|
|
|
|
std::string BaseWindow::GetBackgroundColor() {
|
|
return ToRGBHex(window_->GetBackgroundColor());
|
|
}
|
|
|
|
void BaseWindow::SetHasShadow(bool has_shadow) {
|
|
window_->SetHasShadow(has_shadow);
|
|
}
|
|
|
|
bool BaseWindow::HasShadow() {
|
|
return window_->HasShadow();
|
|
}
|
|
|
|
void BaseWindow::SetOpacity(const double opacity) {
|
|
window_->SetOpacity(opacity);
|
|
}
|
|
|
|
double BaseWindow::GetOpacity() {
|
|
return window_->GetOpacity();
|
|
}
|
|
|
|
void BaseWindow::SetShape(const std::vector<gfx::Rect>& rects) {
|
|
window_->widget()->SetShape(std::make_unique<std::vector<gfx::Rect>>(rects));
|
|
}
|
|
|
|
void BaseWindow::SetRepresentedFilename(const std::string& filename) {
|
|
window_->SetRepresentedFilename(filename);
|
|
}
|
|
|
|
std::string BaseWindow::GetRepresentedFilename() {
|
|
return window_->GetRepresentedFilename();
|
|
}
|
|
|
|
void BaseWindow::SetDocumentEdited(bool edited) {
|
|
window_->SetDocumentEdited(edited);
|
|
}
|
|
|
|
bool BaseWindow::IsDocumentEdited() {
|
|
return window_->IsDocumentEdited();
|
|
}
|
|
|
|
void BaseWindow::SetIgnoreMouseEvents(bool ignore,
|
|
gin_helper::Arguments* args) {
|
|
gin_helper::Dictionary options;
|
|
bool forward = false;
|
|
args->GetNext(&options) && options.Get("forward", &forward);
|
|
return window_->SetIgnoreMouseEvents(ignore, forward);
|
|
}
|
|
|
|
void BaseWindow::SetContentProtection(bool enable) {
|
|
return window_->SetContentProtection(enable);
|
|
}
|
|
|
|
void BaseWindow::SetFocusable(bool focusable) {
|
|
return window_->SetFocusable(focusable);
|
|
}
|
|
|
|
bool BaseWindow::IsFocusable() {
|
|
return window_->IsFocusable();
|
|
}
|
|
|
|
void BaseWindow::SetMenu(v8::Isolate* isolate, v8::Local<v8::Value> value) {
|
|
auto context = isolate->GetCurrentContext();
|
|
gin::Handle<Menu> menu;
|
|
v8::Local<v8::Object> object;
|
|
if (value->IsObject() && value->ToObject(context).ToLocal(&object) &&
|
|
gin::ConvertFromV8(isolate, value, &menu) && !menu.IsEmpty()) {
|
|
menu_.Reset(isolate, menu.ToV8());
|
|
|
|
// We only want to update the menu if the menu has a non-zero item count,
|
|
// or we risk crashes.
|
|
if (menu->model()->GetItemCount() == 0) {
|
|
RemoveMenu();
|
|
} else {
|
|
window_->SetMenu(menu->model());
|
|
}
|
|
} else if (value->IsNull()) {
|
|
RemoveMenu();
|
|
} else {
|
|
isolate->ThrowException(
|
|
v8::Exception::TypeError(gin::StringToV8(isolate, "Invalid Menu")));
|
|
}
|
|
}
|
|
|
|
void BaseWindow::RemoveMenu() {
|
|
menu_.Reset();
|
|
window_->SetMenu(nullptr);
|
|
}
|
|
|
|
void BaseWindow::SetParentWindow(v8::Local<v8::Value> value,
|
|
gin_helper::Arguments* args) {
|
|
if (IsModal()) {
|
|
args->ThrowError("Can not be called for modal window");
|
|
return;
|
|
}
|
|
|
|
gin::Handle<BaseWindow> parent;
|
|
if (value->IsNull() || value->IsUndefined()) {
|
|
RemoveFromParentChildWindows();
|
|
parent_window_.Reset();
|
|
window_->SetParentWindow(nullptr);
|
|
} else if (gin::ConvertFromV8(isolate(), value, &parent)) {
|
|
RemoveFromParentChildWindows();
|
|
parent_window_.Reset(isolate(), value);
|
|
window_->SetParentWindow(parent->window_.get());
|
|
parent->child_windows_.Set(isolate(), weak_map_id(), GetWrapper());
|
|
} else {
|
|
args->ThrowError("Must pass BaseWindow instance or null");
|
|
}
|
|
}
|
|
|
|
void BaseWindow::SetBrowserView(v8::Local<v8::Value> value) {
|
|
ResetBrowserViews();
|
|
AddBrowserView(value);
|
|
}
|
|
|
|
void BaseWindow::AddBrowserView(v8::Local<v8::Value> value) {
|
|
gin::Handle<BrowserView> browser_view;
|
|
if (value->IsObject() &&
|
|
gin::ConvertFromV8(isolate(), value, &browser_view)) {
|
|
auto get_that_view = browser_views_.find(browser_view->ID());
|
|
if (get_that_view == browser_views_.end()) {
|
|
// If we're reparenting a BrowserView, ensure that it's detached from
|
|
// its previous owner window.
|
|
auto* owner_window = browser_view->owner_window();
|
|
if (owner_window && owner_window != window_.get()) {
|
|
owner_window->RemoveBrowserView(browser_view->view());
|
|
browser_view->SetOwnerWindow(nullptr);
|
|
}
|
|
|
|
window_->AddBrowserView(browser_view->view());
|
|
browser_view->SetOwnerWindow(window_.get());
|
|
browser_views_[browser_view->ID()].Reset(isolate(), value);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BaseWindow::RemoveBrowserView(v8::Local<v8::Value> value) {
|
|
gin::Handle<BrowserView> browser_view;
|
|
if (value->IsObject() &&
|
|
gin::ConvertFromV8(isolate(), value, &browser_view)) {
|
|
auto get_that_view = browser_views_.find(browser_view->ID());
|
|
if (get_that_view != browser_views_.end()) {
|
|
window_->RemoveBrowserView(browser_view->view());
|
|
browser_view->SetOwnerWindow(nullptr);
|
|
(*get_that_view).second.Reset(isolate(), value);
|
|
browser_views_.erase(get_that_view);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BaseWindow::SetTopBrowserView(v8::Local<v8::Value> value,
|
|
gin_helper::Arguments* args) {
|
|
gin::Handle<BrowserView> browser_view;
|
|
if (value->IsObject() &&
|
|
gin::ConvertFromV8(isolate(), value, &browser_view)) {
|
|
auto* owner_window = browser_view->owner_window();
|
|
auto get_that_view = browser_views_.find(browser_view->ID());
|
|
if (get_that_view == browser_views_.end() ||
|
|
(owner_window && owner_window != window_.get())) {
|
|
args->ThrowError("Given BrowserView is not attached to the window");
|
|
return;
|
|
}
|
|
|
|
window_->SetTopBrowserView(browser_view->view());
|
|
}
|
|
}
|
|
|
|
std::string BaseWindow::GetMediaSourceId() const {
|
|
return window_->GetDesktopMediaID().ToString();
|
|
}
|
|
|
|
v8::Local<v8::Value> BaseWindow::GetNativeWindowHandle() {
|
|
// TODO(MarshallOfSound): Replace once
|
|
// https://chromium-review.googlesource.com/c/chromium/src/+/1253094/ has
|
|
// landed
|
|
NativeWindowHandle handle = window_->GetNativeWindowHandle();
|
|
return ToBuffer(isolate(), &handle, sizeof(handle));
|
|
}
|
|
|
|
void BaseWindow::SetProgressBar(double progress, gin_helper::Arguments* args) {
|
|
gin_helper::Dictionary options;
|
|
std::string mode;
|
|
args->GetNext(&options) && options.Get("mode", &mode);
|
|
|
|
NativeWindow::ProgressState state = NativeWindow::ProgressState::kNormal;
|
|
if (mode == "error")
|
|
state = NativeWindow::ProgressState::kError;
|
|
else if (mode == "paused")
|
|
state = NativeWindow::ProgressState::kPaused;
|
|
else if (mode == "indeterminate")
|
|
state = NativeWindow::ProgressState::kIndeterminate;
|
|
else if (mode == "none")
|
|
state = NativeWindow::ProgressState::kNone;
|
|
|
|
window_->SetProgressBar(progress, state);
|
|
}
|
|
|
|
void BaseWindow::SetOverlayIcon(const gfx::Image& overlay,
|
|
const std::string& description) {
|
|
window_->SetOverlayIcon(overlay, description);
|
|
}
|
|
|
|
void BaseWindow::SetVisibleOnAllWorkspaces(bool visible,
|
|
gin_helper::Arguments* args) {
|
|
gin_helper::Dictionary options;
|
|
bool visibleOnFullScreen = false;
|
|
bool skipTransformProcessType = false;
|
|
if (args->GetNext(&options)) {
|
|
options.Get("visibleOnFullScreen", &visibleOnFullScreen);
|
|
options.Get("skipTransformProcessType", &skipTransformProcessType);
|
|
}
|
|
return window_->SetVisibleOnAllWorkspaces(visible, visibleOnFullScreen,
|
|
skipTransformProcessType);
|
|
}
|
|
|
|
bool BaseWindow::IsVisibleOnAllWorkspaces() {
|
|
return window_->IsVisibleOnAllWorkspaces();
|
|
}
|
|
|
|
void BaseWindow::SetAutoHideCursor(bool auto_hide) {
|
|
window_->SetAutoHideCursor(auto_hide);
|
|
}
|
|
|
|
void BaseWindow::SetVibrancy(v8::Isolate* isolate, v8::Local<v8::Value> value) {
|
|
std::string type = gin::V8ToString(isolate, value);
|
|
window_->SetVibrancy(type);
|
|
}
|
|
|
|
#if BUILDFLAG(IS_MAC)
|
|
std::string BaseWindow::GetAlwaysOnTopLevel() {
|
|
return window_->GetAlwaysOnTopLevel();
|
|
}
|
|
|
|
void BaseWindow::SetWindowButtonVisibility(bool visible) {
|
|
window_->SetWindowButtonVisibility(visible);
|
|
}
|
|
|
|
bool BaseWindow::GetWindowButtonVisibility() const {
|
|
return window_->GetWindowButtonVisibility();
|
|
}
|
|
|
|
void BaseWindow::SetTrafficLightPosition(const gfx::Point& position) {
|
|
// For backward compatibility we treat (0, 0) as resetting to default.
|
|
if (position.IsOrigin())
|
|
window_->SetTrafficLightPosition(absl::nullopt);
|
|
else
|
|
window_->SetTrafficLightPosition(position);
|
|
}
|
|
|
|
gfx::Point BaseWindow::GetTrafficLightPosition() const {
|
|
// For backward compatibility we treat default value as (0, 0).
|
|
return window_->GetTrafficLightPosition().value_or(gfx::Point());
|
|
}
|
|
#endif
|
|
|
|
void BaseWindow::SetTouchBar(
|
|
std::vector<gin_helper::PersistentDictionary> items) {
|
|
window_->SetTouchBar(std::move(items));
|
|
}
|
|
|
|
void BaseWindow::RefreshTouchBarItem(const std::string& item_id) {
|
|
window_->RefreshTouchBarItem(item_id);
|
|
}
|
|
|
|
void BaseWindow::SetEscapeTouchBarItem(gin_helper::PersistentDictionary item) {
|
|
window_->SetEscapeTouchBarItem(std::move(item));
|
|
}
|
|
|
|
void BaseWindow::SelectPreviousTab() {
|
|
window_->SelectPreviousTab();
|
|
}
|
|
|
|
void BaseWindow::SelectNextTab() {
|
|
window_->SelectNextTab();
|
|
}
|
|
|
|
void BaseWindow::MergeAllWindows() {
|
|
window_->MergeAllWindows();
|
|
}
|
|
|
|
void BaseWindow::MoveTabToNewWindow() {
|
|
window_->MoveTabToNewWindow();
|
|
}
|
|
|
|
void BaseWindow::ToggleTabBar() {
|
|
window_->ToggleTabBar();
|
|
}
|
|
|
|
void BaseWindow::AddTabbedWindow(NativeWindow* window,
|
|
gin_helper::Arguments* args) {
|
|
if (!window_->AddTabbedWindow(window))
|
|
args->ThrowError("AddTabbedWindow cannot be called by a window on itself.");
|
|
}
|
|
|
|
void BaseWindow::SetAutoHideMenuBar(bool auto_hide) {
|
|
window_->SetAutoHideMenuBar(auto_hide);
|
|
}
|
|
|
|
bool BaseWindow::IsMenuBarAutoHide() {
|
|
return window_->IsMenuBarAutoHide();
|
|
}
|
|
|
|
void BaseWindow::SetMenuBarVisibility(bool visible) {
|
|
window_->SetMenuBarVisibility(visible);
|
|
}
|
|
|
|
bool BaseWindow::IsMenuBarVisible() {
|
|
return window_->IsMenuBarVisible();
|
|
}
|
|
|
|
void BaseWindow::SetAspectRatio(double aspect_ratio,
|
|
gin_helper::Arguments* args) {
|
|
gfx::Size extra_size;
|
|
args->GetNext(&extra_size);
|
|
window_->SetAspectRatio(aspect_ratio, extra_size);
|
|
}
|
|
|
|
void BaseWindow::PreviewFile(const std::string& path,
|
|
gin_helper::Arguments* args) {
|
|
std::string display_name;
|
|
if (!args->GetNext(&display_name))
|
|
display_name = path;
|
|
window_->PreviewFile(path, display_name);
|
|
}
|
|
|
|
void BaseWindow::CloseFilePreview() {
|
|
window_->CloseFilePreview();
|
|
}
|
|
|
|
void BaseWindow::SetGTKDarkThemeEnabled(bool use_dark_theme) {
|
|
window_->SetGTKDarkThemeEnabled(use_dark_theme);
|
|
}
|
|
|
|
v8::Local<v8::Value> BaseWindow::GetContentView() const {
|
|
if (content_view_.IsEmpty())
|
|
return v8::Null(isolate());
|
|
else
|
|
return v8::Local<v8::Value>::New(isolate(), content_view_);
|
|
}
|
|
|
|
v8::Local<v8::Value> BaseWindow::GetParentWindow() const {
|
|
if (parent_window_.IsEmpty())
|
|
return v8::Null(isolate());
|
|
else
|
|
return v8::Local<v8::Value>::New(isolate(), parent_window_);
|
|
}
|
|
|
|
std::vector<v8::Local<v8::Object>> BaseWindow::GetChildWindows() const {
|
|
return child_windows_.Values(isolate());
|
|
}
|
|
|
|
v8::Local<v8::Value> BaseWindow::GetBrowserView(
|
|
gin_helper::Arguments* args) const {
|
|
if (browser_views_.empty()) {
|
|
return v8::Null(isolate());
|
|
} else if (browser_views_.size() == 1) {
|
|
auto first_view = browser_views_.begin();
|
|
return v8::Local<v8::Value>::New(isolate(), (*first_view).second);
|
|
} else {
|
|
args->ThrowError(
|
|
"BrowserWindow have multiple BrowserViews, "
|
|
"Use getBrowserViews() instead");
|
|
return v8::Null(isolate());
|
|
}
|
|
}
|
|
|
|
std::vector<v8::Local<v8::Value>> BaseWindow::GetBrowserViews() const {
|
|
std::vector<v8::Local<v8::Value>> ret;
|
|
|
|
for (auto const& views_iter : browser_views_) {
|
|
ret.push_back(v8::Local<v8::Value>::New(isolate(), views_iter.second));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool BaseWindow::IsModal() const {
|
|
return window_->is_modal();
|
|
}
|
|
|
|
bool BaseWindow::SetThumbarButtons(gin_helper::Arguments* args) {
|
|
#if BUILDFLAG(IS_WIN)
|
|
std::vector<TaskbarHost::ThumbarButton> buttons;
|
|
if (!args->GetNext(&buttons)) {
|
|
args->ThrowError();
|
|
return false;
|
|
}
|
|
auto* window = static_cast<NativeWindowViews*>(window_.get());
|
|
return window->taskbar_host().SetThumbarButtons(
|
|
window_->GetAcceleratedWidget(), buttons);
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
#if defined(TOOLKIT_VIEWS)
|
|
void BaseWindow::SetIcon(v8::Isolate* isolate, v8::Local<v8::Value> icon) {
|
|
SetIconImpl(isolate, icon, NativeImage::OnConvertError::kThrow);
|
|
}
|
|
|
|
void BaseWindow::SetIconImpl(v8::Isolate* isolate,
|
|
v8::Local<v8::Value> icon,
|
|
NativeImage::OnConvertError on_error) {
|
|
NativeImage* native_image = nullptr;
|
|
if (!NativeImage::TryConvertNativeImage(isolate, icon, &native_image,
|
|
on_error))
|
|
return;
|
|
|
|
#if BUILDFLAG(IS_WIN)
|
|
static_cast<NativeWindowViews*>(window_.get())
|
|
->SetIcon(native_image->GetHICON(GetSystemMetrics(SM_CXSMICON)),
|
|
native_image->GetHICON(GetSystemMetrics(SM_CXICON)));
|
|
#elif BUILDFLAG(IS_LINUX)
|
|
static_cast<NativeWindowViews*>(window_.get())
|
|
->SetIcon(native_image->image().AsImageSkia());
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
#if BUILDFLAG(IS_WIN)
|
|
bool BaseWindow::HookWindowMessage(UINT message,
|
|
const MessageCallback& callback) {
|
|
messages_callback_map_[message] = callback;
|
|
return true;
|
|
}
|
|
|
|
void BaseWindow::UnhookWindowMessage(UINT message) {
|
|
messages_callback_map_.erase(message);
|
|
}
|
|
|
|
bool BaseWindow::IsWindowMessageHooked(UINT message) {
|
|
return base::Contains(messages_callback_map_, message);
|
|
}
|
|
|
|
void BaseWindow::UnhookAllWindowMessages() {
|
|
messages_callback_map_.clear();
|
|
}
|
|
|
|
bool BaseWindow::SetThumbnailClip(const gfx::Rect& region) {
|
|
auto* window = static_cast<NativeWindowViews*>(window_.get());
|
|
return window->taskbar_host().SetThumbnailClip(
|
|
window_->GetAcceleratedWidget(), region);
|
|
}
|
|
|
|
bool BaseWindow::SetThumbnailToolTip(const std::string& tooltip) {
|
|
auto* window = static_cast<NativeWindowViews*>(window_.get());
|
|
return window->taskbar_host().SetThumbnailToolTip(
|
|
window_->GetAcceleratedWidget(), tooltip);
|
|
}
|
|
|
|
void BaseWindow::SetAppDetails(const gin_helper::Dictionary& options) {
|
|
std::wstring app_id;
|
|
base::FilePath app_icon_path;
|
|
int app_icon_index = 0;
|
|
std::wstring relaunch_command;
|
|
std::wstring relaunch_display_name;
|
|
|
|
options.Get("appId", &app_id);
|
|
options.Get("appIconPath", &app_icon_path);
|
|
options.Get("appIconIndex", &app_icon_index);
|
|
options.Get("relaunchCommand", &relaunch_command);
|
|
options.Get("relaunchDisplayName", &relaunch_display_name);
|
|
|
|
ui::win::SetAppDetailsForWindow(app_id, app_icon_path, app_icon_index,
|
|
relaunch_command, relaunch_display_name,
|
|
window_->GetAcceleratedWidget());
|
|
}
|
|
#endif
|
|
|
|
int32_t BaseWindow::GetID() const {
|
|
return weak_map_id();
|
|
}
|
|
|
|
void BaseWindow::ResetBrowserViews() {
|
|
v8::HandleScope scope(isolate());
|
|
|
|
for (auto& item : browser_views_) {
|
|
gin::Handle<BrowserView> browser_view;
|
|
if (gin::ConvertFromV8(isolate(),
|
|
v8::Local<v8::Value>::New(isolate(), item.second),
|
|
&browser_view) &&
|
|
!browser_view.IsEmpty()) {
|
|
// There's a chance that the BrowserView may have been reparented - only
|
|
// reset if the owner window is *this* window.
|
|
auto* owner_window = browser_view->owner_window();
|
|
if (owner_window && owner_window == window_.get()) {
|
|
browser_view->SetOwnerWindow(nullptr);
|
|
owner_window->RemoveBrowserView(browser_view->view());
|
|
}
|
|
}
|
|
|
|
item.second.Reset();
|
|
}
|
|
|
|
browser_views_.clear();
|
|
}
|
|
|
|
void BaseWindow::RemoveFromParentChildWindows() {
|
|
if (parent_window_.IsEmpty())
|
|
return;
|
|
|
|
gin::Handle<BaseWindow> parent;
|
|
if (!gin::ConvertFromV8(isolate(), GetParentWindow(), &parent) ||
|
|
parent.IsEmpty()) {
|
|
return;
|
|
}
|
|
|
|
parent->child_windows_.Remove(weak_map_id());
|
|
}
|
|
|
|
// static
|
|
gin_helper::WrappableBase* BaseWindow::New(gin_helper::Arguments* args) {
|
|
gin_helper::Dictionary options =
|
|
gin::Dictionary::CreateEmpty(args->isolate());
|
|
args->GetNext(&options);
|
|
|
|
return new BaseWindow(args, options);
|
|
}
|
|
|
|
// static
|
|
void BaseWindow::BuildPrototype(v8::Isolate* isolate,
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
|
prototype->SetClassName(gin::StringToV8(isolate, "BaseWindow"));
|
|
gin_helper::Destroyable::MakeDestroyable(isolate, prototype);
|
|
gin_helper::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
|
|
.SetMethod("setContentView", &BaseWindow::SetContentView)
|
|
.SetMethod("close", &BaseWindow::Close)
|
|
.SetMethod("focus", &BaseWindow::Focus)
|
|
.SetMethod("blur", &BaseWindow::Blur)
|
|
.SetMethod("isFocused", &BaseWindow::IsFocused)
|
|
.SetMethod("show", &BaseWindow::Show)
|
|
.SetMethod("showInactive", &BaseWindow::ShowInactive)
|
|
.SetMethod("hide", &BaseWindow::Hide)
|
|
.SetMethod("isVisible", &BaseWindow::IsVisible)
|
|
.SetMethod("isEnabled", &BaseWindow::IsEnabled)
|
|
.SetMethod("setEnabled", &BaseWindow::SetEnabled)
|
|
.SetMethod("maximize", &BaseWindow::Maximize)
|
|
.SetMethod("unmaximize", &BaseWindow::Unmaximize)
|
|
.SetMethod("isMaximized", &BaseWindow::IsMaximized)
|
|
.SetMethod("minimize", &BaseWindow::Minimize)
|
|
.SetMethod("restore", &BaseWindow::Restore)
|
|
.SetMethod("isMinimized", &BaseWindow::IsMinimized)
|
|
.SetMethod("setFullScreen", &BaseWindow::SetFullScreen)
|
|
.SetMethod("isFullScreen", &BaseWindow::IsFullscreen)
|
|
.SetMethod("setBounds", &BaseWindow::SetBounds)
|
|
.SetMethod("getBounds", &BaseWindow::GetBounds)
|
|
.SetMethod("isNormal", &BaseWindow::IsNormal)
|
|
.SetMethod("getNormalBounds", &BaseWindow::GetNormalBounds)
|
|
.SetMethod("setSize", &BaseWindow::SetSize)
|
|
.SetMethod("getSize", &BaseWindow::GetSize)
|
|
.SetMethod("setContentBounds", &BaseWindow::SetContentBounds)
|
|
.SetMethod("getContentBounds", &BaseWindow::GetContentBounds)
|
|
.SetMethod("setContentSize", &BaseWindow::SetContentSize)
|
|
.SetMethod("getContentSize", &BaseWindow::GetContentSize)
|
|
.SetMethod("setMinimumSize", &BaseWindow::SetMinimumSize)
|
|
.SetMethod("getMinimumSize", &BaseWindow::GetMinimumSize)
|
|
.SetMethod("setMaximumSize", &BaseWindow::SetMaximumSize)
|
|
.SetMethod("getMaximumSize", &BaseWindow::GetMaximumSize)
|
|
.SetMethod("setSheetOffset", &BaseWindow::SetSheetOffset)
|
|
.SetMethod("moveAbove", &BaseWindow::MoveAbove)
|
|
.SetMethod("moveTop", &BaseWindow::MoveTop)
|
|
.SetMethod("setResizable", &BaseWindow::SetResizable)
|
|
.SetMethod("isResizable", &BaseWindow::IsResizable)
|
|
.SetMethod("setMovable", &BaseWindow::SetMovable)
|
|
.SetMethod("isMovable", &BaseWindow::IsMovable)
|
|
.SetMethod("setMinimizable", &BaseWindow::SetMinimizable)
|
|
.SetMethod("isMinimizable", &BaseWindow::IsMinimizable)
|
|
.SetMethod("setMaximizable", &BaseWindow::SetMaximizable)
|
|
.SetMethod("isMaximizable", &BaseWindow::IsMaximizable)
|
|
.SetMethod("setFullScreenable", &BaseWindow::SetFullScreenable)
|
|
.SetMethod("isFullScreenable", &BaseWindow::IsFullScreenable)
|
|
.SetMethod("setClosable", &BaseWindow::SetClosable)
|
|
.SetMethod("isClosable", &BaseWindow::IsClosable)
|
|
.SetMethod("setAlwaysOnTop", &BaseWindow::SetAlwaysOnTop)
|
|
.SetMethod("isAlwaysOnTop", &BaseWindow::IsAlwaysOnTop)
|
|
.SetMethod("center", &BaseWindow::Center)
|
|
.SetMethod("setPosition", &BaseWindow::SetPosition)
|
|
.SetMethod("getPosition", &BaseWindow::GetPosition)
|
|
.SetMethod("setTitle", &BaseWindow::SetTitle)
|
|
.SetMethod("getTitle", &BaseWindow::GetTitle)
|
|
.SetProperty("accessibleTitle", &BaseWindow::GetAccessibleTitle,
|
|
&BaseWindow::SetAccessibleTitle)
|
|
.SetMethod("flashFrame", &BaseWindow::FlashFrame)
|
|
.SetMethod("setSkipTaskbar", &BaseWindow::SetSkipTaskbar)
|
|
.SetMethod("setSimpleFullScreen", &BaseWindow::SetSimpleFullScreen)
|
|
.SetMethod("isSimpleFullScreen", &BaseWindow::IsSimpleFullScreen)
|
|
.SetMethod("setKiosk", &BaseWindow::SetKiosk)
|
|
.SetMethod("isKiosk", &BaseWindow::IsKiosk)
|
|
.SetMethod("isTabletMode", &BaseWindow::IsTabletMode)
|
|
.SetMethod("setBackgroundColor", &BaseWindow::SetBackgroundColor)
|
|
.SetMethod("getBackgroundColor", &BaseWindow::GetBackgroundColor)
|
|
.SetMethod("setHasShadow", &BaseWindow::SetHasShadow)
|
|
.SetMethod("hasShadow", &BaseWindow::HasShadow)
|
|
.SetMethod("setOpacity", &BaseWindow::SetOpacity)
|
|
.SetMethod("getOpacity", &BaseWindow::GetOpacity)
|
|
.SetMethod("setShape", &BaseWindow::SetShape)
|
|
.SetMethod("setRepresentedFilename", &BaseWindow::SetRepresentedFilename)
|
|
.SetMethod("getRepresentedFilename", &BaseWindow::GetRepresentedFilename)
|
|
.SetMethod("setDocumentEdited", &BaseWindow::SetDocumentEdited)
|
|
.SetMethod("isDocumentEdited", &BaseWindow::IsDocumentEdited)
|
|
.SetMethod("setIgnoreMouseEvents", &BaseWindow::SetIgnoreMouseEvents)
|
|
.SetMethod("setContentProtection", &BaseWindow::SetContentProtection)
|
|
.SetMethod("setFocusable", &BaseWindow::SetFocusable)
|
|
.SetMethod("isFocusable", &BaseWindow::IsFocusable)
|
|
.SetMethod("setMenu", &BaseWindow::SetMenu)
|
|
.SetMethod("removeMenu", &BaseWindow::RemoveMenu)
|
|
.SetMethod("setParentWindow", &BaseWindow::SetParentWindow)
|
|
.SetMethod("setBrowserView", &BaseWindow::SetBrowserView)
|
|
.SetMethod("addBrowserView", &BaseWindow::AddBrowserView)
|
|
.SetMethod("removeBrowserView", &BaseWindow::RemoveBrowserView)
|
|
.SetMethod("setTopBrowserView", &BaseWindow::SetTopBrowserView)
|
|
.SetMethod("getMediaSourceId", &BaseWindow::GetMediaSourceId)
|
|
.SetMethod("getNativeWindowHandle", &BaseWindow::GetNativeWindowHandle)
|
|
.SetMethod("setProgressBar", &BaseWindow::SetProgressBar)
|
|
.SetMethod("setOverlayIcon", &BaseWindow::SetOverlayIcon)
|
|
.SetMethod("setVisibleOnAllWorkspaces",
|
|
&BaseWindow::SetVisibleOnAllWorkspaces)
|
|
.SetMethod("isVisibleOnAllWorkspaces",
|
|
&BaseWindow::IsVisibleOnAllWorkspaces)
|
|
#if BUILDFLAG(IS_MAC)
|
|
.SetMethod("_getAlwaysOnTopLevel", &BaseWindow::GetAlwaysOnTopLevel)
|
|
.SetMethod("setAutoHideCursor", &BaseWindow::SetAutoHideCursor)
|
|
#endif
|
|
.SetMethod("setVibrancy", &BaseWindow::SetVibrancy)
|
|
#if BUILDFLAG(IS_MAC)
|
|
.SetMethod("setTrafficLightPosition",
|
|
&BaseWindow::SetTrafficLightPosition)
|
|
.SetMethod("getTrafficLightPosition",
|
|
&BaseWindow::GetTrafficLightPosition)
|
|
#endif
|
|
.SetMethod("_setTouchBarItems", &BaseWindow::SetTouchBar)
|
|
.SetMethod("_refreshTouchBarItem", &BaseWindow::RefreshTouchBarItem)
|
|
.SetMethod("_setEscapeTouchBarItem", &BaseWindow::SetEscapeTouchBarItem)
|
|
#if BUILDFLAG(IS_MAC)
|
|
.SetMethod("selectPreviousTab", &BaseWindow::SelectPreviousTab)
|
|
.SetMethod("selectNextTab", &BaseWindow::SelectNextTab)
|
|
.SetMethod("mergeAllWindows", &BaseWindow::MergeAllWindows)
|
|
.SetMethod("moveTabToNewWindow", &BaseWindow::MoveTabToNewWindow)
|
|
.SetMethod("toggleTabBar", &BaseWindow::ToggleTabBar)
|
|
.SetMethod("addTabbedWindow", &BaseWindow::AddTabbedWindow)
|
|
.SetMethod("setWindowButtonVisibility",
|
|
&BaseWindow::SetWindowButtonVisibility)
|
|
.SetMethod("_getWindowButtonVisibility",
|
|
&BaseWindow::GetWindowButtonVisibility)
|
|
.SetProperty("excludedFromShownWindowsMenu",
|
|
&BaseWindow::IsExcludedFromShownWindowsMenu,
|
|
&BaseWindow::SetExcludedFromShownWindowsMenu)
|
|
#endif
|
|
.SetMethod("setAutoHideMenuBar", &BaseWindow::SetAutoHideMenuBar)
|
|
.SetMethod("isMenuBarAutoHide", &BaseWindow::IsMenuBarAutoHide)
|
|
.SetMethod("setMenuBarVisibility", &BaseWindow::SetMenuBarVisibility)
|
|
.SetMethod("isMenuBarVisible", &BaseWindow::IsMenuBarVisible)
|
|
.SetMethod("setAspectRatio", &BaseWindow::SetAspectRatio)
|
|
.SetMethod("previewFile", &BaseWindow::PreviewFile)
|
|
.SetMethod("closeFilePreview", &BaseWindow::CloseFilePreview)
|
|
.SetMethod("getContentView", &BaseWindow::GetContentView)
|
|
.SetMethod("getParentWindow", &BaseWindow::GetParentWindow)
|
|
.SetMethod("getChildWindows", &BaseWindow::GetChildWindows)
|
|
.SetMethod("getBrowserView", &BaseWindow::GetBrowserView)
|
|
.SetMethod("getBrowserViews", &BaseWindow::GetBrowserViews)
|
|
.SetMethod("isModal", &BaseWindow::IsModal)
|
|
.SetMethod("setThumbarButtons", &BaseWindow::SetThumbarButtons)
|
|
#if defined(TOOLKIT_VIEWS)
|
|
.SetMethod("setIcon", &BaseWindow::SetIcon)
|
|
#endif
|
|
#if BUILDFLAG(IS_WIN)
|
|
.SetMethod("hookWindowMessage", &BaseWindow::HookWindowMessage)
|
|
.SetMethod("isWindowMessageHooked", &BaseWindow::IsWindowMessageHooked)
|
|
.SetMethod("unhookWindowMessage", &BaseWindow::UnhookWindowMessage)
|
|
.SetMethod("unhookAllWindowMessages",
|
|
&BaseWindow::UnhookAllWindowMessages)
|
|
.SetMethod("setThumbnailClip", &BaseWindow::SetThumbnailClip)
|
|
.SetMethod("setThumbnailToolTip", &BaseWindow::SetThumbnailToolTip)
|
|
.SetMethod("setAppDetails", &BaseWindow::SetAppDetails)
|
|
#endif
|
|
.SetProperty("id", &BaseWindow::GetID);
|
|
}
|
|
|
|
} // namespace api
|
|
|
|
} // namespace electron
|
|
|
|
namespace {
|
|
|
|
using electron::api::BaseWindow;
|
|
|
|
void Initialize(v8::Local<v8::Object> exports,
|
|
v8::Local<v8::Value> unused,
|
|
v8::Local<v8::Context> context,
|
|
void* priv) {
|
|
v8::Isolate* isolate = context->GetIsolate();
|
|
BaseWindow::SetConstructor(isolate, base::BindRepeating(&BaseWindow::New));
|
|
|
|
gin_helper::Dictionary constructor(isolate,
|
|
BaseWindow::GetConstructor(isolate)
|
|
->GetFunction(context)
|
|
.ToLocalChecked());
|
|
constructor.SetMethod("fromId", &BaseWindow::FromWeakMapID);
|
|
constructor.SetMethod("getAllWindows", &BaseWindow::GetAll);
|
|
|
|
gin_helper::Dictionary dict(isolate, exports);
|
|
dict.Set("BaseWindow", constructor);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
NODE_LINKED_MODULE_CONTEXT_AWARE(electron_browser_base_window, Initialize)
|