123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628 |
- // 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>
- #import <ServiceManagement/ServiceManagement.h>
- #include "base/apple/bridging.h"
- #include "base/apple/bundle_locations.h"
- #include "base/apple/scoped_cftyperef.h"
- #include "base/i18n/rtl.h"
- #include "base/mac/mac_util.h"
- #include "base/mac/mac_util.mm"
- #include "base/strings/sys_string_conversions.h"
- #include "chrome/browser/browser_process.h"
- #include "net/base/apple/url_conversions.h"
- #include "shell/browser/badging/badge_manager.h"
- #include "shell/browser/javascript_environment.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_converters/login_item_settings_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/base/resource/resource_scale_factor.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::apple::ScopedCFTypeRef<CFErrorRef> out_err;
- base::apple::ScopedCFTypeRef<CFURLRef> openingApp(
- LSCopyDefaultApplicationURLForURL(base::apple::NSToCFPtrCast(ns_url),
- kLSRolesAll, out_err.InitializeInto()));
- if (out_err) {
- // likely kLSApplicationNotFoundErr
- return nullptr;
- }
- NSString* app_path = [base::apple::CFToNSPtrCast(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::apple::ScopedCFTypeRef<LSSharedFileListItemRef> item(
- login_items.GetLoginItemForMainApp());
- if (!item.get())
- return false;
- if (is_hidden)
- *is_hidden = base::mac::IsHiddenLoginItem(item.get());
- return true;
- }
- LoginItemSettings GetLoginItemSettingsDeprecated() {
- LoginItemSettings settings;
- 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();
- return settings;
- }
- #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();
- auto dict = gin_helper::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::apple::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::apple::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::apple::NSToCFPtrCast(protocol_ns);
- // TODO(codebytere): Use -[NSWorkspace URLForApplicationToOpenURL:] instead
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdeprecated-declarations"
- CFArrayRef bundleList = LSCopyAllHandlersForURLScheme(protocol_cf);
- #pragma clang diagnostic pop
- 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::apple::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::apple::NSToCFPtrCast(@"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::apple::MainBundle() bundleIdentifier];
- if (!identifier)
- return false;
- NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
- OSStatus return_code =
- LSSetDefaultHandlerForURLScheme(base::apple::NSToCFPtrCast(protocol_ns),
- base::apple::NSToCFPtrCast(identifier));
- return return_code == noErr;
- }
- bool Browser::IsDefaultProtocolClient(const std::string& protocol,
- gin::Arguments* args) {
- if (protocol.empty())
- return false;
- NSString* identifier = [base::apple::MainBundle() bundleIdentifier];
- if (!identifier)
- return false;
- NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
- // TODO(codebytere): Use -[NSWorkspace URLForApplicationToOpenURL:] instead
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdeprecated-declarations"
- base::apple::ScopedCFTypeRef<CFStringRef> bundleId(
- LSCopyDefaultHandlerForURLScheme(
- base::apple::NSToCFPtrCast(protocol_ns)));
- #pragma clang diagnostic pop
- if (!bundleId)
- return false;
- // Ensure the comparison is case-insensitive
- // as LS does not persist the case of the bundle id.
- NSComparisonResult result = [base::apple::CFToNSPtrCast(bundleId.get())
- 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(std::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;
- }
- // Modified from chrome/browser/ui/cocoa/l10n_util.mm.
- void Browser::ApplyForcedRTL() {
- NSUserDefaults* defaults = NSUserDefaults.standardUserDefaults;
- auto dir = base::i18n::GetForcedTextDirection();
- // An Electron app should respect RTL behavior of application locale over
- // system locale.
- auto should_be_rtl = dir == base::i18n::RIGHT_TO_LEFT || IsAppRTL();
- auto should_be_ltr = dir == base::i18n::LEFT_TO_RIGHT || !IsAppRTL();
- // -registerDefaults: won't do the trick here because these defaults exist
- // (in the global domain) to reflect the system locale. They need to be set
- // in Chrome's domain to supersede the system value.
- if (should_be_rtl) {
- [defaults setBool:YES forKey:@"AppleTextDirection"];
- [defaults setBool:YES forKey:@"NSForceRightToLeftWritingDirection"];
- } else if (should_be_ltr) {
- [defaults setBool:YES forKey:@"AppleTextDirection"];
- [defaults setBool:NO forKey:@"NSForceRightToLeftWritingDirection"];
- } else {
- [defaults removeObjectForKey:@"AppleTextDirection"];
- [defaults removeObjectForKey:@"NSForceRightToLeftWritingDirection"];
- }
- }
- v8::Local<v8::Value> Browser::GetLoginItemSettings(
- const LoginItemSettings& options) {
- LoginItemSettings settings;
- v8::Isolate* isolate = JavascriptEnvironment::GetIsolate();
- if (options.type != "mainAppService" && options.service_name.empty()) {
- gin_helper::ErrorThrower(isolate).ThrowTypeError(
- "'name' is required when type is not mainAppService");
- return v8::Local<v8::Value>();
- }
- #if IS_MAS_BUILD()
- const std::string status =
- platform_util::GetLoginItemEnabled(options.type, options.service_name);
- settings.open_at_login =
- status == "enabled" || status == "enabled-deprecated";
- if (@available(macOS 13, *))
- settings.status = status;
- #else
- // If the app was previously set as a LoginItem with the deprecated API,
- // we should report its LoginItemSettings via the old API.
- LoginItemSettings settings_deprecated = GetLoginItemSettingsDeprecated();
- if (@available(macOS 13, *)) {
- const std::string status =
- platform_util::GetLoginItemEnabled(options.type, options.service_name);
- if (status == "enabled-deprecated") {
- settings = settings_deprecated;
- } else {
- settings.open_at_login = status == "enabled";
- settings.status = status;
- }
- } else {
- settings = settings_deprecated;
- }
- #endif
- return gin::ConvertToV8(isolate, settings);
- }
- void Browser::SetLoginItemSettings(LoginItemSettings settings) {
- if (settings.type != "mainAppService" && settings.service_name.empty()) {
- gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate())
- .ThrowTypeError("'name' is required when type is not mainAppService");
- return;
- }
- #if IS_MAS_BUILD()
- platform_util::SetLoginItemEnabled(settings.type, settings.service_name,
- settings.open_at_login);
- #else
- const base::FilePath bundle_path = base::apple::MainBundlePath();
- if (@available(macOS 13, *)) {
- // If the app was previously set as a LoginItem with the old API, remove it
- // as a LoginItem via the old API before re-enabling with the new API.
- const std::string status =
- platform_util::GetLoginItemEnabled("mainAppService", "");
- if (status == "enabled-deprecated") {
- base::mac::RemoveFromLoginItems(bundle_path);
- if (settings.open_at_login) {
- platform_util::SetLoginItemEnabled(settings.type, settings.service_name,
- settings.open_at_login);
- }
- } else {
- platform_util::SetLoginItemEnabled(settings.type, settings.service_name,
- settings.open_at_login);
- }
- } else {
- if (settings.open_at_login) {
- base::mac::AddToLoginItems(bundle_path, settings.open_as_hidden);
- } else {
- base::mac::RemoveFromLoginItems(bundle_path);
- }
- }
- #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 to avoid a hard CHECK 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())
- ui::SetSupportedResourceScaleFactors({ui::k100Percent});
- [[AtomApplication sharedApplication]
- setApplicationIconImage:image.AsNSImage()];
- }
- void Browser::ShowAboutPanel() {
- NSDictionary* options = DictionaryValueToNSDictionary(about_panel_options_);
- // Credits must be a NSAttributedString instead of NSString
- NSString* credits = (NSString*)options[@"Credits"];
- if (credits != nil) {
- NSMutableDictionary* mutable_options = [options mutableCopy];
- 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_.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_.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
|