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.
|
|
|
|
|
|
|
|
#include "browser/api/atom_api_menu.h"
|
|
|
|
|
2013-05-14 13:37:59 +00:00
|
|
|
#include "browser/accelerator_util.h"
|
2013-05-14 11:26:16 +00:00
|
|
|
#include "browser/api/atom_api_window.h"
|
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
#define UNWRAP_MEMNU_AND_CHECK \
|
|
|
|
Menu* self = ObjectWrap::Unwrap<Menu>(args.This()); \
|
|
|
|
if (self == NULL) \
|
|
|
|
return node::ThrowError("Menu is already destroyed")
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
namespace atom {
|
|
|
|
|
|
|
|
namespace api {
|
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Converts a V8 value to a string16.
|
|
|
|
string16 V8ValueToUTF16(v8::Handle<v8::Value> value) {
|
|
|
|
v8::String::Value s(value);
|
|
|
|
return string16(reinterpret_cast<const char16*>(*s), s.length());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Converts string16 to V8 String.
|
|
|
|
v8::Handle<v8::Value> UTF16ToV8Value(const string16& s) {
|
|
|
|
return v8::String::New(reinterpret_cast<const uint16_t*>(s.data()), s.size());
|
|
|
|
}
|
|
|
|
|
2013-05-14 11:26:16 +00:00
|
|
|
// Call method of delegate object.
|
|
|
|
v8::Handle<v8::Value> CallDelegate(v8::Handle<v8::Value> default_value,
|
|
|
|
v8::Handle<v8::Object> menu,
|
|
|
|
const char* method,
|
|
|
|
int command_id) {
|
|
|
|
v8::HandleScope scope;
|
|
|
|
|
|
|
|
v8::Handle<v8::Value> delegate = menu->Get(v8::String::New("delegate"));
|
|
|
|
if (!delegate->IsObject())
|
|
|
|
return default_value;
|
|
|
|
|
|
|
|
v8::Handle<v8::Function> function = v8::Handle<v8::Function>::Cast(
|
|
|
|
delegate->ToObject()->Get(v8::String::New(method)));
|
|
|
|
if (!function->IsFunction())
|
|
|
|
return default_value;
|
|
|
|
|
2013-05-16 07:32:37 +00:00
|
|
|
v8::Handle<v8::Value> argv = v8::Integer::New(command_id);
|
|
|
|
|
2013-05-14 11:26:16 +00:00
|
|
|
return scope.Close(
|
2013-05-16 07:32:37 +00:00
|
|
|
function->Call(v8::Context::GetCurrent()->Global(), 1, &argv));
|
2013-05-14 11:26:16 +00:00
|
|
|
}
|
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
} // namespace
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
Menu::Menu(v8::Handle<v8::Object> wrapper)
|
|
|
|
: EventEmitter(wrapper),
|
|
|
|
model_(new ui::SimpleMenuModel(this)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
Menu::~Menu() {
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Menu::IsCommandIdChecked(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return CallDelegate(v8::False(),
|
|
|
|
handle(),
|
|
|
|
"isCommandIdChecked",
|
|
|
|
command_id)->BooleanValue();
|
2013-05-06 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Menu::IsCommandIdEnabled(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return CallDelegate(v8::True(),
|
|
|
|
handle(),
|
|
|
|
"isCommandIdEnabled",
|
|
|
|
command_id)->BooleanValue();
|
2013-05-06 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
2013-05-14 08:50:56 +00:00
|
|
|
bool Menu::IsCommandIdVisible(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return CallDelegate(v8::True(),
|
|
|
|
handle(),
|
|
|
|
"isCommandIdVisible",
|
|
|
|
command_id)->BooleanValue();
|
2013-05-14 08:50:56 +00:00
|
|
|
}
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
bool Menu::GetAcceleratorForCommandId(int command_id,
|
2013-05-14 11:26:16 +00:00
|
|
|
ui::Accelerator* accelerator) {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 13:37:59 +00:00
|
|
|
v8::Handle<v8::Value> shortcut = CallDelegate(v8::Undefined(),
|
|
|
|
handle(),
|
|
|
|
"getAcceleratorForCommandId",
|
|
|
|
command_id);
|
|
|
|
if (shortcut->IsString()) {
|
|
|
|
std::string shortcut_str(*v8::String::Utf8Value(shortcut));
|
|
|
|
return accelerator_util::StringToAccelerator(shortcut_str, accelerator);
|
|
|
|
}
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-05-14 08:50:56 +00:00
|
|
|
bool Menu::IsItemForCommandIdDynamic(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return CallDelegate(v8::False(),
|
|
|
|
handle(),
|
|
|
|
"isItemForCommandIdDynamic",
|
|
|
|
command_id)->BooleanValue();
|
2013-05-14 08:50:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string16 Menu::GetLabelForCommandId(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return V8ValueToUTF16(CallDelegate(v8::False(),
|
|
|
|
handle(),
|
|
|
|
"getLabelForCommandId",
|
|
|
|
command_id));
|
2013-05-14 08:50:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string16 Menu::GetSublabelForCommandId(int command_id) const {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 11:26:16 +00:00
|
|
|
return V8ValueToUTF16(CallDelegate(v8::False(),
|
|
|
|
handle(),
|
|
|
|
"getSubLabelForCommandId",
|
|
|
|
command_id));
|
2013-05-14 08:50:56 +00:00
|
|
|
}
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
void Menu::ExecuteCommand(int command_id, int event_flags) {
|
2013-05-16 02:54:37 +00:00
|
|
|
v8::HandleScope scope;
|
2013-05-14 12:24:53 +00:00
|
|
|
v8::Handle<v8::Value> args[] = {
|
|
|
|
v8::String::New("execute"),
|
|
|
|
v8::Integer::New(command_id)
|
|
|
|
};
|
|
|
|
node::MakeCallback(handle(), "emit", 2, args);
|
2013-05-06 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::New(const v8::Arguments &args) {
|
|
|
|
v8::HandleScope scope;
|
|
|
|
|
|
|
|
if (!args.IsConstructCall())
|
|
|
|
return node::ThrowError("Require constructor call");
|
|
|
|
|
|
|
|
Menu::Create(args.This());
|
|
|
|
|
|
|
|
return args.This();
|
|
|
|
}
|
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::InsertItem(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() || !args[1]->IsNumber() || !args[2]->IsString())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
|
|
|
|
if (index < 0)
|
|
|
|
self->model_->AddItem(args[1]->IntegerValue(), V8ValueToUTF16(args[2]));
|
|
|
|
else
|
|
|
|
self->model_->InsertItemAt(
|
|
|
|
index, args[1]->IntegerValue(), V8ValueToUTF16(args[2]));
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::InsertCheckItem(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() || !args[1]->IsNumber() || !args[2]->IsString())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
int index = args[0]->IntegerValue();
|
2013-05-07 03:43:21 +00:00
|
|
|
int command_id = args[1]->IntegerValue();
|
2013-05-06 16:03:34 +00:00
|
|
|
|
|
|
|
if (index < 0)
|
2013-05-07 03:43:21 +00:00
|
|
|
self->model_->AddCheckItem(command_id, V8ValueToUTF16(args[2]));
|
2013-05-06 16:03:34 +00:00
|
|
|
else
|
2013-05-07 03:43:21 +00:00
|
|
|
self->model_->InsertCheckItemAt(index, command_id, V8ValueToUTF16(args[2]));
|
2013-05-06 16:03:34 +00:00
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::InsertRadioItem(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() ||
|
|
|
|
!args[1]->IsNumber() ||
|
|
|
|
!args[2]->IsString() ||
|
|
|
|
!args[3]->IsNumber())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
int command_id = args[1]->IntegerValue();
|
|
|
|
int group_id = args[3]->IntegerValue();
|
|
|
|
|
|
|
|
if (index < 0)
|
|
|
|
self->model_->AddRadioItem(command_id, V8ValueToUTF16(args[2]), group_id);
|
|
|
|
else
|
|
|
|
self->model_->InsertRadioItemAt(
|
|
|
|
index, command_id, V8ValueToUTF16(args[2]), group_id);
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::InsertSeparator(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
2013-05-16 11:34:23 +00:00
|
|
|
if (!args[0]->IsNumber())
|
2013-05-06 16:03:34 +00:00
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
|
|
|
|
if (index < 0)
|
|
|
|
self->model_->AddSeparator(ui::NORMAL_SEPARATOR);
|
|
|
|
else
|
|
|
|
self->model_->InsertSeparatorAt(index, ui::NORMAL_SEPARATOR);
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::InsertSubMenu(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() ||
|
|
|
|
!args[1]->IsNumber() ||
|
|
|
|
!args[2]->IsString() ||
|
|
|
|
!args[3]->IsObject())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
Menu* submenu = ObjectWrap::Unwrap<Menu>(args[3]->ToObject());
|
|
|
|
if (!submenu)
|
|
|
|
return node::ThrowTypeError("The submenu is already destroyed");
|
|
|
|
|
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
int command_id = args[1]->IntegerValue();
|
|
|
|
|
|
|
|
if (index < 0)
|
|
|
|
self->model_->AddSubMenu(
|
|
|
|
command_id, V8ValueToUTF16(args[2]), submenu->model_.get());
|
|
|
|
else
|
|
|
|
self->model_->InsertSubMenuAt(
|
|
|
|
index, command_id, V8ValueToUTF16(args[2]), submenu->model_.get());
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::SetIcon(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() || !args[1]->IsString())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
// FIXME use webkit_glue's image decoder here.
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::SetSublabel(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
if (!args[0]->IsNumber() || !args[1]->IsString())
|
|
|
|
return node::ThrowTypeError("Bad argument");
|
|
|
|
|
|
|
|
self->model_->SetSublabel(args[0]->IntegerValue(), V8ValueToUTF16(args[1]));
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::Clear(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
|
|
|
self->model_->Clear();
|
|
|
|
|
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::GetIndexOfCommandId(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
2013-05-07 03:43:21 +00:00
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
return v8::Integer::New(self->model_->GetIndexOfCommandId(index));
|
2013-05-06 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::GetItemCount(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
return v8::Integer::New(self->model_->GetItemCount());
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::GetCommandIdAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
2013-05-07 03:43:21 +00:00
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
return v8::Integer::New(self->model_->GetCommandIdAt(index));
|
2013-05-06 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::GetLabelAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
2013-05-07 03:43:21 +00:00
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
return UTF16ToV8Value(self->model_->GetLabelAt(index));
|
2013-05-06 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::GetSublabelAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
2013-05-07 03:43:21 +00:00
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
return UTF16ToV8Value(self->model_->GetSublabelAt(index));
|
2013-05-06 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::IsItemCheckedAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
2013-05-07 03:43:21 +00:00
|
|
|
int index = args[0]->IntegerValue();
|
|
|
|
return v8::Boolean::New(self->model_->IsItemCheckedAt(index));
|
2013-05-06 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::IsEnabledAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
return v8::Boolean::New(self->model_->IsEnabledAt(args[0]->IntegerValue()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::IsVisibleAt(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
return v8::Boolean::New(self->model_->IsVisibleAt(args[0]->IntegerValue()));
|
|
|
|
}
|
|
|
|
|
2013-05-14 08:50:56 +00:00
|
|
|
// static
|
|
|
|
v8::Handle<v8::Value> Menu::Popup(const v8::Arguments &args) {
|
|
|
|
UNWRAP_MEMNU_AND_CHECK;
|
|
|
|
|
2013-05-14 11:26:16 +00:00
|
|
|
Window* window = Window::Unwrap<Window>(args[0]->ToObject());
|
2013-05-14 08:50:56 +00:00
|
|
|
if (!window)
|
|
|
|
return node::ThrowTypeError("Invalid window");
|
|
|
|
|
2013-05-14 11:26:16 +00:00
|
|
|
self->Popup(window->window());
|
2013-05-14 08:50:56 +00:00
|
|
|
return v8::Undefined();
|
|
|
|
}
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
// static
|
|
|
|
void Menu::Initialize(v8::Handle<v8::Object> target) {
|
|
|
|
v8::HandleScope scope;
|
|
|
|
|
|
|
|
v8::Local<v8::FunctionTemplate> t(v8::FunctionTemplate::New(Menu::New));
|
|
|
|
t->InstanceTemplate()->SetInternalFieldCount(1);
|
|
|
|
t->SetClassName(v8::String::NewSymbol("Menu"));
|
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "insertItem", InsertItem);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "insertCheckItem", InsertCheckItem);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "insertRadioItem", InsertRadioItem);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "insertSeparator", InsertSeparator);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "insertSubMenu", InsertSubMenu);
|
2013-05-14 08:50:56 +00:00
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "setIcon", SetIcon);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "setSublabel", SetSublabel);
|
2013-05-14 08:50:56 +00:00
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "clear", Clear);
|
2013-05-14 08:50:56 +00:00
|
|
|
|
2013-05-06 16:03:34 +00:00
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "getIndexOfCommandId", GetIndexOfCommandId);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "getItemCount", GetItemCount);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "getCommandIdAt", GetCommandIdAt);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "getLabelAt", GetLabelAt);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "getSublabelAt", GetSublabelAt);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "isItemCheckedAt", IsItemCheckedAt);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "isEnabledAt", IsEnabledAt);
|
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "isVisibleAt", IsVisibleAt);
|
|
|
|
|
2013-05-14 08:50:56 +00:00
|
|
|
NODE_SET_PROTOTYPE_METHOD(t, "popup", Popup);
|
|
|
|
|
2013-05-06 12:27:09 +00:00
|
|
|
target->Set(v8::String::NewSymbol("Menu"), t->GetFunction());
|
2013-05-16 02:54:37 +00:00
|
|
|
|
|
|
|
NODE_SET_METHOD(target, "setApplicationMenu", SetApplicationMenu);
|
2013-05-16 09:25:02 +00:00
|
|
|
NODE_SET_METHOD(
|
|
|
|
target, "sendActionToFirstResponder", SendActionToFirstResponder);
|
2013-05-06 12:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace api
|
|
|
|
|
|
|
|
} // namespace atom
|
|
|
|
|
|
|
|
NODE_MODULE(atom_browser_menu, atom::api::Menu::Initialize)
|