123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 |
- // 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/ui/message_box.h"
- #include <utility>
- #import <Cocoa/Cocoa.h>
- #include "base/containers/contains.h"
- #include "base/containers/flat_map.h"
- #include "base/mac/mac_util.h"
- #include "base/no_destructor.h"
- #include "base/strings/sys_string_conversions.h"
- #include "content/public/browser/browser_task_traits.h"
- #include "content/public/browser/browser_thread.h"
- #include "shell/browser/native_window.h"
- #include "skia/ext/skia_utils_mac.h"
- #include "ui/gfx/image/image_skia.h"
- namespace electron {
- MessageBoxSettings::MessageBoxSettings() = default;
- MessageBoxSettings::MessageBoxSettings(const MessageBoxSettings&) = default;
- MessageBoxSettings::~MessageBoxSettings() = default;
- namespace {
- // <ID, messageBox> map
- base::flat_map<int, NSAlert*>& GetDialogsMap() {
- static base::NoDestructor<base::flat_map<int, NSAlert*>> dialogs;
- return *dialogs;
- }
- NSAlert* CreateNSAlert(const MessageBoxSettings& settings) {
- // Ignore the title; it's the window title on other platforms and ignorable.
- NSAlert* alert = [[NSAlert alloc] init];
- [alert setMessageText:base::SysUTF8ToNSString(settings.message)];
- [alert setInformativeText:base::SysUTF8ToNSString(settings.detail)];
- switch (settings.type) {
- case MessageBoxType::kInformation:
- alert.alertStyle = NSAlertStyleInformational;
- break;
- case MessageBoxType::kWarning:
- case MessageBoxType::kError:
- // NSWarningAlertStyle shows the app icon while NSAlertStyleCritical
- // shows a warning icon with an app icon badge. Since there is no
- // error variant, lets just use NSAlertStyleCritical.
- alert.alertStyle = NSAlertStyleCritical;
- break;
- default:
- break;
- }
- for (size_t i = 0; i < settings.buttons.size(); ++i) {
- NSString* title = base::SysUTF8ToNSString(settings.buttons[i]);
- // An empty title causes crash on macOS.
- if (settings.buttons[i].empty())
- title = @"(empty)";
- NSButton* button = [alert addButtonWithTitle:title];
- [button setTag:i];
- }
- NSArray* ns_buttons = [alert buttons];
- int button_count = static_cast<int>([ns_buttons count]);
- if (settings.default_id >= 0 && settings.default_id < button_count) {
- // The first button added gets set as the default selected, so remove
- // that and set the button @ default_id to be default.
- [[ns_buttons objectAtIndex:0] setKeyEquivalent:@""];
- [[ns_buttons objectAtIndex:settings.default_id] setKeyEquivalent:@"\r"];
- }
- if (button_count > 1 && settings.cancel_id >= 0) {
- // Bind cancel id button to escape key if there is more than one button.
- if (settings.cancel_id < button_count) {
- [[ns_buttons objectAtIndex:settings.cancel_id] setKeyEquivalent:@"\e"];
- }
- // TODO(@codebytere): This behavior violates HIG & should be deprecated.
- if (settings.cancel_id == settings.default_id) {
- [(NSButton*)[ns_buttons objectAtIndex:settings.default_id] highlight:YES];
- }
- }
- if (!settings.checkbox_label.empty()) {
- alert.showsSuppressionButton = YES;
- alert.suppressionButton.title =
- base::SysUTF8ToNSString(settings.checkbox_label);
- alert.suppressionButton.state = settings.checkbox_checked
- ? NSControlStateValueOn
- : NSControlStateValueOff;
- }
- if (!settings.icon.isNull()) {
- NSImage* image = skia::SkBitmapToNSImage(*settings.icon.bitmap());
- [alert setIcon:image];
- }
- if (settings.text_width > 0) {
- NSRect rect = NSMakeRect(0, 0, settings.text_width, 0);
- NSView* accessoryView = [[NSView alloc] initWithFrame:rect];
- [alert setAccessoryView:accessoryView];
- }
- return alert;
- }
- } // namespace
- int ShowMessageBoxSync(const MessageBoxSettings& settings) {
- NSAlert* alert(CreateNSAlert(settings));
- // Use runModal for synchronous alert without parent, since we don't have a
- // window to wait for. Also use it when window is provided but it is not
- // shown as it would be impossible to dismiss the alert if it is connected
- // to invisible window (see #22671).
- if (!settings.parent_window || !settings.parent_window->IsVisible())
- return [alert runModal];
- __block int ret_code = -1;
- NSWindow* window =
- settings.parent_window->GetNativeWindow().GetNativeNSWindow();
- [alert beginSheetModalForWindow:window
- completionHandler:^(NSModalResponse response) {
- ret_code = response;
- [NSApp stopModal];
- }];
- [NSApp runModalForWindow:window];
- return ret_code;
- }
- void ShowMessageBox(const MessageBoxSettings& settings,
- MessageBoxCallback callback) {
- NSAlert* alert = CreateNSAlert(settings);
- // Use runModal for synchronous alert without parent, since we don't have a
- // window to wait for.
- if (!settings.parent_window) {
- int ret = [alert runModal];
- std::move(callback).Run(
- ret, alert.suppressionButton.state == NSControlStateValueOn);
- } else {
- if (settings.id) {
- if (base::Contains(GetDialogsMap(), *settings.id))
- CloseMessageBox(*settings.id);
- GetDialogsMap()[*settings.id] = alert;
- }
- NSWindow* window =
- settings.parent_window
- ? settings.parent_window->GetNativeWindow().GetNativeNSWindow()
- : nil;
- // Duplicate the callback object here since c is a reference and gcd would
- // only store the pointer, by duplication we can force gcd to store a copy.
- __block MessageBoxCallback callback_ = std::move(callback);
- __block std::optional<int> id = std::move(settings.id);
- __block int cancel_id = settings.cancel_id;
- auto handler = ^(NSModalResponse response) {
- if (id)
- GetDialogsMap().erase(*id);
- // When the alert is cancelled programmatically, the response would be
- // something like -1000. This currently only happens when users call
- // CloseMessageBox API, and we should return cancelId as result.
- if (response < 0)
- response = cancel_id;
- bool suppressed = alert.suppressionButton.state == NSControlStateValueOn;
- // The completionHandler runs inside a transaction commit, and we should
- // not do any runModal inside it. However since we can not control what
- // users will run in the callback, we have to delay running the callback
- // until next tick, otherwise crash like this may happen:
- // https://github.com/electron/electron/issues/26884
- content::GetUIThreadTaskRunner({})->PostTask(
- FROM_HERE,
- base::BindOnce(std::move(callback_), response, suppressed));
- };
- [alert beginSheetModalForWindow:window completionHandler:handler];
- }
- }
- void CloseMessageBox(int id) {
- auto it = GetDialogsMap().find(id);
- if (it == GetDialogsMap().end()) {
- LOG(ERROR) << "CloseMessageBox called with nonexistent ID";
- return;
- }
- [NSApp endSheet:it->second.window];
- }
- void ShowErrorBox(const std::u16string& title, const std::u16string& content) {
- NSAlert* alert = [[NSAlert alloc] init];
- [alert setMessageText:base::SysUTF16ToNSString(title)];
- [alert setInformativeText:base::SysUTF16ToNSString(content)];
- [alert setAlertStyle:NSAlertStyleCritical];
- [alert runModal];
- }
- } // namespace electron
|