electron/browser/api/atom_api_menu_mac.mm

110 lines
3.7 KiB
Text
Raw Normal View History

2013-05-06 12:27:09 +00:00
// Copyright (c) 2013 GitHub, Inc. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#import "browser/api/atom_api_menu_mac.h"
2013-05-14 08:50:56 +00:00
#include "base/mac/scoped_sending_event.h"
#include "base/message_loop/message_loop.h"
#include "base/strings/sys_string_conversions.h"
2013-05-14 08:50:56 +00:00
#include "browser/native_window.h"
#include "common/v8/node_common.h"
#include "common/v8/native_type_conversions.h"
2013-05-14 08:50:56 +00:00
#include "content/public/browser/web_contents.h"
#include "content/public/browser/web_contents_view.h"
2013-05-06 12:27:09 +00:00
namespace atom {
namespace api {
MenuMac::MenuMac(v8::Handle<v8::Object> wrapper)
2013-05-14 11:25:14 +00:00
: Menu(wrapper) {
2013-05-06 12:27:09 +00:00
}
MenuMac::~MenuMac() {
}
2013-05-14 11:25:14 +00:00
void MenuMac::Popup(NativeWindow* native_window) {
base::scoped_nsobject<AtomMenuController> menu_controller(
2013-08-14 15:03:02 +00:00
[[AtomMenuController alloc] initWithModel:model_.get()]);
2013-05-14 11:25:14 +00:00
2013-05-14 08:50:56 +00:00
NSWindow* window = native_window->GetNativeWindow();
content::WebContents* web_contents = native_window->GetWebContents();
// Fake out a context menu event.
NSEvent* currentEvent = [NSApp currentEvent];
2013-05-14 11:25:14 +00:00
NSPoint position = [window mouseLocationOutsideOfEventStream];
2013-05-14 08:50:56 +00:00
NSTimeInterval eventTime = [currentEvent timestamp];
NSEvent* clickEvent = [NSEvent mouseEventWithType:NSRightMouseDown
location:position
modifierFlags:NSRightMouseDownMask
timestamp:eventTime
windowNumber:[window windowNumber]
context:nil
eventNumber:0
clickCount:1
pressure:1.0];
{
// Make sure events can be pumped while the menu is up.
base::MessageLoop::ScopedNestableTaskAllower allow(
base::MessageLoop::current());
2013-05-14 08:50:56 +00:00
// One of the events that could be pumped is |window.close()|.
// User-initiated event-tracking loops protect against this by
// setting flags in -[CrApplication sendEvent:], but since
// web-content menus are initiated by IPC message the setup has to
// be done manually.
base::mac::ScopedSendingEvent sendingEventScoper;
// Show the menu.
2013-05-14 11:25:14 +00:00
[NSMenu popUpContextMenu:[menu_controller menu]
2013-05-14 08:50:56 +00:00
withEvent:clickEvent
2013-05-14 11:25:14 +00:00
forView:web_contents->GetView()->GetContentNativeView()];
2013-05-14 08:50:56 +00:00
}
}
// static
void MenuMac::SendActionToFirstResponder(const std::string& action) {
SEL selector = NSSelectorFromString(base::SysUTF8ToNSString(action));
[[NSApplication sharedApplication] sendAction:selector
to:nil
from:[NSApp mainMenu]];
}
// static
void Menu::SetApplicationMenu(const v8::FunctionCallbackInfo<v8::Value>& args) {
if (!args[0]->IsObject())
return node::ThrowTypeError("Bad argument");
MenuMac* menu = ObjectWrap::Unwrap<MenuMac>(args[0]->ToObject());
if (!menu)
return node::ThrowError("Menu is destroyed");
base::scoped_nsobject<AtomMenuController> menu_controller(
2013-08-14 15:03:02 +00:00
[[AtomMenuController alloc] initWithModel:menu->model_.get()]);
[NSApp setMainMenu:[menu_controller menu]];
// Ensure the menu_controller_ is destroyed after main menu is set.
menu_controller.swap(menu->menu_controller_);
}
// static
void Menu::SendActionToFirstResponder(
const v8::FunctionCallbackInfo<v8::Value>& args) {
std::string action;
if (!FromV8Arguments(args, &action))
return node::ThrowTypeError("Bad argument");
MenuMac::SendActionToFirstResponder(action);
}
2013-05-06 12:27:09 +00:00
// static
Menu* Menu::Create(v8::Handle<v8::Object> wrapper) {
return new MenuMac(wrapper);
}
} // namespace api
} // namespace atom