// Copyright 2022-present 650 Industries. All rights reserved. #import #import #import @implementation EXJavaScriptObject { /** Pointer to the `EXJavaScriptRuntime` wrapper. \note It must be weak because only then the original runtime can be safely deallocated when the JS engine wants to without unsetting it on each created object. */ __weak EXJavaScriptRuntime *_runtime; /** Shared pointer to the original JSI object that is being wrapped by `EXJavaScriptObject` class. */ std::shared_ptr _jsObjectPtr; } - (nonnull instancetype)initWith:(std::shared_ptr)jsObjectPtr runtime:(nonnull EXJavaScriptRuntime *)runtime { if (self = [super init]) { _runtime = runtime; _jsObjectPtr = jsObjectPtr; } return self; } - (nonnull jsi::Object *)get { return _jsObjectPtr.get(); } #pragma mark - Accessing object properties - (BOOL)hasProperty:(nonnull NSString *)name { return _jsObjectPtr->hasProperty(*[_runtime get], [name UTF8String]); } - (nonnull EXJavaScriptValue *)getProperty:(nonnull NSString *)name { std::shared_ptr value = std::make_shared(_jsObjectPtr->getProperty(*[_runtime get], [name UTF8String])); return [[EXJavaScriptValue alloc] initWithRuntime:_runtime value:value]; } - (nonnull NSArray *)getPropertyNames { jsi::Runtime *runtime = [_runtime get]; jsi::Array propertyNamesArray = _jsObjectPtr->getPropertyNames(*[_runtime get]); return expo::convertJSIArrayToNSArray(*runtime, propertyNamesArray, nullptr); } #pragma mark - Modifying object properties - (void)setProperty:(nonnull NSString *)name value:(nullable id)value { jsi::Value jsiValue = expo::convertObjCObjectToJSIValue(*[_runtime get], value); _jsObjectPtr->setProperty(*[_runtime get], [name UTF8String], jsiValue); } - (void)defineProperty:(nonnull NSString *)name value:(nullable id)value options:(EXJavaScriptObjectPropertyDescriptor)options { jsi::Runtime *runtime = [_runtime get]; jsi::Object global = runtime->global(); jsi::Object objectClass = global.getPropertyAsObject(*runtime, "Object"); jsi::Function definePropertyFunction = objectClass.getPropertyAsFunction(*runtime, "defineProperty"); jsi::Object descriptor = [self preparePropertyDescriptorWithOptions:options]; descriptor.setProperty(*runtime, "value", expo::convertObjCObjectToJSIValue(*runtime, value)); // This call is basically the same as `Object.defineProperty(object, name, descriptor)` in JS definePropertyFunction.callWithThis(*runtime, objectClass, { jsi::Value(*runtime, *_jsObjectPtr.get()), jsi::String::createFromUtf8(*runtime, [name UTF8String]), std::move(descriptor), }); } #pragma mark - Private helpers - (jsi::Object)preparePropertyDescriptorWithOptions:(EXJavaScriptObjectPropertyDescriptor)options { jsi::Runtime *runtime = [_runtime get]; jsi::Object descriptor(*runtime); descriptor.setProperty(*runtime, "configurable", (bool)(options & EXJavaScriptObjectPropertyDescriptorConfigurable)); descriptor.setProperty(*runtime, "enumerable", (bool)(options & EXJavaScriptObjectPropertyDescriptorEnumerable)); descriptor.setProperty(*runtime, "writable", (bool)(options & EXJavaScriptObjectPropertyDescriptorWritable)); return descriptor; } @end