cppbind.Ghost: register _set called on setting new properties.

This commit is contained in:
Thomas Geymayer
2014-05-05 14:49:50 +02:00
parent f75d1cbcb1
commit b101f64cd8
2 changed files with 80 additions and 8 deletions

View File

@@ -214,6 +214,10 @@ namespace nasal
typedef boost::function<naRef(naContext, raw_type&)> getter_t;
typedef boost::function<void(naContext, raw_type&, naRef)> setter_t;
typedef boost::function<naRef(raw_type&, const CallContext&)> method_t;
typedef boost::function<bool( naContext,
raw_type&,
const std::string&,
naRef )> fallback_setter_t;
class MethodHolder:
public internal::MethodHolder
@@ -390,6 +394,9 @@ namespace nasal
);
}
if( !_fallback_setter )
_fallback_setter = base->_fallback_setter;
return *this;
}
@@ -498,6 +505,47 @@ namespace nasal
return *this;
}
/**
* Register a function which is called upon setting an unknown member of
* this ghost.
*/
Ghost& _set(const fallback_setter_t& setter)
{
_fallback_setter = setter;
return *this;
}
/**
* Register a method which is called upon setting an unknown member of
* this ghost.
*
* @code{cpp}
* class MyClass
* {
* public:
* bool setMember( const std::string& key,
* const std::string& value );
* }
*
* Ghost<MyClassPtr>::init("Test")
* ._set(&MyClass::setMember);
* @endcode
*/
template<class Param>
Ghost _set(bool (raw_type::*setter)(const std::string&, Param))
{
// Setter signature: bool( naContext,
// raw_type&,
// const std::string&,
// naRef )
return _set(boost::bind(
setter,
_2,
_3,
boost::bind(from_nasal_ptr<Param>::get(), _1, _4)
));
}
/**
* Register anything that accepts an object instance and a
* nasal::CallContext and returns naRef as method.
@@ -835,7 +883,8 @@ namespace nasal
};
typedef std::auto_ptr<Ghost> GhostPtr;
MemberMap _members;
MemberMap _members;
fallback_setter_t _fallback_setter;
explicit Ghost(const std::string& name):
GhostMetadata(name, &_ghost_type)
@@ -913,17 +962,23 @@ namespace nasal
static void setMember(naContext c, void* g, naRef field, naRef val)
{
const std::string key = nasal::from_nasal<std::string>(c, field);
typename MemberMap::iterator member =
getSingletonPtr()->_members.find(key);
const MemberMap& members = getSingletonPtr()->_members;
if( member == getSingletonPtr()->_members.end() )
naRuntimeError(c, "ghost: No such member: %s", key.c_str());
typename MemberMap::const_iterator member = members.find(key);
if( member == members.end() )
{
fallback_setter_t fallback_set = getSingletonPtr()->_fallback_setter;
if( !fallback_set )
naRuntimeError(c, "ghost: No such member: %s", key.c_str());
else if( !fallback_set(c, *getRawPtr(g), key, val) )
naRuntimeError(c, "ghost: Failed to write (_set: %s)", key.c_str());
}
else if( member->second.setter.empty() )
naRuntimeError(c, "ghost: Write protected member: %s", key.c_str());
else if( member->second.func )
naRuntimeError(c, "ghost: Write to function: %s", key.c_str());
member->second.setter(c, *getRawPtr(g), val);
else
member->second.setter(c, *getRawPtr(g), val);
}
};

View File

@@ -40,6 +40,10 @@ struct Base
void setVar(const std::string v) { var = v; }
unsigned long getThis() const { return (unsigned long)this; }
bool genericSet(const std::string& key, const std::string& val)
{
return key == "test";
}
};
void baseVoidFunc(Base& b) {}
@@ -206,7 +210,8 @@ int main(int argc, char* argv[])
.method("int2args", &baseFunc2Args)
.method("bool2args", &Base::test2Args)
.method("str_ptr", &testPtr)
.method("this", &Base::getThis);
.method("this", &Base::getThis)
._set(&Base::genericSet);
Ghost<DerivedPtr>::init("DerivedPtr")
.bases<BasePtr>()
.member("x", &Derived::getX, &Derived::setX)
@@ -319,6 +324,18 @@ int main(int argc, char* argv[])
VERIFY( objects[1] == d2 );
VERIFY( objects[2] == d3 );
// Calling fallback setter for unset values
const char* src_code = "me.test = 3;";
int errLine = -1;
naRef code = naParseCode( c, to_nasal(c, "source.nas"), 0,
(char*)src_code, strlen(src_code),
&errLine );
ret = naCallMethod(code, derived, 0, 0, naNil());
VERIFY( !naGetError(c) ) // TODO real error check (this seems to always
// return 0...
VERIFY( from_nasal<int>(c, ret) == 3 )
//----------------------------------------------------------------------------
// Test nasal::CallContext
//----------------------------------------------------------------------------