234 lines
6.5 KiB
C++
234 lines
6.5 KiB
C++
#include <osgIntrospection/PropertyInfo>
|
|
#include <osgIntrospection/Attributes>
|
|
#include <osgIntrospection/variant_cast>
|
|
|
|
using namespace osgIntrospection;
|
|
|
|
void PropertyInfo::getInheritedProviders(CustomAttributeProviderList &providers) const
|
|
{
|
|
for (int i=0; i<decltype_.getNumBaseTypes(); ++i)
|
|
{
|
|
const PropertyInfo *pi = decltype_.getBaseType(i).getProperty(name_, ptype_, getIndexParameters(), false);
|
|
if (pi)
|
|
{
|
|
providers.push_back(pi);
|
|
}
|
|
}
|
|
}
|
|
|
|
Value PropertyInfo::getValue(const Value &instance) const
|
|
{
|
|
const PropertyTypeAttribute *pta = getAttribute<PropertyTypeAttribute>(false);
|
|
const CustomPropertyGetAttribute *cget = getAttribute<CustomPropertyGetAttribute>(false);
|
|
|
|
if (cget)
|
|
{
|
|
if (pta)
|
|
return cget->getGetter()->get(instance).convertTo(pta->getPropertyType());
|
|
return cget->getGetter()->get(instance);
|
|
}
|
|
|
|
if (!getm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::GET);
|
|
|
|
if (pta)
|
|
return getm_->invoke(instance).convertTo(pta->getPropertyType());
|
|
return getm_->invoke(instance);
|
|
}
|
|
|
|
void PropertyInfo::setValue(Value &instance, const Value &value) const
|
|
{
|
|
const CustomPropertySetAttribute *cset = getAttribute<CustomPropertySetAttribute>(false);
|
|
|
|
if (cset)
|
|
{
|
|
cset->getSetter()->set(instance, value);
|
|
return;
|
|
}
|
|
|
|
if (!setm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::SET);
|
|
|
|
ValueList args;
|
|
args.push_back(value);
|
|
setm_->invoke(instance, args);
|
|
}
|
|
|
|
Value PropertyInfo::getIndexedValue(const Value &instance, ValueList &args) const
|
|
{
|
|
const PropertyTypeAttribute *pta = getAttribute<PropertyTypeAttribute>(false);
|
|
const CustomPropertyGetAttribute *cget = getAttribute<CustomPropertyGetAttribute>(false);
|
|
|
|
if (cget)
|
|
{
|
|
if (pta)
|
|
return cget->getGetter()->get(instance, args).convertTo(pta->getPropertyType());
|
|
return cget->getGetter()->get(instance, args);
|
|
}
|
|
|
|
if (!getm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::IGET);
|
|
|
|
if (pta)
|
|
return getm_->invoke(instance, args).convertTo(pta->getPropertyType());
|
|
return getm_->invoke(instance, args);
|
|
}
|
|
|
|
void PropertyInfo::setIndexedValue(Value &instance, ValueList &args, const Value &value) const
|
|
{
|
|
const CustomPropertySetAttribute *cset = getAttribute<CustomPropertySetAttribute>(false);
|
|
if (cset)
|
|
{
|
|
cset->getSetter()->set(instance, args, value);
|
|
return;
|
|
}
|
|
|
|
if (!setm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::ISET);
|
|
|
|
args.push_back(value);
|
|
setm_->invoke(instance, args);
|
|
args.pop_back();
|
|
}
|
|
|
|
int PropertyInfo::getNumArrayItems(const Value &instance) const
|
|
{
|
|
const CustomPropertyCountAttribute *ccount = getAttribute<CustomPropertyCountAttribute>(false);
|
|
if (ccount) return ccount->getCounter()->count(instance);
|
|
|
|
if (!numm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::COUNT);
|
|
|
|
return variant_cast<int>(numm_->invoke(instance));
|
|
}
|
|
|
|
Value PropertyInfo::getArrayItem(const Value &instance, int i) const
|
|
{
|
|
const PropertyTypeAttribute *pta = getAttribute<PropertyTypeAttribute>(false);
|
|
const CustomPropertyGetAttribute *cget = getAttribute<CustomPropertyGetAttribute>(false);
|
|
|
|
if (cget)
|
|
{
|
|
if (pta)
|
|
return cget->getGetter()->get(instance, i).convertTo(pta->getPropertyType());
|
|
return cget->getGetter()->get(instance, i);
|
|
}
|
|
|
|
if (!getm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::AGET);
|
|
|
|
ValueList args;
|
|
args.push_back(i);
|
|
|
|
if (pta)
|
|
return getm_->invoke(instance, args).convertTo(pta->getPropertyType());
|
|
return getm_->invoke(instance, args);
|
|
}
|
|
|
|
void PropertyInfo::setArrayItem(Value &instance, int i, const Value &value) const
|
|
{
|
|
const CustomPropertySetAttribute *cset = getAttribute<CustomPropertySetAttribute>(false);
|
|
if (cset)
|
|
{
|
|
cset->getSetter()->set(instance, i, value);
|
|
return;
|
|
}
|
|
|
|
if (!setm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::ASET);
|
|
|
|
ValueList args;
|
|
args.push_back(i);
|
|
args.push_back(value);
|
|
setm_->invoke(instance, args);
|
|
}
|
|
|
|
void PropertyInfo::addArrayItem(Value &instance, const Value &value) const
|
|
{
|
|
const CustomPropertyAddAttribute *cadd = getAttribute<CustomPropertyAddAttribute>(false);
|
|
if (cadd)
|
|
{
|
|
cadd->getAdder()->add(instance, value);
|
|
return;
|
|
}
|
|
|
|
if (!addm_)
|
|
throw PropertyAccessException(decltype_.getQualifiedName() + "::" + name_, PropertyAccessException::ADD);
|
|
|
|
ValueList args;
|
|
args.push_back(value);
|
|
addm_->invoke(instance, args);
|
|
}
|
|
|
|
Value PropertyInfo::getDefaultValue() const
|
|
{
|
|
if (isArray() || isIndexed()) return Value();
|
|
|
|
const CustomAttributeList &cal = getCustomAttributes();
|
|
for (CustomAttributeList::const_iterator i=cal.begin(); i!=cal.end(); ++i)
|
|
{
|
|
if (dynamic_cast<const NoDefaultValueAttribute *>(*i) != 0)
|
|
return Value();
|
|
|
|
const DefaultValueAttribute *dv = dynamic_cast<const DefaultValueAttribute *>(*i);
|
|
if (dv)
|
|
{
|
|
return dv->getDefaultValue();
|
|
}
|
|
}
|
|
|
|
if (decltype_.isAbstract())
|
|
{
|
|
if (ptype_.isAbstract() || !ptype_.isDefined())
|
|
return Value();
|
|
return ptype_.createInstance();
|
|
}
|
|
|
|
// auto default value
|
|
Value instance = decltype_.createInstance();
|
|
return getValue(instance);
|
|
}
|
|
|
|
void PropertyInfo::getIndexValueSet(int whichindex, const Value &instance, ValueList &values) const
|
|
{
|
|
const CustomIndexAttribute *cia = getAttribute<CustomIndexAttribute>(false);
|
|
if (cia)
|
|
{
|
|
cia->getIndexInfo()->getIndexValueSet(whichindex, instance, values);
|
|
}
|
|
else
|
|
{
|
|
std::map<int, const IndexTypeAttribute *> ita_map;
|
|
const CustomAttributeList &cal = getCustomAttributes();
|
|
for (CustomAttributeList::const_iterator i=cal.begin(); i!=cal.end(); ++i)
|
|
{
|
|
const IndexTypeAttribute *ita = dynamic_cast<const IndexTypeAttribute *>(*i);
|
|
if (ita)
|
|
ita_map[ita->getWhichIndex()] = ita;
|
|
}
|
|
|
|
const EnumLabelMap &elm = getIndexParameters().at(whichindex)->getParameterType().getEnumLabels();
|
|
if (elm.empty())
|
|
throw IndexValuesNotDefinedException(name_, getIndexParameters().at(whichindex)->getName());
|
|
|
|
for (EnumLabelMap::const_iterator i=elm.begin(); i!=elm.end(); ++i)
|
|
{
|
|
if (ita_map[whichindex])
|
|
values.push_back(Value(i->first).convertTo(ita_map[whichindex]->getIndexType()));
|
|
else
|
|
values.push_back(Value(i->first).convertTo(indices_[whichindex]->getParameterType()));
|
|
}
|
|
}
|
|
}
|
|
|
|
const ParameterInfoList &PropertyInfo::getIndexParameters() const
|
|
{
|
|
const CustomIndexAttribute *cia = getAttribute<CustomIndexAttribute>(false);
|
|
if (cia)
|
|
{
|
|
return cia->getIndexInfo()->getIndexParameters();
|
|
}
|
|
|
|
return indices_;
|
|
}
|