2014-10-31 11:17:05 -07:00
|
|
|
// Copyright (c) 2013 GitHub, Inc.
|
2014-04-25 17:49:37 +08:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2013-05-02 23:43:23 +08:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2014-03-16 08:30:26 +08:00
|
|
|
#include "atom/browser/browser.h"
|
2013-05-02 23:43:23 +08:00
|
|
|
|
2015-04-14 15:55:41 +08:00
|
|
|
#include "atom/browser/mac/atom_application.h"
|
|
|
|
#include "atom/browser/mac/atom_application_delegate.h"
|
2016-05-05 16:26:44 +09:00
|
|
|
#include "atom/browser/mac/dict_util.h"
|
2014-06-25 11:55:33 +08:00
|
|
|
#include "atom/browser/native_window.h"
|
|
|
|
#include "atom/browser/window_list.h"
|
2018-04-20 14:47:04 -04:00
|
|
|
#include "atom/common/platform_util.h"
|
2016-03-21 11:24:25 -07:00
|
|
|
#include "base/mac/bundle_locations.h"
|
2015-04-14 15:55:41 +08:00
|
|
|
#include "base/mac/foundation_util.h"
|
2016-07-06 11:43:44 -07:00
|
|
|
#include "base/mac/mac_util.h"
|
2016-07-01 17:39:01 +09:00
|
|
|
#include "base/strings/string_number_conversions.h"
|
2016-07-02 10:36:11 +09:00
|
|
|
#include "base/strings/sys_string_conversions.h"
|
2015-04-14 15:55:41 +08:00
|
|
|
#include "brightray/common/application_info.h"
|
2016-05-23 11:49:46 -04:00
|
|
|
#include "net/base/mac/url_conversions.h"
|
2018-03-06 16:12:10 +09:00
|
|
|
#include "ui/gfx/image/image.h"
|
2016-05-23 11:49:46 -04:00
|
|
|
#include "url/gurl.h"
|
2013-05-02 23:43:23 +08:00
|
|
|
|
|
|
|
namespace atom {
|
|
|
|
|
2018-02-05 16:13:35 +09:00
|
|
|
void Browser::SetShutdownHandler(base::Callback<bool()> handler) {
|
|
|
|
[[AtomApplication sharedApplication] setShutdownHandler:std::move(handler)];
|
|
|
|
}
|
|
|
|
|
2013-05-30 19:24:47 +08:00
|
|
|
void Browser::Focus() {
|
2013-06-26 17:22:24 +08:00
|
|
|
[[AtomApplication sharedApplication] activateIgnoringOtherApps:YES];
|
2013-05-30 19:24:47 +08:00
|
|
|
}
|
|
|
|
|
2016-01-29 12:30:19 +01:00
|
|
|
void Browser::Hide() {
|
|
|
|
[[AtomApplication sharedApplication] hide:nil];
|
|
|
|
}
|
|
|
|
|
2016-01-30 21:40:32 +01:00
|
|
|
void Browser::Show() {
|
|
|
|
[[AtomApplication sharedApplication] unhide:nil];
|
|
|
|
}
|
|
|
|
|
2014-11-17 13:05:06 +08:00
|
|
|
void Browser::AddRecentDocument(const base::FilePath& path) {
|
2015-12-15 17:17:24 +08:00
|
|
|
NSString* path_string = base::mac::FilePathToNSString(path);
|
|
|
|
if (!path_string)
|
|
|
|
return;
|
|
|
|
NSURL* u = [NSURL fileURLWithPath:path_string];
|
|
|
|
if (!u)
|
|
|
|
return;
|
2014-11-17 13:05:06 +08:00
|
|
|
[[NSDocumentController sharedDocumentController] noteNewRecentDocumentURL:u];
|
|
|
|
}
|
|
|
|
|
2014-11-17 16:13:47 +08:00
|
|
|
void Browser::ClearRecentDocuments() {
|
2015-12-29 18:35:54 -06:00
|
|
|
[[NSDocumentController sharedDocumentController] clearRecentDocuments:nil];
|
2014-11-17 16:13:47 +08:00
|
|
|
}
|
|
|
|
|
2016-08-16 15:54:30 +10:00
|
|
|
bool Browser::RemoveAsDefaultProtocolClient(const std::string& protocol,
|
|
|
|
mate::Arguments* args) {
|
2016-05-07 11:07:58 -07:00
|
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
|
|
if (!identifier)
|
|
|
|
return false;
|
|
|
|
|
2016-08-16 17:40:44 +10:00
|
|
|
if (!Browser::IsDefaultProtocolClient(protocol, args))
|
2016-05-08 10:50:17 -07:00
|
|
|
return false;
|
|
|
|
|
2016-05-07 11:07:58 -07:00
|
|
|
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;
|
|
|
|
}
|
2016-06-18 15:26:26 +02:00
|
|
|
// On macOS, we can't query the default, but the handlers list seems to put
|
2016-05-07 11:07:58 -07:00
|
|
|
// Apple's defaults first, so we'll use the first option that isn't our bundle
|
|
|
|
CFStringRef other = nil;
|
2017-03-30 22:56:21 +02:00
|
|
|
for (CFIndex i = 0; i < CFArrayGetCount(bundleList); ++i) {
|
2018-04-20 14:47:04 -04:00
|
|
|
other =
|
|
|
|
base::mac::CFCast<CFStringRef>(CFArrayGetValueAtIndex(bundleList, i));
|
|
|
|
if (![identifier isEqualToString:(__bridge NSString*)other]) {
|
2016-05-07 11:07:58 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
OSStatus return_code = LSSetDefaultHandlerForURLScheme(protocol_cf, other);
|
|
|
|
return return_code == noErr;
|
2016-03-24 10:55:09 -07:00
|
|
|
}
|
|
|
|
|
2016-08-16 15:54:30 +10:00
|
|
|
bool Browser::SetAsDefaultProtocolClient(const std::string& protocol,
|
2016-08-22 09:50:58 +09:00
|
|
|
mate::Arguments* args) {
|
2016-03-21 11:24:25 -07:00
|
|
|
if (protocol.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
|
|
if (!identifier)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
|
2018-04-20 14:47:04 -04:00
|
|
|
OSStatus return_code = LSSetDefaultHandlerForURLScheme(
|
|
|
|
base::mac::NSToCFCast(protocol_ns), base::mac::NSToCFCast(identifier));
|
2016-03-21 11:24:25 -07:00
|
|
|
return return_code == noErr;
|
|
|
|
}
|
|
|
|
|
2016-08-16 15:54:30 +10:00
|
|
|
bool Browser::IsDefaultProtocolClient(const std::string& protocol,
|
|
|
|
mate::Arguments* args) {
|
2016-04-24 22:17:01 -07:00
|
|
|
if (protocol.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
NSString* identifier = [base::mac::MainBundle() bundleIdentifier];
|
|
|
|
if (!identifier)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
NSString* protocol_ns = [NSString stringWithUTF8String:protocol.c_str()];
|
2016-04-30 11:17:29 -07:00
|
|
|
|
2016-04-24 22:17:01 -07:00
|
|
|
CFStringRef bundle =
|
|
|
|
LSCopyDefaultHandlerForURLScheme(base::mac::NSToCFCast(protocol_ns));
|
2018-04-20 14:47:04 -04:00
|
|
|
NSString* bundleId =
|
|
|
|
static_cast<NSString*>(base::mac::CFTypeRefToNSObjectAutorelease(bundle));
|
2016-04-24 22:17:01 -07:00
|
|
|
if (!bundleId)
|
|
|
|
return false;
|
|
|
|
|
2016-04-30 11:17:29 -07:00
|
|
|
// Ensure the comparison is case-insensitive
|
2016-04-24 22:17:01 -07:00
|
|
|
// as LS does not persist the case of the bundle id.
|
2018-04-20 14:47:04 -04:00
|
|
|
NSComparisonResult result = [bundleId caseInsensitiveCompare:identifier];
|
2016-04-24 22:17:01 -07:00
|
|
|
return result == NSOrderedSame;
|
|
|
|
}
|
|
|
|
|
2018-04-20 14:47:04 -04:00
|
|
|
void Browser::SetAppUserModelID(const base::string16& name) {}
|
2015-11-03 15:09:31 +08:00
|
|
|
|
2016-07-01 22:18:39 +09:00
|
|
|
bool Browser::SetBadgeCount(int count) {
|
|
|
|
DockSetBadgeText(count != 0 ? base::IntToString(count) : "");
|
2016-07-02 10:36:11 +09:00
|
|
|
badge_count_ = count;
|
2016-07-01 22:18:39 +09:00
|
|
|
return true;
|
2016-07-01 17:39:01 +09:00
|
|
|
}
|
|
|
|
|
2016-05-23 11:49:46 -04:00
|
|
|
void Browser::SetUserActivity(const std::string& type,
|
|
|
|
const base::DictionaryValue& user_info,
|
|
|
|
mate::Arguments* args) {
|
|
|
|
std::string url_string;
|
|
|
|
args->GetNext(&url_string);
|
|
|
|
|
2016-05-05 16:38:47 +09:00
|
|
|
[[AtomApplication sharedApplication]
|
|
|
|
setCurrentActivity:base::SysUTF8ToNSString(type)
|
2016-05-23 11:49:46 -04:00
|
|
|
withUserInfo:DictionaryValueToNSDictionary(user_info)
|
|
|
|
withWebpageURL:net::NSURLWithGURL(GURL(url_string))];
|
2016-05-03 15:51:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string Browser::GetCurrentActivityType() {
|
2018-04-17 16:45:26 -07:00
|
|
|
if (@available(macOS 10.10, *)) {
|
|
|
|
NSUserActivity* userActivity =
|
|
|
|
[[AtomApplication sharedApplication] getCurrentActivity];
|
|
|
|
return base::SysNSStringToUTF8(userActivity.activityType);
|
|
|
|
} else {
|
|
|
|
return std::string();
|
|
|
|
}
|
2016-04-29 17:35:07 -07:00
|
|
|
}
|
|
|
|
|
2017-06-26 16:14:44 -03:00
|
|
|
void Browser::InvalidateCurrentActivity() {
|
|
|
|
[[AtomApplication sharedApplication] invalidateCurrentActivity];
|
|
|
|
}
|
|
|
|
|
|
|
|
void Browser::UpdateCurrentActivity(const std::string& type,
|
|
|
|
const base::DictionaryValue& user_info) {
|
2017-09-14 16:12:34 +09:00
|
|
|
[[AtomApplication sharedApplication]
|
2017-06-26 16:14:44 -03:00
|
|
|
updateCurrentActivity:base::SysUTF8ToNSString(type)
|
|
|
|
withUserInfo:DictionaryValueToNSDictionary(user_info)];
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Browser::WillContinueUserActivity(const std::string& type) {
|
|
|
|
bool prevent_default = false;
|
2017-09-14 16:12:34 +09:00
|
|
|
for (BrowserObserver& observer : observers_)
|
|
|
|
observer.OnWillContinueUserActivity(&prevent_default, type);
|
2017-06-26 16:14:44 -03:00
|
|
|
return prevent_default;
|
|
|
|
}
|
2017-09-14 16:12:34 +09:00
|
|
|
|
2017-06-26 16:14:44 -03:00
|
|
|
void Browser::DidFailToContinueUserActivity(const std::string& type,
|
|
|
|
const std::string& error) {
|
|
|
|
for (BrowserObserver& observer : observers_)
|
|
|
|
observer.OnDidFailToContinueUserActivity(type, error);
|
|
|
|
}
|
|
|
|
|
2016-05-23 11:49:46 -04:00
|
|
|
bool Browser::ContinueUserActivity(const std::string& type,
|
|
|
|
const base::DictionaryValue& user_info) {
|
2016-05-05 16:26:44 +09:00
|
|
|
bool prevent_default = false;
|
2017-01-24 12:34:39 +09:00
|
|
|
for (BrowserObserver& observer : observers_)
|
|
|
|
observer.OnContinueUserActivity(&prevent_default, type, user_info);
|
2016-05-05 16:26:44 +09:00
|
|
|
return prevent_default;
|
|
|
|
}
|
2017-09-14 16:12:34 +09:00
|
|
|
|
2017-06-26 16:14:44 -03:00
|
|
|
void Browser::UserActivityWasContinued(const std::string& type,
|
|
|
|
const base::DictionaryValue& user_info) {
|
|
|
|
for (BrowserObserver& observer : observers_)
|
|
|
|
observer.OnUserActivityWasContinued(type, user_info);
|
|
|
|
}
|
|
|
|
|
2017-08-09 12:09:47 -03:00
|
|
|
bool Browser::UpdateUserActivityState(const std::string& type,
|
2017-06-26 16:14:44 -03:00
|
|
|
const base::DictionaryValue& user_info) {
|
2017-08-09 12:09:47 -03:00
|
|
|
bool prevent_default = false;
|
2017-06-26 16:14:44 -03:00
|
|
|
for (BrowserObserver& observer : observers_)
|
2017-08-09 12:09:47 -03:00
|
|
|
observer.OnUpdateUserActivityState(&prevent_default, type, user_info);
|
|
|
|
return prevent_default;
|
2017-06-26 16:14:44 -03:00
|
|
|
}
|
2016-05-05 16:26:44 +09:00
|
|
|
|
2017-01-30 14:01:40 -08:00
|
|
|
Browser::LoginItemSettings Browser::GetLoginItemSettings(
|
2017-03-29 21:29:52 +02:00
|
|
|
const LoginItemSettings& options) {
|
2016-07-07 16:29:09 -07:00
|
|
|
LoginItemSettings settings;
|
2017-10-17 16:28:29 +09:00
|
|
|
#if defined(MAS_BUILD)
|
|
|
|
settings.open_at_login = platform_util::GetLoginItemEnabled();
|
|
|
|
#else
|
2018-04-20 14:47:04 -04:00
|
|
|
settings.open_at_login =
|
|
|
|
base::mac::CheckLoginItemStatus(&settings.open_as_hidden);
|
2016-07-07 16:29:09 -07:00
|
|
|
settings.restore_state = base::mac::WasLaunchedAsLoginItemRestoreState();
|
|
|
|
settings.opened_at_login = base::mac::WasLaunchedAsLoginOrResumeItem();
|
|
|
|
settings.opened_as_hidden = base::mac::WasLaunchedAsHiddenLoginItem();
|
2017-10-17 16:28:29 +09:00
|
|
|
#endif
|
2016-07-07 16:29:09 -07:00
|
|
|
return settings;
|
2016-07-06 13:26:16 -07:00
|
|
|
}
|
|
|
|
|
2017-01-30 14:01:40 -08:00
|
|
|
void Browser::SetLoginItemSettings(LoginItemSettings settings) {
|
2017-10-17 16:28:29 +09:00
|
|
|
#if defined(MAS_BUILD)
|
|
|
|
platform_util::SetLoginItemEnabled(settings.open_at_login);
|
|
|
|
#else
|
2016-07-07 16:38:12 -07:00
|
|
|
if (settings.open_at_login)
|
2016-07-07 16:29:09 -07:00
|
|
|
base::mac::AddToLoginItems(settings.open_as_hidden);
|
2016-07-07 16:38:12 -07:00
|
|
|
else
|
2016-07-07 16:29:09 -07:00
|
|
|
base::mac::RemoveFromLoginItems();
|
2017-10-17 16:28:29 +09:00
|
|
|
#endif
|
2016-07-06 13:26:16 -07:00
|
|
|
}
|
|
|
|
|
2013-12-05 10:26:01 +08:00
|
|
|
std::string Browser::GetExecutableFileVersion() const {
|
2015-04-14 15:55:41 +08:00
|
|
|
return brightray::GetApplicationVersion();
|
2013-06-19 13:41:54 +08:00
|
|
|
}
|
|
|
|
|
2013-12-05 10:42:04 +08:00
|
|
|
std::string Browser::GetExecutableFileProductName() const {
|
2015-04-14 15:55:41 +08:00
|
|
|
return brightray::GetApplicationName();
|
2013-12-05 10:42:04 +08:00
|
|
|
}
|
|
|
|
|
2013-08-06 16:19:56 +08:00
|
|
|
int Browser::DockBounce(BounceType type) {
|
2015-12-21 10:52:49 +08:00
|
|
|
return [[AtomApplication sharedApplication]
|
2017-03-30 23:05:47 +02:00
|
|
|
requestUserAttention:static_cast<NSRequestUserAttentionType>(type)];
|
2013-08-06 16:19:56 +08:00
|
|
|
}
|
|
|
|
|
2016-01-23 22:59:05 +09:00
|
|
|
void Browser::DockCancelBounce(int request_id) {
|
|
|
|
[[AtomApplication sharedApplication] cancelUserAttentionRequest:request_id];
|
2013-08-06 16:19:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Browser::DockSetBadgeText(const std::string& label) {
|
2018-04-20 14:47:04 -04:00
|
|
|
NSDockTile* tile = [[AtomApplication sharedApplication] dockTile];
|
2013-08-06 16:19:56 +08:00
|
|
|
[tile setBadgeLabel:base::SysUTF8ToNSString(label)];
|
|
|
|
}
|
|
|
|
|
2016-05-10 15:02:56 -04:00
|
|
|
void Browser::DockDownloadFinished(const std::string& filePath) {
|
|
|
|
[[NSDistributedNotificationCenter defaultCenter]
|
2018-04-20 14:47:04 -04:00
|
|
|
postNotificationName:@"com.apple.DownloadFileFinished"
|
|
|
|
object:base::SysUTF8ToNSString(filePath)];
|
2016-05-10 15:02:56 -04:00
|
|
|
}
|
|
|
|
|
2013-08-06 16:39:31 +08:00
|
|
|
std::string Browser::DockGetBadgeText() {
|
2018-04-20 14:47:04 -04:00
|
|
|
NSDockTile* tile = [[AtomApplication sharedApplication] dockTile];
|
2013-08-06 16:39:31 +08:00
|
|
|
return base::SysNSStringToUTF8([tile badgeLabel]);
|
|
|
|
}
|
|
|
|
|
2014-06-25 11:55:33 +08:00
|
|
|
void Browser::DockHide() {
|
2018-04-17 15:41:47 -07:00
|
|
|
for (auto* const& window : WindowList::GetWindows())
|
2017-04-06 14:43:37 -07:00
|
|
|
[window->GetNativeWindow() setCanHide:NO];
|
2014-06-25 11:55:33 +08:00
|
|
|
|
2018-04-20 14:47:04 -04:00
|
|
|
ProcessSerialNumber psn = {0, kCurrentProcess};
|
2014-06-25 11:55:33 +08:00
|
|
|
TransformProcessType(&psn, kProcessTransformToUIElementApplication);
|
|
|
|
}
|
|
|
|
|
2016-08-01 15:22:37 -07:00
|
|
|
bool Browser::DockIsVisible() {
|
|
|
|
// Because DockShow has a slight delay this may not be true immediately
|
|
|
|
// after that call.
|
2018-04-20 14:47:04 -04:00
|
|
|
return ([[NSRunningApplication currentApplication] activationPolicy] ==
|
|
|
|
NSApplicationActivationPolicyRegular);
|
2016-08-01 15:22:37 -07:00
|
|
|
}
|
|
|
|
|
2014-06-25 11:55:33 +08:00
|
|
|
void Browser::DockShow() {
|
2015-12-17 14:00:04 -06:00
|
|
|
BOOL active = [[NSRunningApplication currentApplication] isActive];
|
2018-04-20 14:47:04 -04:00
|
|
|
ProcessSerialNumber psn = {0, kCurrentProcess};
|
2015-12-17 14:00:04 -06:00
|
|
|
if (active) {
|
2015-12-21 10:52:49 +08:00
|
|
|
// Workaround buggy behavior of TransformProcessType.
|
|
|
|
// http://stackoverflow.com/questions/7596643/
|
|
|
|
NSArray* runningApps = [NSRunningApplication
|
|
|
|
runningApplicationsWithBundleIdentifier:@"com.apple.dock"];
|
|
|
|
for (NSRunningApplication* app in runningApps) {
|
2015-12-17 14:00:04 -06:00
|
|
|
[app activateWithOptions:NSApplicationActivateIgnoringOtherApps];
|
|
|
|
break;
|
|
|
|
}
|
2015-12-21 10:52:49 +08:00
|
|
|
dispatch_time_t one_ms = dispatch_time(DISPATCH_TIME_NOW, USEC_PER_SEC);
|
|
|
|
dispatch_after(one_ms, dispatch_get_main_queue(), ^{
|
2015-12-17 14:00:04 -06:00
|
|
|
TransformProcessType(&psn, kProcessTransformToForegroundApplication);
|
2015-12-21 10:52:49 +08:00
|
|
|
dispatch_time_t one_ms = dispatch_time(DISPATCH_TIME_NOW, USEC_PER_SEC);
|
|
|
|
dispatch_after(one_ms, dispatch_get_main_queue(), ^{
|
|
|
|
[[NSRunningApplication currentApplication]
|
|
|
|
activateWithOptions:NSApplicationActivateIgnoringOtherApps];
|
|
|
|
});
|
2015-12-17 14:00:04 -06:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
TransformProcessType(&psn, kProcessTransformToForegroundApplication);
|
|
|
|
}
|
2014-06-25 11:55:33 +08:00
|
|
|
}
|
|
|
|
|
2016-07-02 11:47:40 +09:00
|
|
|
void Browser::DockSetMenu(AtomMenuModel* model) {
|
2018-04-20 14:47:04 -04:00
|
|
|
AtomApplicationDelegate* delegate =
|
|
|
|
(AtomApplicationDelegate*)[NSApp delegate];
|
2014-11-16 23:04:31 +08:00
|
|
|
[delegate setApplicationDockMenu:model];
|
|
|
|
}
|
|
|
|
|
2016-01-24 08:30:14 +09:00
|
|
|
void Browser::DockSetIcon(const gfx::Image& image) {
|
2016-02-01 17:06:09 +09:00
|
|
|
[[AtomApplication sharedApplication]
|
2016-01-24 08:30:14 +09:00
|
|
|
setApplicationIconImage:image.AsNSImage()];
|
|
|
|
}
|
|
|
|
|
2016-10-10 13:30:58 -07:00
|
|
|
void Browser::ShowAboutPanel() {
|
|
|
|
NSDictionary* options = DictionaryValueToNSDictionary(about_panel_options_);
|
2016-11-15 09:36:23 -08:00
|
|
|
|
|
|
|
// Credits must be a NSAttributedString instead of NSString
|
|
|
|
id credits = options[@"Credits"];
|
|
|
|
if (credits != nil) {
|
|
|
|
NSMutableDictionary* mutable_options = [options mutableCopy];
|
|
|
|
mutable_options[@"Credits"] = [[[NSAttributedString alloc]
|
|
|
|
initWithString:(NSString*)credits] autorelease];
|
|
|
|
options = [NSDictionary dictionaryWithDictionary:mutable_options];
|
|
|
|
}
|
|
|
|
|
2016-10-10 13:30:58 -07:00
|
|
|
[[AtomApplication sharedApplication]
|
|
|
|
orderFrontStandardAboutPanelWithOptions:options];
|
|
|
|
}
|
|
|
|
|
|
|
|
void Browser::SetAboutPanelOptions(const base::DictionaryValue& options) {
|
|
|
|
about_panel_options_.Clear();
|
2016-10-12 10:52:59 -07:00
|
|
|
|
|
|
|
// Upper case option keys for orderFrontStandardAboutPanelWithOptions format
|
2018-04-20 14:47:04 -04:00
|
|
|
for (base::DictionaryValue::Iterator iter(options); !iter.IsAtEnd();
|
2016-10-12 10:52:59 -07:00
|
|
|
iter.Advance()) {
|
|
|
|
std::string key = iter.key();
|
|
|
|
std::string value;
|
|
|
|
if (!key.empty() && iter.value().GetAsString(&value)) {
|
|
|
|
key[0] = base::ToUpperASCII(key[0]);
|
|
|
|
about_panel_options_.SetString(key, value);
|
|
|
|
}
|
|
|
|
}
|
2016-10-10 13:30:58 -07:00
|
|
|
}
|
|
|
|
|
2013-05-02 23:43:23 +08:00
|
|
|
} // namespace atom
|