|
@@ -8,7 +8,12 @@
|
|
|
|
|
|
#include "content/public/browser/render_frame_host.h"
|
|
|
#include "content/public/browser/render_process_host.h"
|
|
|
+#include "gin/handle.h"
|
|
|
#include "mojo/public/cpp/bindings/self_owned_receiver.h"
|
|
|
+#include "shell/browser/api/electron_api_session.h"
|
|
|
+#include "shell/common/gin_converters/content_converter.h"
|
|
|
+#include "shell/common/gin_converters/frame_converter.h"
|
|
|
+#include "shell/common/gin_helper/event.h"
|
|
|
|
|
|
namespace electron {
|
|
|
ElectronApiIPCHandlerImpl::ElectronApiIPCHandlerImpl(
|
|
@@ -38,57 +43,128 @@ void ElectronApiIPCHandlerImpl::OnConnectionError() {
|
|
|
void ElectronApiIPCHandlerImpl::Message(bool internal,
|
|
|
const std::string& channel,
|
|
|
blink::CloneableMessage arguments) {
|
|
|
- api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
- if (api_web_contents) {
|
|
|
- api_web_contents->Message(internal, channel, std::move(arguments),
|
|
|
- GetRenderFrameHost());
|
|
|
- }
|
|
|
+ auto* session = GetSession();
|
|
|
+ v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
|
+ v8::HandleScope handle_scope(isolate);
|
|
|
+ auto event = MakeIPCEvent(isolate, session, internal);
|
|
|
+ if (event.IsEmpty())
|
|
|
+ return;
|
|
|
+ session->Message(event, channel, std::move(arguments));
|
|
|
}
|
|
|
void ElectronApiIPCHandlerImpl::Invoke(bool internal,
|
|
|
const std::string& channel,
|
|
|
blink::CloneableMessage arguments,
|
|
|
InvokeCallback callback) {
|
|
|
- api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
- if (api_web_contents) {
|
|
|
- api_web_contents->Invoke(internal, channel, std::move(arguments),
|
|
|
- std::move(callback), GetRenderFrameHost());
|
|
|
- }
|
|
|
+ auto* session = GetSession();
|
|
|
+ v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
|
+ v8::HandleScope handle_scope(isolate);
|
|
|
+ auto event = MakeIPCEvent(isolate, session, internal, std::move(callback));
|
|
|
+ if (event.IsEmpty())
|
|
|
+ return;
|
|
|
+ session->Invoke(event, channel, std::move(arguments));
|
|
|
}
|
|
|
|
|
|
void ElectronApiIPCHandlerImpl::ReceivePostMessage(
|
|
|
const std::string& channel,
|
|
|
blink::TransferableMessage message) {
|
|
|
- api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
- if (api_web_contents) {
|
|
|
- api_web_contents->ReceivePostMessage(channel, std::move(message),
|
|
|
- GetRenderFrameHost());
|
|
|
- }
|
|
|
+ auto* session = GetSession();
|
|
|
+ v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
|
+ v8::HandleScope handle_scope(isolate);
|
|
|
+ auto event = MakeIPCEvent(isolate, session, false);
|
|
|
+ if (event.IsEmpty())
|
|
|
+ return;
|
|
|
+ session->ReceivePostMessage(event, channel, std::move(message));
|
|
|
}
|
|
|
|
|
|
void ElectronApiIPCHandlerImpl::MessageSync(bool internal,
|
|
|
const std::string& channel,
|
|
|
blink::CloneableMessage arguments,
|
|
|
MessageSyncCallback callback) {
|
|
|
- api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
- if (api_web_contents) {
|
|
|
- api_web_contents->MessageSync(internal, channel, std::move(arguments),
|
|
|
- std::move(callback), GetRenderFrameHost());
|
|
|
- }
|
|
|
+ auto* session = GetSession();
|
|
|
+ v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
|
+ v8::HandleScope handle_scope(isolate);
|
|
|
+ auto event = MakeIPCEvent(isolate, session, internal, std::move(callback));
|
|
|
+ if (event.IsEmpty())
|
|
|
+ return;
|
|
|
+ session->MessageSync(event, channel, std::move(arguments));
|
|
|
}
|
|
|
|
|
|
void ElectronApiIPCHandlerImpl::MessageHost(const std::string& channel,
|
|
|
blink::CloneableMessage arguments) {
|
|
|
- api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
- if (api_web_contents) {
|
|
|
- api_web_contents->MessageHost(channel, std::move(arguments),
|
|
|
- GetRenderFrameHost());
|
|
|
- }
|
|
|
+ auto* session = GetSession();
|
|
|
+ v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
|
+ v8::HandleScope handle_scope(isolate);
|
|
|
+ auto event = MakeIPCEvent(isolate, session, false);
|
|
|
+ if (event.IsEmpty())
|
|
|
+ return;
|
|
|
+ session->MessageHost(event, channel, std::move(arguments));
|
|
|
}
|
|
|
|
|
|
content::RenderFrameHost* ElectronApiIPCHandlerImpl::GetRenderFrameHost() {
|
|
|
return content::RenderFrameHost::FromID(render_frame_host_id_);
|
|
|
}
|
|
|
|
|
|
+api::Session* ElectronApiIPCHandlerImpl::GetSession() {
|
|
|
+ auto* rfh = GetRenderFrameHost();
|
|
|
+ return rfh ? api::Session::FromBrowserContext(rfh->GetBrowserContext())
|
|
|
+ : nullptr;
|
|
|
+}
|
|
|
+
|
|
|
+gin::Handle<gin_helper::internal::Event>
|
|
|
+ElectronApiIPCHandlerImpl::MakeIPCEvent(
|
|
|
+ v8::Isolate* isolate,
|
|
|
+ api::Session* session,
|
|
|
+ bool internal,
|
|
|
+ electron::mojom::ElectronApiIPC::InvokeCallback callback) {
|
|
|
+ if (!session) {
|
|
|
+ if (callback) {
|
|
|
+ // We must always invoke the callback if present.
|
|
|
+ gin_helper::internal::ReplyChannel::Create(isolate, std::move(callback))
|
|
|
+ ->SendError("Session does not exist");
|
|
|
+ }
|
|
|
+ return {};
|
|
|
+ }
|
|
|
+
|
|
|
+ api::WebContents* api_web_contents = api::WebContents::From(web_contents());
|
|
|
+ if (!api_web_contents) {
|
|
|
+ if (callback) {
|
|
|
+ // We must always invoke the callback if present.
|
|
|
+ gin_helper::internal::ReplyChannel::Create(isolate, std::move(callback))
|
|
|
+ ->SendError("WebContents does not exist");
|
|
|
+ }
|
|
|
+ return {};
|
|
|
+ }
|
|
|
+
|
|
|
+ v8::Local<v8::Object> wrapper;
|
|
|
+ if (!api_web_contents->GetWrapper(isolate).ToLocal(&wrapper)) {
|
|
|
+ if (callback) {
|
|
|
+ // We must always invoke the callback if present.
|
|
|
+ gin_helper::internal::ReplyChannel::Create(isolate, std::move(callback))
|
|
|
+ ->SendError("WebContents was destroyed");
|
|
|
+ }
|
|
|
+ return {};
|
|
|
+ }
|
|
|
+
|
|
|
+ content::RenderFrameHost* frame = GetRenderFrameHost();
|
|
|
+ gin::Handle<gin_helper::internal::Event> event =
|
|
|
+ gin_helper::internal::Event::New(isolate);
|
|
|
+ gin_helper::Dictionary dict(isolate, event.ToV8().As<v8::Object>());
|
|
|
+ dict.Set("type", "frame");
|
|
|
+ dict.Set("sender", web_contents());
|
|
|
+ if (internal)
|
|
|
+ dict.SetHidden("internal", internal);
|
|
|
+ if (callback)
|
|
|
+ dict.Set("_replyChannel", gin_helper::internal::ReplyChannel::Create(
|
|
|
+ isolate, std::move(callback)));
|
|
|
+ if (frame) {
|
|
|
+ dict.SetGetter("senderFrame", frame);
|
|
|
+ dict.Set("frameId", frame->GetRoutingID());
|
|
|
+ dict.Set("processId", frame->GetProcess()->GetID().GetUnsafeValue());
|
|
|
+ dict.Set("frameTreeNodeId", frame->GetFrameTreeNodeId());
|
|
|
+ }
|
|
|
+ return event;
|
|
|
+}
|
|
|
+
|
|
|
// static
|
|
|
void ElectronApiIPCHandlerImpl::Create(
|
|
|
content::RenderFrameHost* frame_host,
|