// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #ifndef SHELL_BROWSER_API_EVENT_EMITTER_DEPRECATED_H_ #define SHELL_BROWSER_API_EVENT_EMITTER_DEPRECATED_H_ #include #include #include "base/optional.h" #include "content/public/browser/browser_thread.h" #include "electron/shell/common/api/api.mojom.h" #include "native_mate/wrappable.h" #include "shell/common/api/event_emitter_caller_deprecated.h" namespace content { class RenderFrameHost; } namespace mate { namespace internal { v8::Local CreateJSEvent( v8::Isolate* isolate, v8::Local object, content::RenderFrameHost* sender, base::Optional callback); v8::Local CreateEmptyJSEvent(v8::Isolate* isolate); v8::Local CreateCustomEvent(v8::Isolate* isolate, v8::Local object, v8::Local event); v8::Local CreateEventFromFlags(v8::Isolate* isolate, int flags); } // namespace internal // Provide helperers to emit event in JavaScript. template class EventEmitter : public Wrappable { public: typedef std::vector> ValueArray; // Make the convinient methods visible: // https://isocpp.org/wiki/faq/templates#nondependent-name-lookup-members v8::Isolate* isolate() const { return Wrappable::isolate(); } v8::Local GetWrapper() const { return Wrappable::GetWrapper(); } v8::MaybeLocal GetWrapper(v8::Isolate* isolate) const { return Wrappable::GetWrapper(isolate); } // this.emit(name, event, args...); template bool EmitCustomEvent(base::StringPiece name, v8::Local event, Args&&... args) { return EmitWithEvent( name, internal::CreateCustomEvent(isolate(), GetWrapper(), event), std::forward(args)...); } // this.emit(name, new Event(flags), args...); template bool EmitWithFlags(base::StringPiece name, int flags, Args&&... args) { return EmitCustomEvent(name, internal::CreateEventFromFlags(isolate(), flags), std::forward(args)...); } // this.emit(name, new Event(), args...); template bool Emit(base::StringPiece name, Args&&... args) { return EmitWithSender(name, nullptr, base::nullopt, std::forward(args)...); } // this.emit(name, new Event(sender, message), args...); template bool EmitWithSender( base::StringPiece name, content::RenderFrameHost* sender, base::Optional callback, Args&&... args) { DCHECK_CURRENTLY_ON(content::BrowserThread::UI); v8::Locker locker(isolate()); v8::HandleScope handle_scope(isolate()); v8::Local wrapper = GetWrapper(); if (wrapper.IsEmpty()) { return false; } v8::Local event = internal::CreateJSEvent( isolate(), wrapper, sender, std::move(callback)); return EmitWithEvent(name, event, std::forward(args)...); } protected: EventEmitter() {} private: // this.emit(name, event, args...); template bool EmitWithEvent(base::StringPiece name, v8::Local event, Args&&... args) { DCHECK_CURRENTLY_ON(content::BrowserThread::UI); // It's possible that |this| will be deleted by EmitEvent, so save anything // we need from |this| before calling EmitEvent. auto* isolate = this->isolate(); v8::Locker locker(isolate); v8::HandleScope handle_scope(isolate); auto context = isolate->GetCurrentContext(); EmitEvent(isolate, GetWrapper(), name, event, std::forward(args)...); v8::Local defaultPrevented; if (event->Get(context, StringToV8(isolate, "defaultPrevented")) .ToLocal(&defaultPrevented)) { return defaultPrevented->BooleanValue(isolate); } return false; } DISALLOW_COPY_AND_ASSIGN(EventEmitter); }; } // namespace mate #endif // SHELL_BROWSER_API_EVENT_EMITTER_DEPRECATED_H_