Add ipc.sendSync in renderer.
This commit is contained in:
parent
8f0b2b6363
commit
1b87a31940
10 changed files with 132 additions and 4 deletions
|
@ -50,7 +50,7 @@ void AtomBrowserBindings::OnRendererMessage(int process_id,
|
||||||
|
|
||||||
scoped_ptr<V8ValueConverter> converter(new V8ValueConverterImpl());
|
scoped_ptr<V8ValueConverter> converter(new V8ValueConverterImpl());
|
||||||
|
|
||||||
// process.emit('ATOM_INTERNAL_MESSAGE', 'message', process_id, routing_id);
|
// process.emit(channel, 'message', process_id, routing_id);
|
||||||
std::vector<v8::Handle<v8::Value>> arguments;
|
std::vector<v8::Handle<v8::Value>> arguments;
|
||||||
arguments.reserve(3 + args.GetSize());
|
arguments.reserve(3 + args.GetSize());
|
||||||
arguments.push_back(v8::String::New(channel.c_str(), channel.size()));
|
arguments.push_back(v8::String::New(channel.c_str(), channel.size()));
|
||||||
|
@ -69,4 +69,43 @@ void AtomBrowserBindings::OnRendererMessage(int process_id,
|
||||||
node::MakeCallback(node::process, "emit", arguments.size(), &arguments[0]);
|
node::MakeCallback(node::process, "emit", arguments.size(), &arguments[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void AtomBrowserBindings::OnRendererMessageSync(
|
||||||
|
int process_id,
|
||||||
|
int routing_id,
|
||||||
|
const std::string& channel,
|
||||||
|
const base::ListValue& args,
|
||||||
|
base::DictionaryValue* result) {
|
||||||
|
v8::HandleScope scope;
|
||||||
|
|
||||||
|
v8::Handle<v8::Context> context = v8::Context::GetCurrent();
|
||||||
|
|
||||||
|
scoped_ptr<V8ValueConverter> converter(new V8ValueConverterImpl());
|
||||||
|
|
||||||
|
v8::Handle<v8::Object> event = v8::Object::New();
|
||||||
|
|
||||||
|
// process.emit(channel, 'sync-message', event, process_id, routing_id);
|
||||||
|
std::vector<v8::Handle<v8::Value>> arguments;
|
||||||
|
arguments.reserve(3 + args.GetSize());
|
||||||
|
arguments.push_back(v8::String::New(channel.c_str(), channel.size()));
|
||||||
|
const base::Value* value;
|
||||||
|
if (args.Get(0, &value))
|
||||||
|
arguments.push_back(converter->ToV8Value(value, context));
|
||||||
|
arguments.push_back(event);
|
||||||
|
arguments.push_back(v8::Integer::New(process_id));
|
||||||
|
arguments.push_back(v8::Integer::New(routing_id));
|
||||||
|
|
||||||
|
for (size_t i = 1; i < args.GetSize(); i++) {
|
||||||
|
const base::Value* value;
|
||||||
|
if (args.Get(i, &value))
|
||||||
|
arguments.push_back(converter->ToV8Value(value, context));
|
||||||
|
}
|
||||||
|
|
||||||
|
node::MakeCallback(node::process, "emit", arguments.size(), &arguments[0]);
|
||||||
|
|
||||||
|
scoped_ptr<base::Value> base_event(converter->FromV8Value(event, context));
|
||||||
|
DCHECK(base_event && base_event->IsType(base::Value::TYPE_DICTIONARY));
|
||||||
|
|
||||||
|
result->Swap(static_cast<base::DictionaryValue*>(base_event.get()));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace atom
|
} // namespace atom
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
#include "common/api/atom_bindings.h"
|
#include "common/api/atom_bindings.h"
|
||||||
|
|
||||||
namespace base {
|
namespace base {
|
||||||
|
class DictionaryValue;
|
||||||
class ListValue;
|
class ListValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,6 +30,13 @@ class AtomBrowserBindings : public AtomBindings {
|
||||||
const std::string& channel,
|
const std::string& channel,
|
||||||
const base::ListValue& args);
|
const base::ListValue& args);
|
||||||
|
|
||||||
|
// Called when received a synchronous message from renderer.
|
||||||
|
void OnRendererMessageSync(int process_id,
|
||||||
|
int routing_id,
|
||||||
|
const std::string& channel,
|
||||||
|
const base::ListValue& args,
|
||||||
|
base::DictionaryValue* result);
|
||||||
|
|
||||||
// The require('atom').browserMainParts object.
|
// The require('atom').browserMainParts object.
|
||||||
v8::Handle<v8::Object> browser_main_parts() {
|
v8::Handle<v8::Object> browser_main_parts() {
|
||||||
return browser_main_parts_;
|
return browser_main_parts_;
|
||||||
|
|
|
@ -5,6 +5,8 @@ class Ipc extends EventEmitter
|
||||||
constructor: ->
|
constructor: ->
|
||||||
process.on 'ATOM_INTERNAL_MESSAGE', (args...) =>
|
process.on 'ATOM_INTERNAL_MESSAGE', (args...) =>
|
||||||
@emit(args...)
|
@emit(args...)
|
||||||
|
process.on 'ATOM_INTERNAL_MESSAGE_SYNC', (args...) =>
|
||||||
|
@emit(args...)
|
||||||
|
|
||||||
send: (process_id, routing_id, args...) ->
|
send: (process_id, routing_id, args...) ->
|
||||||
@sendChannel(process_id, routing_id, 'message', args...)
|
@sendChannel(process_id, routing_id, 'message', args...)
|
||||||
|
|
|
@ -9,6 +9,10 @@ ipc.on('message', function(process_id, routing_id) {
|
||||||
ipc.send.apply(ipc, arguments);
|
ipc.send.apply(ipc, arguments);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
ipc.on('sync-message', function(event, process_id, routing_id) {
|
||||||
|
event.result = arguments;
|
||||||
|
});
|
||||||
|
|
||||||
atom.browserMainParts.preMainMessageLoopRun = function() {
|
atom.browserMainParts.preMainMessageLoopRun = function() {
|
||||||
mainWindow = new Window({ width: 800, height: 600 });
|
mainWindow = new Window({ width: 800, height: 600 });
|
||||||
mainWindow.url = 'file://' + __dirname + '/index.html';
|
mainWindow.url = 'file://' + __dirname + '/index.html';
|
||||||
|
|
|
@ -134,6 +134,7 @@ bool NativeWindow::OnMessageReceived(const IPC::Message& message) {
|
||||||
bool handled = true;
|
bool handled = true;
|
||||||
IPC_BEGIN_MESSAGE_MAP(NativeWindow, message)
|
IPC_BEGIN_MESSAGE_MAP(NativeWindow, message)
|
||||||
IPC_MESSAGE_HANDLER(AtomViewHostMsg_Message, OnRendererMessage)
|
IPC_MESSAGE_HANDLER(AtomViewHostMsg_Message, OnRendererMessage)
|
||||||
|
IPC_MESSAGE_HANDLER(AtomViewHostMsg_Message_Sync, OnRendererMessageSync)
|
||||||
IPC_MESSAGE_UNHANDLED(handled = false)
|
IPC_MESSAGE_UNHANDLED(handled = false)
|
||||||
IPC_END_MESSAGE_MAP()
|
IPC_END_MESSAGE_MAP()
|
||||||
|
|
||||||
|
@ -169,4 +170,15 @@ void NativeWindow::OnRendererMessage(const std::string& channel,
|
||||||
args);
|
args);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void NativeWindow::OnRendererMessageSync(const std::string& channel,
|
||||||
|
const base::ListValue& args,
|
||||||
|
base::DictionaryValue* result) {
|
||||||
|
AtomBrowserMainParts::Get()->atom_bindings()->OnRendererMessageSync(
|
||||||
|
GetWebContents()->GetRenderProcessHost()->GetID(),
|
||||||
|
GetWebContents()->GetRoutingID(),
|
||||||
|
channel,
|
||||||
|
args,
|
||||||
|
result);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace atom
|
} // namespace atom
|
||||||
|
|
|
@ -123,6 +123,10 @@ class NativeWindow : public content::WebContentsDelegate,
|
||||||
void OnRendererMessage(const std::string& channel,
|
void OnRendererMessage(const std::string& channel,
|
||||||
const base::ListValue& args);
|
const base::ListValue& args);
|
||||||
|
|
||||||
|
void OnRendererMessageSync(const std::string& channel,
|
||||||
|
const base::ListValue& args,
|
||||||
|
base::DictionaryValue* result);
|
||||||
|
|
||||||
// Notification manager.
|
// Notification manager.
|
||||||
content::NotificationRegistrar registrar_;
|
content::NotificationRegistrar registrar_;
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,16 @@ IPC_MESSAGE_ROUTED2(AtomViewHostMsg_Message,
|
||||||
std::string /* channel */,
|
std::string /* channel */,
|
||||||
ListValue /* arguments */)
|
ListValue /* arguments */)
|
||||||
|
|
||||||
|
IPC_SYNC_MESSAGE_ROUTED2_1(AtomViewHostMsg_Message_Sync,
|
||||||
|
std::string /* channel */,
|
||||||
|
ListValue /* arguments */,
|
||||||
|
DictionaryValue /* result */)
|
||||||
|
|
||||||
IPC_MESSAGE_ROUTED2(AtomViewMsg_Message,
|
IPC_MESSAGE_ROUTED2(AtomViewMsg_Message,
|
||||||
std::string /* channel */,
|
std::string /* channel */,
|
||||||
ListValue /* arguments */)
|
ListValue /* arguments */)
|
||||||
|
|
||||||
|
IPC_SYNC_MESSAGE_ROUTED2_1(AtomViewMsg_Message_Sync,
|
||||||
|
std::string /* channel */,
|
||||||
|
ListValue /* arguments */,
|
||||||
|
DictionaryValue /* result */)
|
||||||
|
|
|
@ -63,17 +63,57 @@ v8::Handle<v8::Value> RendererIPC::Send(const v8::Arguments &args) {
|
||||||
|
|
||||||
DCHECK(arguments && arguments->IsType(base::Value::TYPE_LIST));
|
DCHECK(arguments && arguments->IsType(base::Value::TYPE_LIST));
|
||||||
|
|
||||||
render_view->Send(new AtomViewHostMsg_Message(
|
bool success = render_view->Send(new AtomViewHostMsg_Message(
|
||||||
render_view->GetRoutingID(),
|
render_view->GetRoutingID(),
|
||||||
channel,
|
channel,
|
||||||
*static_cast<base::ListValue*>(arguments.get())));
|
*static_cast<base::ListValue*>(arguments.get())));
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
return node::ThrowError("Unable to send AtomViewHostMsg_Message");
|
||||||
|
|
||||||
return v8::Undefined();
|
return v8::Undefined();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
v8::Handle<v8::Value> RendererIPC::SendSync(const v8::Arguments &args) {
|
||||||
|
v8::HandleScope scope;
|
||||||
|
|
||||||
|
if (!args[0]->IsString())
|
||||||
|
return node::ThrowTypeError("Bad argument");
|
||||||
|
|
||||||
|
v8::Handle<v8::Context> context = v8::Context::GetCurrent();
|
||||||
|
std::string channel(*v8::String::Utf8Value(args[0]));
|
||||||
|
|
||||||
|
// Convert Arguments to Array, so we can use V8ValueConverter to convert it
|
||||||
|
// to ListValue.
|
||||||
|
v8::Local<v8::Array> v8_args = v8::Array::New(args.Length() - 1);
|
||||||
|
for (int i = 0; i < args.Length() - 1; ++i)
|
||||||
|
v8_args->Set(i, args[i + 1]);
|
||||||
|
|
||||||
|
scoped_ptr<V8ValueConverter> converter(V8ValueConverter::create());
|
||||||
|
scoped_ptr<base::Value> arguments(converter->FromV8Value(v8_args, context));
|
||||||
|
|
||||||
|
DCHECK(arguments && arguments->IsType(base::Value::TYPE_LIST));
|
||||||
|
|
||||||
|
RenderView* render_view = GetCurrentRenderView();
|
||||||
|
|
||||||
|
base::DictionaryValue result;
|
||||||
|
bool success = render_view->Send(new AtomViewHostMsg_Message_Sync(
|
||||||
|
render_view->GetRoutingID(),
|
||||||
|
channel,
|
||||||
|
*static_cast<base::ListValue*>(arguments.get()),
|
||||||
|
&result));
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
return node::ThrowError("Unable to send AtomViewHostMsg_Message_Sync");
|
||||||
|
|
||||||
|
return scope.Close(converter->ToV8Value(&result, context));
|
||||||
|
}
|
||||||
|
|
||||||
// static
|
// static
|
||||||
void RendererIPC::Initialize(v8::Handle<v8::Object> target) {
|
void RendererIPC::Initialize(v8::Handle<v8::Object> target) {
|
||||||
node::SetMethod(target, "send", Send);
|
node::SetMethod(target, "send", Send);
|
||||||
|
node::SetMethod(target, "sendSync", SendSync);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|
|
@ -18,6 +18,7 @@ class RendererIPC {
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static v8::Handle<v8::Value> Send(const v8::Arguments &args);
|
static v8::Handle<v8::Value> Send(const v8::Arguments &args);
|
||||||
|
static v8::Handle<v8::Value> SendSync(const v8::Arguments &args);
|
||||||
|
|
||||||
DISALLOW_IMPLICIT_CONSTRUCTORS(RendererIPC);
|
DISALLOW_IMPLICIT_CONSTRUCTORS(RendererIPC);
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,15 +1,23 @@
|
||||||
EventEmitter = require('events').EventEmitter
|
EventEmitter = require('events').EventEmitter
|
||||||
send = process.atom_binding('ipc').send
|
ipc = process.atom_binding('ipc')
|
||||||
|
|
||||||
class Ipc extends EventEmitter
|
class Ipc extends EventEmitter
|
||||||
constructor: ->
|
constructor: ->
|
||||||
process.on 'ATOM_INTERNAL_MESSAGE', (args...) =>
|
process.on 'ATOM_INTERNAL_MESSAGE', (args...) =>
|
||||||
@emit(args...)
|
@emit(args...)
|
||||||
|
process.on 'ATOM_INTERNAL_MESSAGE_SYNC', (args...) =>
|
||||||
|
@emit(args...)
|
||||||
|
|
||||||
send: (args...) ->
|
send: (args...) ->
|
||||||
@sendChannel('message', args...)
|
@sendChannel('message', args...)
|
||||||
|
|
||||||
sendChannel: (args...) ->
|
sendChannel: (args...) ->
|
||||||
send('ATOM_INTERNAL_MESSAGE', args...)
|
ipc.send('ATOM_INTERNAL_MESSAGE', args...)
|
||||||
|
|
||||||
|
sendSync: (args...) ->
|
||||||
|
ipc.sendSync('ATOM_INTERNAL_MESSAGE_SYNC', 'sync-message', args...).result
|
||||||
|
|
||||||
|
sendChannelSync: (args...) ->
|
||||||
|
ipc.sendSync('ATOM_INTERNAL_MESSAGE_SYNC', args...).result
|
||||||
|
|
||||||
module.exports = new Ipc
|
module.exports = new Ipc
|
||||||
|
|
Loading…
Reference in a new issue