90af7d7fe2
* chore: bump chromium in DEPS to 110.0.5461.0 * chore: update patches * 3903024: hid: Add connection count tracking methods for HidDelegate https://chromium-review.googlesource.com/c/chromium/src/+/3903024 * 4076211: Turn FrameTreeNode::frame_tree into raw_ref. https://chromium-review.googlesource.com/c/chromium/src/+/4076211 * chore: bump chromium in DEPS to 110.0.5463.0 * chore: bump chromium in DEPS to 110.0.5465.0 * fix patches * 3835037: Add new slides media session actions to Picture-in-Picture window https://chromium-review.googlesource.com/c/chromium/src/+/3835037 * chore: update patches * chore: bump chromium in DEPS to 110.0.5467.0 * chore: update patches * chore: bump chromium in DEPS to 110.0.5469.0 * chore: bump chromium in DEPS to 110.0.5471.0 * chore: bump chromium in DEPS to 110.0.5473.0 * chore: bump chromium in DEPS to 110.0.5475.0 * chore: update patches * 4074449: Add gl::FrameData to software path https://chromium-review.googlesource.com/c/chromium/src/+/4074449 * 4065264: [Extensions] Add a new side panel view type https://chromium-review.googlesource.com/c/chromium/src/+/4065264 * 4060548: Remove base::Value::GetListDeprecated(). https://chromium-review.googlesource.com/c/chromium/src/+/4060548 * chore: add missing RefCountedMemory include * 4081108: task posting v3: remove task_runner_util{.h,_unittest.cc} https://chromium-review.googlesource.com/c/chromium/src/+/4081108 * 4072471: Rename Mixed Download Blocking to Insecure Download Blocking https://chromium-review.googlesource.com/c/chromium/src/+/4072471 * 4025927: [Code Health] Migrate e/c/manifest.cc to base::Value::Dict interface https://chromium-review.googlesource.com/c/chromium/src/+/4025927 * chore: fixup patch indices * chore: bump chromium in DEPS to 110.0.5477.0 * chore: fixup preconnect_manager.patch * chore: fixup patch indices * fixup! 4074449: Add gl::FrameData to software path * 4074449: Add gl::FrameData to software path This commit also reformatted the two files in this patch. The only change here is the addition of the |data| arg to |OnSwapBuffers|. https://chromium-review.googlesource.com/c/chromium/src/+/4074449 * 4081108: task posting v3: remove task_runner_util{.h,_unittest.cc} https://chromium-review.googlesource.com/c/chromium/src/+/4081108 * 4085814: [Test Automation] Move NativeWindowTracker to ui/views https://chromium-review.googlesource.com/c/chromium/src/+/4085814 * 4032656: hid: Abstract HidSystemTrayIcon class for profiles' HID connections https://chromium-review.googlesource.com/c/chromium/src/+/4032656 * chore: bump chromium in DEPS to 110.0.5479.0 * chore: fixup patches & simplify printing patch To be specific, I replaced some combination of line removals & commenting-out with `#if 0` blocks since they were already there for android. Should be functionally the same, just written differently for better patch maintainability. * chore: bump chromium in DEPS to 110.0.5481.0 * chore: update patch indicies * 4098946: Migrate Extension::Create() argument to base::Value::Dict (part 4 of 4) https://chromium-review.googlesource.com/c/chromium/src/+/4098946 * chore: bump chromium in DEPS to 111.0.5482.0 * chore: bump chromium in DEPS to 111.0.5484.0 * chore: bump chromium in DEPS to 111.0.5486.0 * chore: update patch indices * 4112903: Reland "Move gl::FrameData to gfx::FrameData" https://chromium-review.googlesource.com/c/chromium/src/+/4112903 * 4056216: Option to create a tab target with Target.createTarget in /json/new https://chromium-review.googlesource.com/c/chromium/src/+/4056216 * chore: bump chromium in DEPS to 111.0.5488.0 * chore: bump chromium in DEPS to 111.0.5490.0 * chore: bump chromium in DEPS to 111.0.5492.0 * chore: bump chromium in DEPS to 111.0.5494.0 * chore: bump chromium in DEPS to 111.0.5496.0 * chore: bump chromium in DEPS to 111.0.5498.0 * chore: bump chromium in DEPS to 111.0.5500.0 * chore: bump chromium in DEPS to 111.0.5502.0 * chore: update patch indices + small update to printing.patch due to: 3653941: [printing] Extract settings logic from PrintJobWorker https://chromium-review.googlesource.com/c/chromium/src/+/3653941 * 4113994: Cleanup: Rename webui_generated_resources_* to webui_resources_*. https://chromium-review.googlesource.com/c/chromium/src/+/4113994 * 4112537: Remove DictionaryPrefUpdate. https://chromium-review.googlesource.com/c/chromium/src/+/4112537 * 4072073: Remove //chrome/browser/ash dependency from pdf_extension_util.cc https://chromium-review.googlesource.com/c/chromium/src/+/4072073 * 4055223: [Remove FileSystemConnector] Remove DownloadItemRerouteInfo https://chromium-review.googlesource.com/c/chromium/src/+/4055223 * Migrate base::DictionaryValue to base::Value::Dict This relates to multiple CLs cleaning up this class in this roll, but the ones that are specifically relevant here: 4116096: [CodeHealth] Remove DictionaryValue::GetInteger https://chromium-review.googlesource.com/c/chromium/src/+/4116096 4113764: [CodeHealth] Remove deprecated DictionaryValue::SetInteger() https://chromium-review.googlesource.com/c/chromium/src/+/4113764 * 3653941: [printing] Extract settings logic from PrintJobWorker https://chromium-review.googlesource.com/c/chromium/src/+/3653941 * chore: bump chromium in DEPS to 111.0.5504.0 * chore: update patches * (WIP) 4003663: Enable Microtask queues per WindowAgent. https://chromium-review.googlesource.com/c/chromium/src/+/4003663 * chore: bump chromium in DEPS to 111.0.5506.0 * chore: update patches * fix: printing patch It was complaining that this method isn't used on windows * chore: bump chromium in DEPS to 111.0.5508.0 * chore: bump chromium in DEPS to 111.0.5510.0 * chore: bump chromium in DEPS to 111.0.5512.0 * chore: bump chromium in DEPS to 111.0.5514.0 * chore: bump chromium in DEPS to 111.0.5516.0 * chore: update patches * chore: bump chromium in DEPS to 111.0.5518.0 * chore: update patches * 4027428: [rsafor] Move rsaFor requests to a separate permission https://chromium-review.googlesource.com/c/chromium/src/+/4027428 * Revert "(WIP) 4003663: Enable Microtask queues per WindowAgent." This reverts commit cc36d226e3f3fe5f4bea6538102d55ce3203190f. * chore: disable Microtask queues per WindowAgent. see https://chromium-review.googlesource.com/c/chromium/src/+/4003663 * chore: cleanup after rebase * fixup: disable Microtask queues per WindowAgent. * chore: cleanup password from keychain after test 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: Jeremy Rose <jeremya@chromium.org> Co-authored-by: PatchUp <73610968+patchup[bot]@users.noreply.github.com> Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> Co-authored-by: Calvin Watford <cwatford@slack-corp.com> Co-authored-by: clavin <clavin@electronjs.org>
537 lines
18 KiB
Text
537 lines
18 KiB
Text
// 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/browser.h"
|
|
|
|
#include <memory>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
#include "base/mac/bundle_locations.h"
|
|
#include "base/mac/foundation_util.h"
|
|
#include "base/mac/mac_util.h"
|
|
#include "base/mac/mac_util.mm"
|
|
#include "base/mac/scoped_cftyperef.h"
|
|
#include "base/strings/string_number_conversions.h"
|
|
#include "base/strings/sys_string_conversions.h"
|
|
#include "net/base/mac/url_conversions.h"
|
|
#include "shell/browser/badging/badge_manager.h"
|
|
#include "shell/browser/mac/dict_util.h"
|
|
#include "shell/browser/mac/electron_application.h"
|
|
#include "shell/browser/mac/electron_application_delegate.h"
|
|
#include "shell/browser/native_window.h"
|
|
#include "shell/browser/window_list.h"
|
|
#include "shell/common/api/electron_api_native_image.h"
|
|
#include "shell/common/application_info.h"
|
|
#include "shell/common/gin_converters/image_converter.h"
|
|
#include "shell/common/gin_helper/arguments.h"
|
|
#include "shell/common/gin_helper/dictionary.h"
|
|
#include "shell/common/gin_helper/error_thrower.h"
|
|
#include "shell/common/gin_helper/promise.h"
|
|
#include "shell/common/platform_util.h"
|
|
#include "ui/gfx/image/image.h"
|
|
#include "url/gurl.h"
|
|
|
|
namespace electron {
|
|
|
|
namespace {
|
|
|
|
NSString* GetAppPathForProtocol(const GURL& url) {
|
|
NSURL* ns_url = [NSURL
|
|
URLWithString:base::SysUTF8ToNSString(url.possibly_invalid_spec())];
|
|
base::ScopedCFTypeRef<CFErrorRef> out_err;
|
|
|
|
base::ScopedCFTypeRef<CFURLRef> openingApp(LSCopyDefaultApplicationURLForURL(
|
|
(CFURLRef)ns_url, kLSRolesAll, out_err.InitializeInto()));
|
|
|
|
if (out_err) {
|
|
// likely kLSApplicationNotFoundErr
|
|
return nullptr;
|
|
}
|
|
NSString* app_path = [base::mac::CFToNSCast(openingApp.get()) path];
|
|
return app_path;
|
|
}
|
|
|
|
gfx::Image GetApplicationIconForProtocol(NSString* _Nonnull app_path) {
|
|
NSImage* image = [[NSWorkspace sharedWorkspace] iconForFile:app_path];
|
|
gfx::Image icon(image);
|
|
return icon;
|
|
}
|
|
|
|
std::u16string GetAppDisplayNameForProtocol(NSString* app_path) {
|
|
NSString* app_display_name =
|
|
[[NSFileManager defaultManager] displayNameAtPath:app_path];
|
|
return base::SysNSStringToUTF16(app_display_name);
|
|
}
|
|
|
|
#if !IS_MAS_BUILD()
|
|
bool CheckLoginItemStatus(bool* is_hidden) {
|
|
base::mac::LoginItemsFileList login_items;
|
|
if (!login_items.Initialize())
|
|
return false;
|
|
|
|
base::ScopedCFTypeRef<LSSharedFileListItemRef> item(
|
|
login_items.GetLoginItemForMainApp());
|
|
if (!item.get())
|
|
return false;
|
|
|
|
if (is_hidden)
|
|
*is_hidden = base::mac::IsHiddenLoginItem(item);
|
|
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
} // namespace
|
|
|
|
v8::Local<v8::Promise> Browser::GetApplicationInfoForProtocol(
|
|
v8::Isolate* isolate,
|
|
const GURL& url) {
|
|
gin_helper::Promise<gin_helper::Dictionary> promise(isolate);
|
|
v8::Local<v8::Promise> handle = promise.GetHandle();
|
|
gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate);
|
|
|
|
NSString* ns_app_path = GetAppPathForProtocol(url);
|
|
|
|
if (!ns_app_path) {
|
|
promise.RejectWithErrorMessage(
|
|
"Unable to retrieve installation path to app");
|
|
return handle;
|
|
}
|
|
|
|
std::u16string app_path = base::SysNSStringToUTF16(ns_app_path);
|
|
std::u16string app_display_name = GetAppDisplayNameForProtocol(ns_app_path);
|
|
gfx::Image app_icon = GetApplicationIconForProtocol(ns_app_path);
|
|
|
|
dict.Set("name", app_display_name);
|
|
dict.Set("path", app_path);
|
|
dict.Set("icon", app_icon);
|
|
|
|
promise.Resolve(dict);
|
|
return handle;
|
|
}
|
|
|
|
void Browser::SetShutdownHandler(base::RepeatingCallback<bool()> handler) {
|
|
[[AtomApplication sharedApplication] setShutdownHandler:std::move(handler)];
|
|
}
|
|
|
|
void Browser::Focus(gin::Arguments* args) {
|
|
gin_helper::Dictionary opts;
|
|
bool steal_focus = false;
|
|
|
|
if (args->GetNext(&opts)) {
|
|
gin_helper::ErrorThrower thrower(args->isolate());
|
|
if (!opts.Get("steal", &steal_focus)) {
|
|
thrower.ThrowError(
|
|
"Expected options object to contain a 'steal' boolean property");
|
|
return;
|
|
}
|
|
}
|
|
|
|
[[AtomApplication sharedApplication] activateIgnoringOtherApps:steal_focus];
|
|
}
|
|
|
|
void Browser::Hide() {
|
|
[[AtomApplication sharedApplication] hide:nil];
|
|
}
|
|
|
|
bool Browser::IsHidden() {
|
|
return [[AtomApplication sharedApplication] isHidden];
|
|
}
|
|
|
|
void Browser::Show() {
|
|
[[AtomApplication sharedApplication] unhide:nil];
|
|
}
|
|
|
|
void Browser::AddRecentDocument(const base::FilePath& path) {
|
|
NSString* path_string = base::mac::FilePathToNSString(path);
|
|
if (!path_string)
|
|
return;
|
|
NSURL* u = [NSURL fileURLWithPath:path_string];
|
|
if (!u)
|
|
return;
|
|
[[NSDocumentController sharedDocumentController] noteNewRecentDocumentURL:u];
|
|
}
|
|
|
|
void Browser::ClearRecentDocuments() {
|
|
[[NSDocumentController sharedDocumentController] clearRecentDocuments:nil];
|
|
}
|
|
|
|
bool Browser::RemoveAsDefaultProtocolClient(const std::string& protocol,
|
|
gin::Arguments* args) {
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
if (!identifier)
|
|
return false;
|
|
|
|
if (!Browser::IsDefaultProtocolClient(protocol, args))
|
|
return false;
|
|
|
|
NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
|
|
CFStringRef protocol_cf = base::mac::NSToCFCast(protocol_ns);
|
|
CFArrayRef bundleList = LSCopyAllHandlersForURLScheme(protocol_cf);
|
|
if (!bundleList) {
|
|
return false;
|
|
}
|
|
// On macOS, we can't query the default, but the handlers list seems to put
|
|
// Apple's defaults first, so we'll use the first option that isn't our bundle
|
|
CFStringRef other = nil;
|
|
for (CFIndex i = 0; i < CFArrayGetCount(bundleList); ++i) {
|
|
other =
|
|
base::mac::CFCast<CFStringRef>(CFArrayGetValueAtIndex(bundleList, i));
|
|
if (![identifier isEqualToString:(__bridge NSString*)other]) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// No other app was found set it to none instead of setting it back to itself.
|
|
if ([identifier isEqualToString:(__bridge NSString*)other]) {
|
|
other = base::mac::NSToCFCast(@"None");
|
|
}
|
|
|
|
OSStatus return_code = LSSetDefaultHandlerForURLScheme(protocol_cf, other);
|
|
return return_code == noErr;
|
|
}
|
|
|
|
bool Browser::SetAsDefaultProtocolClient(const std::string& protocol,
|
|
gin::Arguments* args) {
|
|
if (protocol.empty())
|
|
return false;
|
|
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
if (!identifier)
|
|
return false;
|
|
|
|
NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
|
|
OSStatus return_code = LSSetDefaultHandlerForURLScheme(
|
|
base::mac::NSToCFCast(protocol_ns), base::mac::NSToCFCast(identifier));
|
|
return return_code == noErr;
|
|
}
|
|
|
|
bool Browser::IsDefaultProtocolClient(const std::string& protocol,
|
|
gin::Arguments* args) {
|
|
if (protocol.empty())
|
|
return false;
|
|
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
if (!identifier)
|
|
return false;
|
|
|
|
NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
|
|
|
|
base::ScopedCFTypeRef<CFStringRef> bundleId(
|
|
LSCopyDefaultHandlerForURLScheme(base::mac::NSToCFCast(protocol_ns)));
|
|
|
|
if (!bundleId)
|
|
return false;
|
|
|
|
// Ensure the comparison is case-insensitive
|
|
// as LS does not persist the case of the bundle id.
|
|
NSComparisonResult result =
|
|
[base::mac::CFToNSCast(bundleId) caseInsensitiveCompare:identifier];
|
|
return result == NSOrderedSame;
|
|
}
|
|
|
|
std::u16string Browser::GetApplicationNameForProtocol(const GURL& url) {
|
|
NSString* app_path = GetAppPathForProtocol(url);
|
|
if (!app_path) {
|
|
return std::u16string();
|
|
}
|
|
std::u16string app_display_name = GetAppDisplayNameForProtocol(app_path);
|
|
return app_display_name;
|
|
}
|
|
|
|
bool Browser::SetBadgeCount(absl::optional<int> count) {
|
|
DockSetBadgeText(!count.has_value() || count.value() != 0
|
|
? badging::BadgeManager::GetBadgeString(count)
|
|
: "");
|
|
if (count.has_value()) {
|
|
badge_count_ = count.value();
|
|
} else {
|
|
badge_count_ = 0;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void Browser::SetUserActivity(const std::string& type,
|
|
base::Value::Dict user_info,
|
|
gin::Arguments* args) {
|
|
std::string url_string;
|
|
args->GetNext(&url_string);
|
|
|
|
[[AtomApplication sharedApplication]
|
|
setCurrentActivity:base::SysUTF8ToNSString(type)
|
|
withUserInfo:DictionaryValueToNSDictionary(std::move(user_info))
|
|
withWebpageURL:net::NSURLWithGURL(GURL(url_string))];
|
|
}
|
|
|
|
std::string Browser::GetCurrentActivityType() {
|
|
NSUserActivity* userActivity =
|
|
[[AtomApplication sharedApplication] getCurrentActivity];
|
|
return base::SysNSStringToUTF8(userActivity.activityType);
|
|
}
|
|
|
|
void Browser::InvalidateCurrentActivity() {
|
|
[[AtomApplication sharedApplication] invalidateCurrentActivity];
|
|
}
|
|
|
|
void Browser::ResignCurrentActivity() {
|
|
[[AtomApplication sharedApplication] resignCurrentActivity];
|
|
}
|
|
|
|
void Browser::UpdateCurrentActivity(const std::string& type,
|
|
base::Value::Dict user_info) {
|
|
[[AtomApplication sharedApplication]
|
|
updateCurrentActivity:base::SysUTF8ToNSString(type)
|
|
withUserInfo:DictionaryValueToNSDictionary(
|
|
std::move(user_info))];
|
|
}
|
|
|
|
bool Browser::WillContinueUserActivity(const std::string& type) {
|
|
bool prevent_default = false;
|
|
for (BrowserObserver& observer : observers_)
|
|
observer.OnWillContinueUserActivity(&prevent_default, type);
|
|
return prevent_default;
|
|
}
|
|
|
|
void Browser::DidFailToContinueUserActivity(const std::string& type,
|
|
const std::string& error) {
|
|
for (BrowserObserver& observer : observers_)
|
|
observer.OnDidFailToContinueUserActivity(type, error);
|
|
}
|
|
|
|
bool Browser::ContinueUserActivity(const std::string& type,
|
|
base::Value::Dict user_info,
|
|
base::Value::Dict details) {
|
|
bool prevent_default = false;
|
|
for (BrowserObserver& observer : observers_)
|
|
observer.OnContinueUserActivity(&prevent_default, type, user_info.Clone(),
|
|
details.Clone());
|
|
return prevent_default;
|
|
}
|
|
|
|
void Browser::UserActivityWasContinued(const std::string& type,
|
|
base::Value::Dict user_info) {
|
|
for (BrowserObserver& observer : observers_)
|
|
observer.OnUserActivityWasContinued(type, user_info.Clone());
|
|
}
|
|
|
|
bool Browser::UpdateUserActivityState(const std::string& type,
|
|
base::Value::Dict user_info) {
|
|
bool prevent_default = false;
|
|
for (BrowserObserver& observer : observers_)
|
|
observer.OnUpdateUserActivityState(&prevent_default, type,
|
|
user_info.Clone());
|
|
return prevent_default;
|
|
}
|
|
|
|
Browser::LoginItemSettings Browser::GetLoginItemSettings(
|
|
const LoginItemSettings& options) {
|
|
LoginItemSettings settings;
|
|
#if IS_MAS_BUILD()
|
|
settings.open_at_login = platform_util::GetLoginItemEnabled();
|
|
#else
|
|
settings.open_at_login = CheckLoginItemStatus(&settings.open_as_hidden);
|
|
settings.restore_state = base::mac::WasLaunchedAsLoginItemRestoreState();
|
|
settings.opened_at_login = base::mac::WasLaunchedAsLoginOrResumeItem();
|
|
settings.opened_as_hidden = base::mac::WasLaunchedAsHiddenLoginItem();
|
|
#endif
|
|
return settings;
|
|
}
|
|
|
|
void Browser::SetLoginItemSettings(LoginItemSettings settings) {
|
|
#if IS_MAS_BUILD()
|
|
if (!platform_util::SetLoginItemEnabled(settings.open_at_login)) {
|
|
LOG(ERROR) << "Unable to set login item enabled on sandboxed app.";
|
|
}
|
|
#else
|
|
if (settings.open_at_login) {
|
|
base::mac::AddToLoginItems(base::mac::MainBundlePath(),
|
|
settings.open_as_hidden);
|
|
} else {
|
|
base::mac::RemoveFromLoginItems(base::mac::MainBundlePath());
|
|
}
|
|
#endif
|
|
}
|
|
|
|
std::string Browser::GetExecutableFileVersion() const {
|
|
return GetApplicationVersion();
|
|
}
|
|
|
|
std::string Browser::GetExecutableFileProductName() const {
|
|
return GetApplicationName();
|
|
}
|
|
|
|
int Browser::DockBounce(BounceType type) {
|
|
return [[AtomApplication sharedApplication]
|
|
requestUserAttention:static_cast<NSRequestUserAttentionType>(type)];
|
|
}
|
|
|
|
void Browser::DockCancelBounce(int request_id) {
|
|
[[AtomApplication sharedApplication] cancelUserAttentionRequest:request_id];
|
|
}
|
|
|
|
void Browser::DockSetBadgeText(const std::string& label) {
|
|
NSDockTile* tile = [[AtomApplication sharedApplication] dockTile];
|
|
[tile setBadgeLabel:base::SysUTF8ToNSString(label)];
|
|
}
|
|
|
|
void Browser::DockDownloadFinished(const std::string& filePath) {
|
|
[[NSDistributedNotificationCenter defaultCenter]
|
|
postNotificationName:@"com.apple.DownloadFileFinished"
|
|
object:base::SysUTF8ToNSString(filePath)];
|
|
}
|
|
|
|
std::string Browser::DockGetBadgeText() {
|
|
NSDockTile* tile = [[AtomApplication sharedApplication] dockTile];
|
|
return base::SysNSStringToUTF8([tile badgeLabel]);
|
|
}
|
|
|
|
void Browser::DockHide() {
|
|
// Transforming application state from UIElement to Foreground is an
|
|
// asynchronous operation, and unfortunately there is currently no way to know
|
|
// when it is finished.
|
|
// So if we call DockHide => DockShow => DockHide => DockShow in a very short
|
|
// time, we would trigger a bug of macOS that, there would be multiple dock
|
|
// icons of the app left in system.
|
|
// To work around this, we make sure DockHide does nothing if it is called
|
|
// immediately after DockShow. After some experiments, 1 second seems to be
|
|
// a proper interval.
|
|
if (!last_dock_show_.is_null() &&
|
|
base::Time::Now() - last_dock_show_ < base::Seconds(1)) {
|
|
return;
|
|
}
|
|
|
|
for (auto* const& window : WindowList::GetWindows())
|
|
[window->GetNativeWindow().GetNativeNSWindow() setCanHide:NO];
|
|
|
|
ProcessSerialNumber psn = {0, kCurrentProcess};
|
|
TransformProcessType(&psn, kProcessTransformToUIElementApplication);
|
|
}
|
|
|
|
bool Browser::DockIsVisible() {
|
|
// Because DockShow has a slight delay this may not be true immediately
|
|
// after that call.
|
|
return ([[NSRunningApplication currentApplication] activationPolicy] ==
|
|
NSApplicationActivationPolicyRegular);
|
|
}
|
|
|
|
v8::Local<v8::Promise> Browser::DockShow(v8::Isolate* isolate) {
|
|
last_dock_show_ = base::Time::Now();
|
|
gin_helper::Promise<void> promise(isolate);
|
|
v8::Local<v8::Promise> handle = promise.GetHandle();
|
|
|
|
BOOL active = [[NSRunningApplication currentApplication] isActive];
|
|
ProcessSerialNumber psn = {0, kCurrentProcess};
|
|
if (active) {
|
|
// Workaround buggy behavior of TransformProcessType.
|
|
// http://stackoverflow.com/questions/7596643/
|
|
NSArray* runningApps = [NSRunningApplication
|
|
runningApplicationsWithBundleIdentifier:@"com.apple.dock"];
|
|
for (NSRunningApplication* app in runningApps) {
|
|
[app activateWithOptions:NSApplicationActivateIgnoringOtherApps];
|
|
break;
|
|
}
|
|
__block gin_helper::Promise<void> p = std::move(promise);
|
|
dispatch_time_t one_ms = dispatch_time(DISPATCH_TIME_NOW, USEC_PER_SEC);
|
|
dispatch_after(one_ms, dispatch_get_main_queue(), ^{
|
|
TransformProcessType(&psn, kProcessTransformToForegroundApplication);
|
|
dispatch_time_t one_ms_2 = dispatch_time(DISPATCH_TIME_NOW, USEC_PER_SEC);
|
|
dispatch_after(one_ms_2, dispatch_get_main_queue(), ^{
|
|
[[NSRunningApplication currentApplication]
|
|
activateWithOptions:NSApplicationActivateIgnoringOtherApps];
|
|
p.Resolve();
|
|
});
|
|
});
|
|
} else {
|
|
TransformProcessType(&psn, kProcessTransformToForegroundApplication);
|
|
promise.Resolve();
|
|
}
|
|
return handle;
|
|
}
|
|
|
|
void Browser::DockSetMenu(ElectronMenuModel* model) {
|
|
ElectronApplicationDelegate* delegate =
|
|
(ElectronApplicationDelegate*)[NSApp delegate];
|
|
[delegate setApplicationDockMenu:model];
|
|
}
|
|
|
|
void Browser::DockSetIcon(v8::Isolate* isolate, v8::Local<v8::Value> icon) {
|
|
gfx::Image image;
|
|
|
|
if (!icon->IsNull()) {
|
|
api::NativeImage* native_image = nullptr;
|
|
if (!api::NativeImage::TryConvertNativeImage(isolate, icon, &native_image))
|
|
return;
|
|
image = native_image->image();
|
|
}
|
|
|
|
// This is needed when this fn is called before the browser
|
|
// process is ready, since supported scales are normally set
|
|
// by ui::ResourceBundle::InitSharedInstance
|
|
// during browser process startup.
|
|
if (!is_ready())
|
|
gfx::ImageSkia::SetSupportedScales({1.0f});
|
|
|
|
[[AtomApplication sharedApplication]
|
|
setApplicationIconImage:image.AsNSImage()];
|
|
}
|
|
|
|
void Browser::ShowAboutPanel() {
|
|
NSDictionary* options =
|
|
DictionaryValueToNSDictionary(about_panel_options_.GetDict());
|
|
|
|
// Credits must be a NSAttributedString instead of NSString
|
|
NSString* credits = (NSString*)options[@"Credits"];
|
|
if (credits != nil) {
|
|
base::scoped_nsobject<NSMutableDictionary> mutable_options(
|
|
[options mutableCopy]);
|
|
base::scoped_nsobject<NSAttributedString> creditString(
|
|
[[NSAttributedString alloc]
|
|
initWithString:credits
|
|
attributes:@{
|
|
NSForegroundColorAttributeName : [NSColor textColor]
|
|
}]);
|
|
|
|
[mutable_options setValue:creditString forKey:@"Credits"];
|
|
options = [NSDictionary dictionaryWithDictionary:mutable_options];
|
|
}
|
|
|
|
[[AtomApplication sharedApplication]
|
|
orderFrontStandardAboutPanelWithOptions:options];
|
|
}
|
|
|
|
void Browser::SetAboutPanelOptions(base::Value::Dict options) {
|
|
about_panel_options_.GetDict().clear();
|
|
|
|
for (const auto pair : options) {
|
|
std::string key = pair.first;
|
|
if (!key.empty() && pair.second.is_string()) {
|
|
key[0] = base::ToUpperASCII(key[0]);
|
|
about_panel_options_.GetDict().Set(key, pair.second.Clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
void Browser::ShowEmojiPanel() {
|
|
[[AtomApplication sharedApplication] orderFrontCharacterPalette:nil];
|
|
}
|
|
|
|
bool Browser::IsEmojiPanelSupported() {
|
|
return true;
|
|
}
|
|
|
|
bool Browser::IsSecureKeyboardEntryEnabled() {
|
|
return password_input_enabler_.get() != nullptr;
|
|
}
|
|
|
|
void Browser::SetSecureKeyboardEntryEnabled(bool enabled) {
|
|
if (enabled) {
|
|
password_input_enabler_ =
|
|
std::make_unique<ui::ScopedPasswordInputEnabler>();
|
|
} else {
|
|
password_input_enabler_.reset();
|
|
}
|
|
}
|
|
|
|
} // namespace electron
|