mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-26 04:41:03 +00:00
400 lines
17 KiB
C++
400 lines
17 KiB
C++
/****************************************************************************
|
|
Copyright (c) 2016 Chukong Technologies Inc.
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
****************************************************************************/
|
|
#pragma once
|
|
|
|
#include "../config.hpp"
|
|
|
|
#if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|
|
|
|
#include "Base.h"
|
|
#include "../Value.hpp"
|
|
#include "../RefCounter.hpp"
|
|
|
|
namespace se {
|
|
|
|
class Class;
|
|
|
|
/**
|
|
* se::Object represents JavaScript Object.
|
|
*/
|
|
class Object final : public RefCounter
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Creates a JavaScript Object like `{} or new Object()`.
|
|
* @return A JavaScript Object, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createPlainObject();
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Array Object like `[] or new Array()`.
|
|
* @param[in] length The initical length of array.
|
|
* @return A JavaScript Array Object, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createArrayObject(size_t length);
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Typed Array Object with uint8 format from an existing pointer.
|
|
* @param[in] bytes A pointer to the byte buffer to be used as the backing store of the Typed Array object.
|
|
* @param[in] byteLength The number of bytes pointed to by the parameter bytes.
|
|
* @return A JavaScript Typed Array Object whose backing store is the same as the one pointed data, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
* @deprecated This method is deprecated, please use `se::Object::createTypedArray` instead.
|
|
*/
|
|
SE_DEPRECATED_ATTRIBUTE static Object* createUint8TypedArray(uint8_t* data, size_t byteLength);
|
|
|
|
enum class TypedArrayType
|
|
{
|
|
NONE,
|
|
INT8,
|
|
INT16,
|
|
INT32,
|
|
UINT8,
|
|
UINT8_CLAMPED,
|
|
UINT16,
|
|
UINT32,
|
|
FLOAT32,
|
|
FLOAT64
|
|
};
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Typed Array Object with specified format from an existing pointer,
|
|
if provide a null pointer,then will create a empty JavaScript Typed Array Object.
|
|
* @param[in] type The format of typed array.
|
|
* @param[in] data A pointer to the byte buffer to be used as the backing store of the Typed Array object.
|
|
* @param[in] byteLength The number of bytes pointed to by the parameter bytes.
|
|
* @return A JavaScript Typed Array Object whose backing store is the same as the one pointed data, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createTypedArray(TypedArrayType type, void* data, size_t byteLength);
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Array Buffer object from an existing pointer.
|
|
* @param[in] bytes A pointer to the byte buffer to be used as the backing store of the Typed Array object.
|
|
* @param[in] byteLength The number of bytes pointed to by the parameter bytes.
|
|
* @return A Array Buffer Object whose backing store is the same as the one pointed to data, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createArrayBufferObject(void* data, size_t byteLength);
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Object from a JSON formatted string.
|
|
* @param[in] jsonStr The utf-8 string containing the JSON string to be parsed.
|
|
* @return A JavaScript Object containing the parsed value, or nullptr if the input is invalid.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createJSONObject(const std::string& jsonStr);
|
|
|
|
/**
|
|
* @brief Creates a JavaScript Native Binding Object from an existing se::Class instance.
|
|
* @param[in] cls The se::Class instance which stores native callback informations.
|
|
* @return A JavaScript Native Binding Object, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* createObjectWithClass(Class* cls);
|
|
|
|
/**
|
|
* @brief Gets a se::Object from an existing native object pointer.
|
|
* @param[in] ptr The native object pointer associated with the se::Object
|
|
* @return A JavaScript Native Binding Object, or nullptr if there is an error.
|
|
* @note The return value (non-null) has to be released manually.
|
|
*/
|
|
static Object* getObjectWithPtr(void* ptr);
|
|
|
|
/**
|
|
* @brief Gets a property from an object.
|
|
* @param[in] name A utf-8 string containing the property's name.
|
|
* @param[out] value The property's value if object has the property, otherwise the undefined value.
|
|
* @return true if object has the property, otherwise false.
|
|
*/
|
|
bool getProperty(const char* name, Value* value);
|
|
|
|
/**
|
|
* @brief Sets a property to an object.
|
|
* @param[in] name A utf-8 string containing the property's name.
|
|
* @param[in] value A value to be used as the property's value.
|
|
* @return true if the property is set successfully, otherwise false.
|
|
*/
|
|
bool setProperty(const char* name, const Value& value);
|
|
|
|
/**
|
|
* @brief Defines a property with native accessor callbacks for an object.
|
|
* @param[in] name A utf-8 string containing the property's name.
|
|
* @param[in] getter The native callback for getter.
|
|
* @param[in] setter The native callback for setter.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool defineProperty(const char *name, JSNative getter, JSNative setter);
|
|
|
|
/**
|
|
* @brief Defines a function with a native callback for an object.
|
|
* @param[in] funcName A utf-8 string containing the function name.
|
|
* @param[in] func The native callback triggered by JavaScript code.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool defineFunction(const char *funcName, JSNative func);
|
|
|
|
/**
|
|
* @brief Tests whether an object can be called as a function.
|
|
* @return true if object can be called as a function, otherwise false.
|
|
*/
|
|
bool isFunction() const;
|
|
|
|
/**
|
|
* @brief Calls an object as a function.
|
|
* @param[in] args A se::Value array of arguments to pass to the function. Pass se::EmptyValueArray if argumentCount is 0.
|
|
* @param[in] thisObject The object to use as "this," or NULL to use the global object as "this."
|
|
* @param[out] rval The se::Value that results from calling object as a function, passing nullptr if return value is ignored.
|
|
* @return true if object is a function and there isn't any errors, otherwise false.
|
|
*/
|
|
bool call(const ValueArray& args, Object* thisObject, Value* rval = nullptr);
|
|
|
|
/**
|
|
* @brief Tests whether an object is an array.
|
|
* @return true if object is an array, otherwise false.
|
|
*/
|
|
bool isArray() const;
|
|
|
|
/**
|
|
* @brief Gets array length of an array object.
|
|
* @param[out] length The array length to be stored. It's set to 0 if there is an error.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool getArrayLength(uint32_t* length) const;
|
|
|
|
/**
|
|
* @brief Gets an element from an array object by numeric index.
|
|
* @param[in] index An integer value for index.
|
|
* @param[out] data The se::Value to be stored for the element in certain index.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool getArrayElement(uint32_t index, Value* data) const;
|
|
|
|
/**
|
|
* @brief Sets an element to an array object by numeric index.
|
|
* @param[in] index An integer value for index.
|
|
* @param[in] data The se::Value to be set to array with certain index.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool setArrayElement(uint32_t index, const Value& data);
|
|
|
|
/** @brief Tests whether an object is a typed array.
|
|
* @return true if object is a typed array, otherwise false.
|
|
*/
|
|
bool isTypedArray() const;
|
|
|
|
/**
|
|
* @brief Gets the type of a typed array object.
|
|
* @return The type of a typed array object.
|
|
*/
|
|
TypedArrayType getTypedArrayType() const;
|
|
|
|
/**
|
|
* @brief Gets backing store of a typed array object.
|
|
* @param[out] ptr A temporary pointer to the backing store of a JavaScript Typed Array object.
|
|
* @param[out] length The byte length of a JavaScript Typed Array object.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool getTypedArrayData(uint8_t** ptr, size_t* length) const;
|
|
|
|
/**
|
|
* @brief Tests whether an object is an array buffer object.
|
|
* @return true if object is an array buffer object, otherwise false.
|
|
*/
|
|
bool isArrayBuffer() const;
|
|
|
|
/**
|
|
* @brief Gets buffer data of an array buffer object.
|
|
* @param[out] ptr A pointer to the data buffer that serves as the backing store for a JavaScript Typed Array object.
|
|
* @param[out] length The number of bytes in a JavaScript data object.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool getArrayBufferData(uint8_t** ptr, size_t* length) const;
|
|
|
|
/**
|
|
* @brief Gets all property names of an object.
|
|
* @param[out] allKeys A string vector to store all property names.
|
|
* @return true if succeed, otherwise false.
|
|
*/
|
|
bool getAllKeys(std::vector<std::string>* allKeys) const;
|
|
|
|
/**
|
|
* @brief Sets a pointer to private data on an object.
|
|
* @param[in] data A void* to set as the object's private data.
|
|
* @note This method will associate private data with se::Object by std::unordered_map::emplace.
|
|
* It's used for search a se::Object via a void* private data.
|
|
*/
|
|
void setPrivateData(void* data);
|
|
|
|
/**
|
|
* @brief Gets an object's private data.
|
|
* @return A void* that is the object's private data, if the object has private data, otherwise nullptr.
|
|
*/
|
|
void* getPrivateData() const;
|
|
|
|
/**
|
|
* @brief Clears private data of an object.
|
|
* @param clearMapping Whether to clear the mapping of native object & se::Object.
|
|
*/
|
|
void clearPrivateData(bool clearMapping = true);
|
|
|
|
/**
|
|
* @brief Roots an object from garbage collection.
|
|
* @note Use this method when you want to store an object in a global or on the heap, where the garbage collector will not be able to discover your reference to it.
|
|
* An object may be rooted multiple times and must be unrooted an equal number of times before becoming eligible for garbage collection.
|
|
*/
|
|
void root();
|
|
|
|
/**
|
|
* @brief Unroots an object from garbage collection.
|
|
* @note An object may be rooted multiple times and must be unrooted an equal number of times before becoming eligible for garbage collection.
|
|
*/
|
|
void unroot();
|
|
|
|
/**
|
|
* @brief Tests whether an object is rooted.
|
|
* @return true if it has been already rooted, otherwise false.
|
|
*/
|
|
bool isRooted() const;
|
|
|
|
/**
|
|
* @brief Tests whether two objects are strict equal, as compared by the JS === operator.
|
|
* @param[in] o The object to be tested with this object.
|
|
* @return true if the two values are strict equal, otherwise false.
|
|
*/
|
|
bool strictEquals(Object* o) const;
|
|
|
|
/**
|
|
* @brief Attaches an object to current object.
|
|
* @param[in] obj The object to be attached.
|
|
* @return true if succeed, otherwise false.
|
|
* @note This method will set `obj` as a property of current object, therefore the lifecycle of child object will depend on current object,
|
|
* which means `obj` may be garbage collected only after current object is garbage collected.
|
|
* It's normally used in binding a native callback method. For example:
|
|
|
|
```javascript
|
|
var self = this;
|
|
someObject.setCallback(function(){}, self);
|
|
```
|
|
|
|
```c++
|
|
static bool SomeObject_setCallback(se::State& s)
|
|
{
|
|
SomeObject* cobj = (SomeObject*)s.nativeThisObject();
|
|
const auto& args = s.args();
|
|
size_t argc = args.size();
|
|
if (argc == 2) {
|
|
std::function<void()> arg0;
|
|
do {
|
|
if (args[0].isObject() && args[0].toObject()->isFunction())
|
|
{
|
|
se::Value jsThis(args[1]);
|
|
se::Value jsFunc(args[0]);
|
|
|
|
jsThis.toObject()->attachObject(jsFunc.toObject());
|
|
|
|
auto lambda = [=]() -> void {
|
|
...
|
|
// Call jsFunc stuff...
|
|
...
|
|
};
|
|
arg0 = lambda;
|
|
}
|
|
else
|
|
{
|
|
arg0 = nullptr;
|
|
}
|
|
} while(false);
|
|
SE_PRECONDITION2(ok, false, "Error processing arguments");
|
|
cobj->setCallback(arg0);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
```
|
|
*/
|
|
bool attachObject(Object* obj);
|
|
|
|
/**
|
|
* @brief Detaches an object from current object.
|
|
* @param[in] obj The object to be detached.
|
|
* @return true if succeed, otherwise false.
|
|
* @note The attached object will not be released if current object is not garbage collected.
|
|
*/
|
|
bool detachObject(Object* obj);
|
|
|
|
/**
|
|
* @brief Returns the string for describing current object.
|
|
* @return The string for describing current object.
|
|
*/
|
|
std::string toString() const;
|
|
|
|
// Private API used in wrapper
|
|
static Object* _createJSObject(Class* cls, JSObject* obj);
|
|
void _setFinalizeCallback(JSFinalizeOp finalizeCb);
|
|
bool _isNativeFunction(JSNative func) const;
|
|
JSObject* _getJSObject() const;
|
|
Class* _getClass() const { return _cls; }
|
|
//
|
|
|
|
private:
|
|
Object();
|
|
bool init(Class* cls, JSObject* obj);
|
|
virtual ~Object();
|
|
|
|
static void setContext(JSContext* cx);
|
|
static void cleanup();
|
|
|
|
void trace(JSTracer* tracer, void* data);
|
|
bool updateAfterGC(void* data);
|
|
|
|
void protect();
|
|
void unprotect();
|
|
void reset();
|
|
|
|
JS::Heap<JSObject*> _heap; /* should be untouched if in rooted mode */
|
|
JS::PersistentRootedObject* _root; /* should be null if not in rooted mode */
|
|
|
|
void* _privateData;
|
|
|
|
Class* _cls;
|
|
JSFinalizeOp _finalizeCb;
|
|
|
|
uint32_t _rootCount;
|
|
uint32_t _currentVMId;
|
|
|
|
friend class ScriptEngine;
|
|
};
|
|
|
|
extern std::unordered_map<Object*, void*> __objectMap; // Currently, the value `void*` is always nullptr
|
|
|
|
} // namespace se {
|
|
|
|
#endif // #if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|
|
|