mirror of
https://github.com/status-im/react-native.git
synced 2025-01-26 09:19:10 +00:00
8a0640716b
Summary: Adds: - callAsConstructor - Ability to provide `this` object to callAsFunction - getPropertyNames() now returns Strings (which can be converted by the caller to std::string if they want). Fixes: - double free issue with the String move constructor Reviewed By: lexs Differential Revision: D3515398 fbshipit-source-id: afa1342044e41fdd833dd27b8a244a58d4078442
182 lines
5.7 KiB
C++
182 lines
5.7 KiB
C++
// Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
#include <folly/json.h>
|
|
|
|
#include "Value.h"
|
|
|
|
#include "JSCHelpers.h"
|
|
|
|
namespace facebook {
|
|
namespace react {
|
|
|
|
Value::Value(JSContextRef context, JSValueRef value) :
|
|
m_context(context),
|
|
m_value(value)
|
|
{
|
|
}
|
|
|
|
Value::Value(JSContextRef context, JSStringRef str) :
|
|
m_context(context),
|
|
m_value(JSValueMakeString(context, str))
|
|
{
|
|
}
|
|
|
|
Value::Value(Value&& other) :
|
|
m_context(other.m_context),
|
|
m_value(other.m_value)
|
|
{
|
|
other.m_value = nullptr;
|
|
}
|
|
|
|
JSContextRef Value::context() const {
|
|
return m_context;
|
|
}
|
|
|
|
std::string Value::toJSONString(unsigned indent) const throw(JSException) {
|
|
JSValueRef exn;
|
|
auto stringToAdopt = JSValueCreateJSONString(m_context, m_value, indent, &exn);
|
|
if (stringToAdopt == nullptr) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Exception creating JSON string: %s", exceptionText.c_str());
|
|
}
|
|
return String::adopt(stringToAdopt).str();
|
|
}
|
|
|
|
/* static */
|
|
Value Value::fromJSON(JSContextRef ctx, const String& json) throw(JSException) {
|
|
auto result = JSValueMakeFromJSONString(ctx, json);
|
|
if (!result) {
|
|
throwJSExecutionException("Failed to create String from JSON");
|
|
}
|
|
return Value(ctx, result);
|
|
}
|
|
|
|
Value Value::fromDynamic(JSContextRef ctx, folly::dynamic value) throw(JSException) {
|
|
auto json = folly::toJson(value);
|
|
return fromJSON(ctx, String(json.c_str()));
|
|
}
|
|
|
|
Object Value::asObject() {
|
|
JSValueRef exn;
|
|
JSObjectRef jsObj = JSValueToObject(context(), m_value, &exn);
|
|
if (!jsObj) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Failed to convert to object: %s", exceptionText.c_str());
|
|
}
|
|
Object ret = Object(context(), jsObj);
|
|
m_value = nullptr;
|
|
return ret;
|
|
}
|
|
|
|
Object::operator Value() const {
|
|
return Value(m_context, m_obj);
|
|
}
|
|
|
|
Value Object::callAsFunction(std::initializer_list<JSValueRef> args) const {
|
|
return callAsFunction(nullptr, args.size(), args.begin());
|
|
}
|
|
|
|
Value Object::callAsFunction(const Object& thisObj, std::initializer_list<JSValueRef> args) const {
|
|
return callAsFunction((JSObjectRef) thisObj, args.size(), args.begin());
|
|
}
|
|
|
|
Value Object::callAsFunction(int nArgs, const JSValueRef args[]) const {
|
|
return callAsFunction(nullptr, nArgs, args);
|
|
}
|
|
|
|
Value Object::callAsFunction(const Object& thisObj, int nArgs, const JSValueRef args[]) const {
|
|
return callAsFunction((JSObjectRef) thisObj, nArgs, args);
|
|
}
|
|
|
|
Value Object::callAsFunction(JSObjectRef thisObj, int nArgs, const JSValueRef args[]) const {
|
|
JSValueRef exn;
|
|
JSValueRef result = JSObjectCallAsFunction(m_context, m_obj, thisObj, nArgs, args, &exn);
|
|
if (!result) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Exception calling object as function: %s", exceptionText.c_str());
|
|
}
|
|
return Value(m_context, result);
|
|
}
|
|
|
|
Object Object::callAsConstructor(std::initializer_list<JSValueRef> args) const {
|
|
JSValueRef exn;
|
|
JSObjectRef result = JSObjectCallAsConstructor(m_context, m_obj, args.size(), args.begin(), &exn);
|
|
if (!result) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Exception calling object as constructor: %s", exceptionText.c_str());
|
|
}
|
|
return Object(m_context, result);
|
|
}
|
|
|
|
Value Object::getProperty(const String& propName) const {
|
|
JSValueRef exn;
|
|
JSValueRef property = JSObjectGetProperty(m_context, m_obj, propName, &exn);
|
|
if (!property) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Failed to get property: %s", exceptionText.c_str());
|
|
}
|
|
return Value(m_context, property);
|
|
}
|
|
|
|
Value Object::getPropertyAtIndex(unsigned index) const {
|
|
JSValueRef exn;
|
|
JSValueRef property = JSObjectGetPropertyAtIndex(m_context, m_obj, index, &exn);
|
|
if (!property) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Failed to get property at index %u: %s", index, exceptionText.c_str());
|
|
}
|
|
return Value(m_context, property);
|
|
}
|
|
|
|
Value Object::getProperty(const char *propName) const {
|
|
return getProperty(String(propName));
|
|
}
|
|
|
|
void Object::setProperty(const String& propName, const Value& value) const {
|
|
JSValueRef exn = NULL;
|
|
JSObjectSetProperty(m_context, m_obj, propName, value, kJSPropertyAttributeNone, &exn);
|
|
if (exn) {
|
|
std::string exceptionText = Value(m_context, exn).toString().str();
|
|
throwJSExecutionException("Failed to set property: %s", exceptionText.c_str());
|
|
}
|
|
}
|
|
|
|
void Object::setProperty(const char *propName, const Value& value) const {
|
|
setProperty(String(propName), value);
|
|
}
|
|
|
|
std::vector<String> Object::getPropertyNames() const {
|
|
auto namesRef = JSObjectCopyPropertyNames(m_context, m_obj);
|
|
size_t count = JSPropertyNameArrayGetCount(namesRef);
|
|
std::vector<String> names;
|
|
names.reserve(count);
|
|
for (size_t i = 0; i < count; i++) {
|
|
names.emplace_back(String::ref(JSPropertyNameArrayGetNameAtIndex(namesRef, i)));
|
|
}
|
|
JSPropertyNameArrayRelease(namesRef);
|
|
return names;
|
|
}
|
|
|
|
std::unordered_map<std::string, std::string> Object::toJSONMap() const {
|
|
std::unordered_map<std::string, std::string> map;
|
|
auto namesRef = JSObjectCopyPropertyNames(m_context, m_obj);
|
|
size_t count = JSPropertyNameArrayGetCount(namesRef);
|
|
for (size_t i = 0; i < count; i++) {
|
|
auto key = String::ref(JSPropertyNameArrayGetNameAtIndex(namesRef, i));
|
|
map.emplace(key.str(), getProperty(key).toJSONString());
|
|
}
|
|
JSPropertyNameArrayRelease(namesRef);
|
|
return map;
|
|
}
|
|
|
|
/* static */
|
|
Object Object::create(JSContextRef ctx) {
|
|
JSObjectRef newObj = JSObjectMake(
|
|
ctx,
|
|
NULL, // create instance of default object class
|
|
NULL); // no private data
|
|
return Object(ctx, newObj);
|
|
}
|
|
|
|
} }
|