// Copyright (c) 2014 GitHub, Inc. All rights reserved. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/browser/api/event_emitter.h" #include "atom/browser/api/event.h" #include "atom/common/native_mate_converters/v8_value_converter.h" #include "base/memory/scoped_ptr.h" #include "base/values.h" #include "native_mate/arguments.h" #include "native_mate/object_template_builder.h" #include "atom/common/node_includes.h" namespace mate { namespace { v8::Persistent<v8::ObjectTemplate> event_template; void PreventDefault(mate::Arguments* args) { args->GetThis()->SetHiddenValue( StringToV8(args->isolate(), "prevent_default"), v8::True(args->isolate())); } // Create a pure JavaScript Event object. v8::Local<v8::Object> CreateEventObject(v8::Isolate* isolate) { if (event_template.IsEmpty()) { event_template.Reset(isolate, ObjectTemplateBuilder(isolate) .SetMethod("preventDefault", &PreventDefault) .Build()); } return v8::Local<v8::ObjectTemplate>::New( isolate, event_template)->NewInstance(); } } // namespace EventEmitter::EventEmitter() { } bool EventEmitter::Emit(const base::StringPiece& name) { return Emit(name, base::ListValue()); } bool EventEmitter::Emit(const base::StringPiece& name, const base::ListValue& args) { return Emit(name, args, NULL, NULL); } bool EventEmitter::Emit(const base::StringPiece& name, const base::ListValue& args, content::WebContents* sender, IPC::Message* message) { v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::Locker locker(isolate); v8::HandleScope handle_scope(isolate); v8::Handle<v8::Context> context = isolate->GetCurrentContext(); scoped_ptr<atom::V8ValueConverter> converter(new atom::V8ValueConverter); // v8_args = [args...]; Arguments v8_args; v8_args.reserve(args.GetSize()); for (size_t i = 0; i < args.GetSize(); i++) { const base::Value* value(NULL); if (args.Get(i, &value)) v8_args.push_back(converter->ToV8Value(value, context)); } return Emit(isolate, name, v8_args, sender, message); } bool EventEmitter::Emit(v8::Isolate* isolate, const base::StringPiece& name, Arguments args, content::WebContents* sender, IPC::Message* message) { v8::Handle<v8::Object> event; bool use_native_event = sender && message; if (use_native_event) { mate::Handle<mate::Event> native_event = mate::Event::Create(isolate); native_event->SetSenderAndMessage(sender, message); event = v8::Handle<v8::Object>::Cast(native_event.ToV8()); } else { event = CreateEventObject(isolate); } // args = [name, event, args...]; args.insert(args.begin(), event); args.insert(args.begin(), mate::StringToV8(isolate, name)); // this.emit.apply(this, args); node::MakeCallback(isolate, GetWrapper(isolate), "emit", args.size(), &args[0]); if (use_native_event) { Handle<Event> native_event; if (ConvertFromV8(isolate, event, &native_event)) return native_event->prevent_default(); } v8::Handle<v8::Value> prevent_default = event->GetHiddenValue(StringToSymbol(isolate, "prevent_default")); if (prevent_default.IsEmpty()) return false; return prevent_default->BooleanValue(); } } // namespace mate