JSMessageEventCustom.cpp [plain text]
#include "config.h"
#include "JSMessageEvent.h"
#include "JSBlob.h"
#include "JSDOMBinding.h"
#include "JSDOMWindow.h"
#include "JSEventTarget.h"
#include "JSMessagePortCustom.h"
#include "MessageEvent.h"
#include <runtime/JSArray.h>
#include <runtime/JSArrayBuffer.h>
using namespace JSC;
namespace WebCore {
JSValue JSMessageEvent::data(ExecState& state) const
{
if (JSValue cachedValue = m_data.get()) {
if (!cachedValue.isObject() || &worldForDOMObject(cachedValue.getObject()) == ¤tWorld(&state))
return cachedValue;
ASSERT_NOT_REACHED();
}
MessageEvent& event = wrapped();
JSValue result;
switch (event.dataType()) {
case MessageEvent::DataTypeScriptValue: {
JSValue dataValue = event.dataAsScriptValue();
if (!dataValue)
result = jsNull();
else {
if (dataValue.isObject() && &worldForDOMObject(dataValue.getObject()) != ¤tWorld(&state)) {
RefPtr<SerializedScriptValue> serializedValue = event.trySerializeData(&state);
if (serializedValue)
result = serializedValue->deserialize(&state, globalObject(), nullptr);
else
result = jsNull();
} else
result = dataValue;
}
break;
}
case MessageEvent::DataTypeSerializedScriptValue:
if (RefPtr<SerializedScriptValue> serializedValue = event.dataAsSerializedScriptValue()) {
MessagePortArray ports = wrapped().ports();
result = serializedValue->deserialize(&state, globalObject(), &ports, NonThrowing);
} else
result = jsNull();
break;
case MessageEvent::DataTypeString:
result = jsStringWithCache(&state, event.dataAsString());
break;
case MessageEvent::DataTypeBlob:
result = toJS(&state, globalObject(), event.dataAsBlob());
break;
case MessageEvent::DataTypeArrayBuffer:
result = toJS(&state, globalObject(), event.dataAsArrayBuffer());
break;
}
m_data.set(state.vm(), this, result);
return result;
}
static JSC::JSValue handleInitMessageEvent(JSMessageEvent* jsEvent, JSC::ExecState& state)
{
const String& typeArg = state.argument(0).toString(&state)->value(&state);
bool canBubbleArg = state.argument(1).toBoolean(&state);
bool cancelableArg = state.argument(2).toBoolean(&state);
const String originArg = state.argument(4).toString(&state)->value(&state);
const String lastEventIdArg = state.argument(5).toString(&state)->value(&state);
DOMWindow* sourceArg = JSDOMWindow::toWrapped(state, state.argument(6));
std::unique_ptr<MessagePortArray> messagePorts;
std::unique_ptr<ArrayBufferArray> arrayBuffers;
if (!state.argument(7).isUndefinedOrNull()) {
messagePorts = std::make_unique<MessagePortArray>();
arrayBuffers = std::make_unique<ArrayBufferArray>();
fillMessagePortArray(state, state.argument(7), *messagePorts, *arrayBuffers);
if (state.hadException())
return jsUndefined();
}
Deprecated::ScriptValue dataArg(state.vm(), state.argument(3));
if (state.hadException())
return jsUndefined();
MessageEvent& event = jsEvent->wrapped();
event.initMessageEvent(typeArg, canBubbleArg, cancelableArg, dataArg, originArg, lastEventIdArg, sourceArg, WTFMove(messagePorts));
jsEvent->m_data.set(state.vm(), jsEvent, dataArg.jsValue());
return jsUndefined();
}
JSC::JSValue JSMessageEvent::initMessageEvent(JSC::ExecState& state)
{
return handleInitMessageEvent(this, state);
}
JSC::JSValue JSMessageEvent::webkitInitMessageEvent(JSC::ExecState& state)
{
return handleInitMessageEvent(this, state);
}
}