Implements basics for UserData
This commit is contained in:
parent
831dbe6917
commit
996b5be496
|
@ -3,6 +3,7 @@
|
|||
#include "Binder.hpp"
|
||||
#include "../TableScriptType.hpp"
|
||||
#include "BoundExpressions/BoundTableExpression.hpp"
|
||||
#include "../UserData/UserDataScriptType.hpp"
|
||||
#include <memory>
|
||||
|
||||
BoundScriptStatement *Binder::Bind(Script* script, const ParsedScriptStatement *s, BoundScope* scriptScope) {
|
||||
|
@ -74,7 +75,7 @@ std::shared_ptr<ScriptType> ParseTypeIdentifier(HashedString s){
|
|||
case HashedString::ConstHash("number"): return std::make_shared<NumericScriptType>(false, false);
|
||||
case HashedString::ConstHash("bool"): return std::make_shared<ScriptType>(TypeClass::Bool);
|
||||
case HashedString::ConstHash("string"): return std::make_shared<StringScriptType>(false, 0);
|
||||
default: return std::make_shared<ScriptType>(TypeClass::Error); // todo: change to userdata
|
||||
default: return std::make_shared<UserDataScriptType>(s.GetHash());
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -102,6 +102,7 @@ shared_ptr<EvalValue> Evaluator::EvaluateExpression(const BoundExpression *expre
|
|||
case TypeClass ::Function: return this->EvaluateFunctionExpression(expression);
|
||||
case TypeClass ::Nil: return this->EvaluateNilExpression(expression);
|
||||
case TypeClass ::Table: return this-> EvaluateTableExpression(expression);
|
||||
case TypeClass ::UserData: return this -> EvaluateUserDataExpression(expression);
|
||||
default: throw;
|
||||
}
|
||||
}
|
||||
|
@ -291,3 +292,11 @@ shared_ptr<EvalValue> Evaluator::EvaluateComplexTableExpression(const BoundExpre
|
|||
this -> _evaluationScope = currentEvaluator;
|
||||
return make_shared<TableEvalValue>(variables);
|
||||
}
|
||||
|
||||
shared_ptr<EvalValue> Evaluator::EvaluateUserDataExpression(const BoundExpression *expression) {
|
||||
switch (expression->GetKind()){
|
||||
case BoundExpressionKind ::Variable: return this->GetVariable((BoundVariableExpression*)expression);
|
||||
case BoundExpressionKind ::Index: return this -> EvaluateIndexExpression(expression);
|
||||
default: throw;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -48,6 +48,7 @@ class Evaluator {
|
|||
shared_ptr<EvalValue> EvaluateIndexExpression(const BoundExpression* expression);
|
||||
shared_ptr<EvalValue> EvaluateNumericTableExpression(const BoundExpression* expression);
|
||||
shared_ptr<EvalValue> EvaluateComplexTableExpression(const BoundExpression *expression);
|
||||
shared_ptr<EvalValue> EvaluateUserDataExpression(const BoundExpression *expression);
|
||||
|
||||
shared_ptr<EvalValue> GetVariable(const BoundVariableExpression *expression);
|
||||
public:
|
||||
|
|
|
@ -0,0 +1,2 @@
|
|||
|
||||
#include "UserData.hpp"
|
|
@ -0,0 +1,25 @@
|
|||
#ifndef PORYGONLANG_USERDATA_HPP
|
||||
#define PORYGONLANG_USERDATA_HPP
|
||||
|
||||
#include <utility>
|
||||
#include <unordered_map>
|
||||
#include "UserDataField.hpp"
|
||||
|
||||
class UserData {
|
||||
std::unordered_map<int, UserDataField*> _fields;
|
||||
public:
|
||||
explicit UserData(std::unordered_map<int, UserDataField*> fields){
|
||||
_fields = std::move(fields);
|
||||
}
|
||||
|
||||
bool ContainsField(int fieldId){
|
||||
return _fields.find(fieldId) != _fields.end();
|
||||
}
|
||||
|
||||
UserDataField* GetField(int fieldId){
|
||||
return _fields[fieldId];
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
#endif //PORYGONLANG_USERDATA_HPP
|
|
@ -0,0 +1,2 @@
|
|||
|
||||
#include "UserDataField.hpp"
|
|
@ -0,0 +1,33 @@
|
|||
|
||||
#ifndef PORYGONLANG_USERDATAFIELD_HPP
|
||||
#define PORYGONLANG_USERDATAFIELD_HPP
|
||||
|
||||
|
||||
#include "../Evaluator/EvalValues/EvalValue.hpp"
|
||||
|
||||
class UserDataField {
|
||||
shared_ptr<ScriptType> _type;
|
||||
EvalValue* (*_get)(void* obj);
|
||||
void (*_set)(void* obj, EvalValue*);
|
||||
public:
|
||||
UserDataField(ScriptType* type, EvalValue* (*getter)(void* obj), void (*setter)(void* obj, EvalValue*)){
|
||||
_type = shared_ptr<ScriptType>(type);
|
||||
_get = getter;
|
||||
_set = setter;
|
||||
}
|
||||
|
||||
shared_ptr<ScriptType> GetType(){
|
||||
return _type;
|
||||
}
|
||||
|
||||
EvalValue* Get(void* obj){
|
||||
return this ->_get(obj);
|
||||
}
|
||||
|
||||
void Set(void* obj, EvalValue* val){
|
||||
this->_set(obj, val);
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
#endif //PORYGONLANG_USERDATAFIELD_HPP
|
|
@ -0,0 +1,2 @@
|
|||
|
||||
#include "UserDataScriptType.hpp"
|
|
@ -0,0 +1,31 @@
|
|||
|
||||
#ifndef PORYGONLANG_USERDATASCRIPTTYPE_HPP
|
||||
#define PORYGONLANG_USERDATASCRIPTTYPE_HPP
|
||||
|
||||
|
||||
#include "../ScriptType.hpp"
|
||||
#include "UserData.hpp"
|
||||
#include "UserDataStorage.hpp"
|
||||
|
||||
class UserDataScriptType : public ScriptType{
|
||||
UserData* _userData;
|
||||
public:
|
||||
explicit UserDataScriptType(int id) : ScriptType(TypeClass::UserData){
|
||||
_userData = UserDataStorage::GetUserDataType(id);
|
||||
}
|
||||
|
||||
bool CanBeIndexedWith(ScriptType* indexer) final{
|
||||
return indexer->GetClass() == TypeClass ::String;
|
||||
}
|
||||
|
||||
shared_ptr<ScriptType> GetIndexedType(ScriptType* indexer) final{
|
||||
auto stringKey = (StringScriptType*)indexer;
|
||||
if (stringKey->IsKnownAtBind()){
|
||||
return _userData->GetField(stringKey->GetHashValue())->GetType();
|
||||
}
|
||||
throw "TODO: indexing with dynamic keys";
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
#endif //PORYGONLANG_USERDATASCRIPTTYPE_HPP
|
|
@ -0,0 +1,4 @@
|
|||
|
||||
#include "UserDataStorage.hpp"
|
||||
|
||||
std::unordered_map<int, UserData*> UserDataStorage::_userData = {};
|
|
@ -0,0 +1,21 @@
|
|||
|
||||
#ifndef PORYGONLANG_USERDATASTORAGE_HPP
|
||||
#define PORYGONLANG_USERDATASTORAGE_HPP
|
||||
|
||||
#include <unordered_map>
|
||||
#include "UserData.hpp"
|
||||
|
||||
class UserDataStorage {
|
||||
static std::unordered_map<int, UserData*> _userData;
|
||||
public:
|
||||
static void RegisterType(int i, UserData* ud){
|
||||
UserDataStorage::_userData.insert({i, ud});
|
||||
}
|
||||
|
||||
static UserData* GetUserDataType(int i){
|
||||
return UserDataStorage::_userData[i];
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
#endif //PORYGONLANG_USERDATASTORAGE_HPP
|
|
@ -0,0 +1,2 @@
|
|||
|
||||
#include "UserDataValue.hpp"
|
|
@ -0,0 +1,50 @@
|
|||
|
||||
#ifndef PORYGONLANG_USERDATAVALUE_HPP
|
||||
#define PORYGONLANG_USERDATAVALUE_HPP
|
||||
|
||||
|
||||
#include "../Evaluator/EvalValues/EvalValue.hpp"
|
||||
#include "UserData.hpp"
|
||||
#include "UserDataStorage.hpp"
|
||||
|
||||
class UserDataValue : public EvalValue{
|
||||
UserData* _userData;
|
||||
void* _obj;
|
||||
public:
|
||||
UserDataValue(UserData* userData, void* obj){
|
||||
_userData = userData;
|
||||
_obj = obj;
|
||||
}
|
||||
|
||||
UserDataValue(int userDataId, void* obj){
|
||||
_userData = UserDataStorage::GetUserDataType(userDataId);
|
||||
_obj = obj;
|
||||
}
|
||||
|
||||
const TypeClass GetTypeClass() final{
|
||||
return TypeClass ::UserData;
|
||||
}
|
||||
|
||||
bool operator ==(EvalValue* b) final {
|
||||
if (b->GetTypeClass() != TypeClass::UserData)
|
||||
return false;
|
||||
return _obj == ((UserDataValue*)b)->_obj;
|
||||
}
|
||||
|
||||
shared_ptr<EvalValue> Clone() final{
|
||||
return make_shared<UserDataValue>(_userData, _obj);
|
||||
}
|
||||
|
||||
std::size_t GetHashCode() final{
|
||||
return reinterpret_cast<intptr_t>(_obj);
|
||||
}
|
||||
|
||||
shared_ptr<EvalValue> IndexValue(EvalValue* val) final {
|
||||
auto fieldId = val->GetHashCode();
|
||||
auto field = _userData->GetField(fieldId);
|
||||
return shared_ptr<EvalValue>(field->Get(_obj));
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
#endif //PORYGONLANG_USERDATAVALUE_HPP
|
|
@ -0,0 +1,48 @@
|
|||
|
||||
#ifdef TESTS_BUILD
|
||||
#include <catch.hpp>
|
||||
#include "../src/Script.hpp"
|
||||
#include "../../src/UserData/UserData.hpp"
|
||||
#include "../../src/UserData/UserDataStorage.hpp"
|
||||
#include "../../src/UserData/UserDataValue.hpp"
|
||||
|
||||
class UserDataTestObject{
|
||||
public:
|
||||
int foo = 10;
|
||||
|
||||
static EvalValue* GetFoo(void* obj){
|
||||
return new IntegerEvalValue(((UserDataTestObject*)obj)->foo);
|
||||
}
|
||||
|
||||
static void SetFoo(void* obj, EvalValue* val){
|
||||
((UserDataTestObject*)obj)->foo = val->EvaluateInteger();
|
||||
}
|
||||
|
||||
static UserData* CreateData(){
|
||||
return new UserData({
|
||||
{
|
||||
HashedString::ConstHash("foo"),
|
||||
new UserDataField(new NumericScriptType(true, false), GetFoo, SetFoo)
|
||||
}
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
TEST_CASE( "Gets UserData value", "[integration]" ) {
|
||||
UserDataStorage::RegisterType(HashedString::ConstHash("testObject"), UserDataTestObject::CreateData());
|
||||
Script* script = Script::Create(R"(
|
||||
function testFunc(testObject obj)
|
||||
return obj["foo"]
|
||||
end
|
||||
)");
|
||||
REQUIRE(!script->Diagnostics -> HasErrors());
|
||||
script->Evaluate();
|
||||
auto parameter = new UserDataValue(HashedString::ConstHash("testObject"), new UserDataTestObject());
|
||||
auto variable = script->CallFunction("testFunc", {parameter});
|
||||
REQUIRE(variable != nullptr);
|
||||
REQUIRE(variable->EvaluateInteger() == 10);
|
||||
delete script;
|
||||
}
|
||||
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue