Fixed adding properties to methods

This commit is contained in:
ksuprynowicz 2023-01-15 15:58:21 +01:00
parent 076349a488
commit 386242d9c7
5 changed files with 80 additions and 7 deletions

View file

@ -297,11 +297,23 @@ void ScriptObjectV8Proxy::investigate() {
}
v8::Local<v8::Object> v8Object = objectTemplate->NewInstance(_engine->getContext()).ToLocalChecked();
v8Object->SetAlignedPointerInInternalField(0, const_cast<void*>(internalPointsToQObjectProxy));
v8Object->SetAlignedPointerInInternalField(1, reinterpret_cast<void*>(this));
v8Object->SetInternalField(2, v8::Object::New(_engine->getIsolate()));
// Properties added later will be stored in this object
v8::Local<v8::Object> propertiesObject = v8::Object::New(_engine->getIsolate());
v8Object->SetInternalField(2, propertiesObject);
_v8Object.Reset(_engine->getIsolate(), v8Object);
// Add all the methods objects as properties - this allows adding properties to a given method later. Is used by Script.request.
// V8TODO: Should these be deleted when the script-owned object is destroyed? It needs checking if script-owned objects will be garbage-collected, or will self-referencing prevent it.
for (auto i = _methods.begin(); i != _methods.end(); i++) {
V8ScriptValue method = ScriptMethodV8Proxy::newMethod(_engine, qobject, V8ScriptValue(_engine->getIsolate(), v8Object), i.value().methods, i.value().numMaxParms);
if(!propertiesObject->Set(_engine->getContext(), i.value().name.constGet(), method.get()).FromMaybe(false)) {
Q_ASSERT(false);
}
}
}
QString ScriptObjectV8Proxy::name() const {
@ -461,6 +473,16 @@ V8ScriptValue ScriptObjectV8Proxy::property(const V8ScriptValue& object, const V
qDebug(scriptengine) << "Method with QMetaType::UnknownType " << metaObject->className() << " " << (*iter).name();
}
} //V8TODO: is new method created during every call? It needs to be cached instead
bool isMethodDefined = false;
v8::Local<v8::Value> property;
if(_v8Object.Get(_engine->getIsolate())->GetInternalField(2).As<v8::Object>()->Get(_engine->getContext(), name.constGet()).ToLocal(&property)) {
if (!property->IsUndefined()) {
qDebug() << "Using existing method object";
return V8ScriptValue(_engine->getIsolate(), property);
}
}
Q_ASSERT(false);
qDebug(scriptengine) << "(This should not happen) Creating new method object for " << metaObject->className() << " " << name.toQString();
return ScriptMethodV8Proxy::newMethod(_engine, qobject, object, methodDef.methods, methodDef.numMaxParms);
}
case SIGNAL_TYPE: {
@ -479,6 +501,7 @@ V8ScriptValue ScriptObjectV8Proxy::property(const V8ScriptValue& object, const V
ScriptEngine::QObjectWrapOptions options = ScriptEngine::ExcludeSuperClassContents |
//V8TODO ScriptEngine::ExcludeDeleteLater |
ScriptEngine::PreferExistingWrapperObject;
//V8TODO: why is it returning new object every time?
return ScriptObjectV8Proxy::newQObject(_engine, proxy, ScriptEngine::ScriptOwnership, options);
//return _engine->newQObject(proxy, ScriptEngine::ScriptOwnership, options);
}

View file

@ -112,7 +112,7 @@ private: // storage
QPointer<QObject> _object;
// V8TODO Is this necessary?
v8::UniquePersistent<v8::ObjectTemplate> _v8ObjectTemplate;
// V8TODO Maybe it doesn't really need to point to itsef?
// Handle for its own object
v8::UniquePersistent<v8::Object> _v8Object;
int pointerCorruptionTest = 12345678;

View file

@ -155,6 +155,39 @@ ScriptValue ScriptValueV8Wrapper::construct(const ScriptValue& arguments) {
}
ScriptValue ScriptValueV8Wrapper::data() const {
auto isolate = _engine->getIsolate();
v8::Locker locker(isolate);
v8::Isolate::Scope isolateScope(isolate);
v8::HandleScope handleScope(isolate);
v8::Context::Scope contextScope(_engine->getContext());
// Private properties are an experimental feature for now on V8, so we are using regular value for now
if (_value.constGet()->IsObject()) {
auto v8Object = v8::Local<v8::Object>::Cast(_value.constGet());
v8::Local<v8::Value> data;
//bool createData = false;
if (!v8Object->Get(_engine->getContext(), v8::String::NewFromUtf8(isolate, "__data").ToLocalChecked()).ToLocal(&data)) {
data = v8::Undefined(isolate);
//createData = true;
} /*else {
if (data->IsUndefined()) {
createData = true;
}
}
if (createData) {
qDebug() << "ScriptValueV8Wrapper::data(): Data object doesn't exist, creating new one";
// Create data object if it's non-existent or invalid
data = v8::Object::New(isolate);
if( !v8Object->Set(_engine->getContext(), v8::String::NewFromUtf8(isolate, "__data").ToLocalChecked(), data).FromMaybe(false)) {
qDebug() << "ScriptValueV8Wrapper::data(): Data object couldn't be created";
Q_ASSERT(false);
}
}*/
V8ScriptValue result(isolate, data);
return ScriptValue(new ScriptValueV8Wrapper(_engine, std::move(result)));
} else {
qDebug() << "ScriptValueV8Wrapper::data() was called on a value that is not an object";
Q_ASSERT(false);
}
//V8TODO I'm not sure how this would work in V8
//V8ScriptValue result = _value.data();
//return ScriptValue(new ScriptValueV8Wrapper(_engine, std::move(result)));
@ -255,7 +288,18 @@ void ScriptValueV8Wrapper::setData(const ScriptValue& value) {
v8::HandleScope handleScope(isolate);
v8::Context::Scope contextScope(_engine->getContext());
V8ScriptValue unwrapped = fullUnwrap(value);
(**_value.get()) = (**unwrapped.get());
// V8TODO Check if it uses same isolate. Would pointer check be enough?
// Private properties are an experimental feature for now on V8, so we are using regular value for now
if (_value.constGet()->IsObject()) {
auto v8Object = v8::Local<v8::Object>::Cast(_value.constGet());
if( !v8Object->Set(_engine->getContext(), v8::String::NewFromUtf8(isolate, "__data").ToLocalChecked(), unwrapped.constGet()).FromMaybe(false)) {
qDebug() << "ScriptValueV8Wrapper::data(): Data object couldn't be created";
Q_ASSERT(false);
}
} else {
qDebug() << "ScriptValueV8Wrapper::data() was called on a value that is not an object";
Q_ASSERT(false);
}
}
void ScriptValueV8Wrapper::setProperty(const QString& name, const ScriptValue& value, const ScriptValue::PropertyFlags& flags) {

View file

@ -87,7 +87,7 @@ void ScriptEngineTests::scriptTest() {
qInfo() << "Running test script: " << script;
ac->loadOneScript(script);
}*/
ac->loadOneScript("tests/003_vector_math.js");
//ac->loadOneScript("tests/003_vector_math.js");
ac->loadOneScript("tests/004_require.js");
qDebug() << ac->getRunning();
@ -95,7 +95,7 @@ void ScriptEngineTests::scriptTest() {
// TODO: if I don't have infinite loop here, it exits before scripts finish. It also reports: QSignalSpy: No such signal: 'scriptCountChanged'
for (int n = 0; n > -1; n++) {
QSignalSpy spy(ac.get(), SIGNAL(scriptCountChanged));
spy.wait(1000);
spy.wait(100000);
qDebug() << "Signal happened";
}
//spy.wait(5000);

View file

@ -1,3 +1,9 @@
print("Script.require: " + JSON.stringify(Script.require));
print("Script.require.cache: " + JSON.stringify(Script.require.cache));
print(JSON.stringify("Script.require.test_prop before defining: " + Script.require.test_prop));
Script.require.test_prop = "test property";
print(JSON.stringify("Script.require.test_prop after defining: " + Script.require.test_prop));
print("Before require");
Script.require("001_test_print.js");
// TODO: find correct local path for the test module
Script.require("http://oaktown.pl/scripts/001_test_print.js");
print("After require");