#if !defined(TOONZ_PLUGIN_PARAM_TRAITS_H__)
#define TOONZ_PLUGIN_PARAM_TRAITS_H__
#include <toonz_params.h>
#include <functional>
template <typename First, typename Second>
struct param_bind_t {
typedef First traittype;
typedef Second realtype;
typedef typename std::is_compound<typename First::valuetype>::value_type complextype;
typedef typename First::valuetype valuetype;
static const int RANGED = First::RANGED;
static const size_t valuesize = sizeof(typename First::valuetype);
};
typedef param_bind_t<toonz_param_traits_double_t, TDoubleParam> tpbind_dbl_t;
typedef param_bind_t<toonz_param_traits_range_t, TRangeParam> tpbind_rng_t;
typedef param_bind_t<toonz_param_traits_color_t, TPixelParam> tpbind_col_t;
typedef param_bind_t<toonz_param_traits_point_t, TPointParam> tpbind_pnt_t;
typedef param_bind_t<toonz_param_traits_enum_t, TIntEnumParam> tpbind_enm_t;
typedef param_bind_t<toonz_param_traits_int_t, TIntParam> tpbind_int_t;
typedef param_bind_t<toonz_param_traits_bool_t, TBoolParam> tpbind_bool_t;
typedef param_bind_t<toonz_param_traits_spectrum_t, TSpectrumParam> tpbind_spc_t;
typedef param_bind_t<toonz_param_traits_string_t, TStringParam> tpbind_str_t;
typedef param_bind_t<toonz_param_traits_tonecurve_t, TToneCurveParam> tpbind_tcv_t;
template <typename T>
inline bool is_type_of(const toonz_param_desc_t *desc)
{
if (desc->traits_tag == T::E)
return true;
return false;
}
/* Complex なパラメータは直接 setRangeValue() などを持たず、集約しているサブタイプを返すものがあるので、そのサブタイプを得る関数を取得する */
template <typename RT, typename F = TDoubleParamP>
inline F &get_func_a(RT *t) { assert(false); }
template <typename RT, typename F = TDoubleParamP>
inline F &get_func_b(RT *t) { assert(false); }
/* TRangeParam */
template <>
inline TDoubleParamP &get_func_a<TRangeParam, TDoubleParamP>(TRangeParam *t)
{
printf("get_func_a< TRangeParam, TDoubleParamP& >(TRangeParam* t)\n");
return std::mem_fun(&TRangeParam::getMin)(t);
}
template <>
inline TDoubleParamP &get_func_b<TRangeParam, TDoubleParamP>(TRangeParam *t)
//template<> std::mem_fun_ref_t< TDoubleParamP&, TRangeParam > get_func_b< TRangeParam, std::mem_fun_ref_t< TDoubleParamP&, TRangeParam > >(TRangeParam* t)
{
printf("get_func_b< TRangeParam, TDoubleParamP& >(TRangeParam* t)\n");
return std::mem_fun(&TRangeParam::getMax)(t);
}
/* TPointParam */
template <>
inline TDoubleParamP &get_func_a<TPointParam, TDoubleParamP>(TPointParam *t)
//template<> std::mem_fun_ref_t< TDoubleParamP&, TPointParam > get_func_a< TPointParam, std::mem_fun_ref_t< TDoubleParamP&, TPointParam > >(TPointParam* t)
{
printf("get_func_a< TPointParam, TDoubleParamP& >(TPointParam* t)\n");
return std::mem_fun(&TPointParam::getX)(t);
}
template <>
inline TDoubleParamP &get_func_b<TPointParam, TDoubleParamP>(TPointParam *t)
//template<> std::mem_fun_ref_t< TDoubleParamP&, TPointParam > get_func_b< TPointParam, std::mem_fun_ref_t< TDoubleParamP&, TPointParam > >(TPointParam* t)
{
printf("get_func_b< TPointParam, TDoubleParamP& >(TPointParam* t)\n");
return std::mem_fun(&TPointParam::getY)(t);
}
/* valuetype が集約型の場合、 スカラを取得するための関数 */
template <typename T, typename V>
inline V get_1st_value(const T &) {}
template <typename T, typename V>
inline V get_2nd_value(const T &) {}
template <>
inline double get_1st_value(const toonz_param_traits_range_t::valuetype &r) { return r.a; }
template <>
inline double get_2nd_value(const toonz_param_traits_range_t::valuetype &r) { return r.b; }
template <>
inline double get_1st_value(const toonz_param_traits_point_t::valuetype &p) { return p.x; }
template <>
inline double get_2nd_value(const toonz_param_traits_point_t::valuetype &p) { return p.y; }
template <typename Bind, typename Comp = typename std::is_compound<typename Bind::valuetype>::type, int Ranged = Bind::RANGED>
//template < int Ranged, typename Comp, typename Bind >
struct set_param_range_t {
static bool set_param_range(Param *param, const toonz_param_desc_t *desc)
{
/* 範囲を持たない(Ranged == std::false_type)なら何もすることはない */
printf("(none)set_param_range: p:%p type:%s (Comp:%s Ranged:%d)\n", param, typeid(Bind).name(), typeid(Comp).name(), Ranged);
return false;
}
};
//static_assert(std::is_compound< const char* >(), "false");
/* ranged complextype */
template <typename Bind>
struct set_param_range_t<Bind, std::true_type, 1> {
static bool set_param_range(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
typename Bind::realtype *p = reinterpret_cast<typename Bind::realtype *>(smartptr.getPointer());
if (p) {
const typename Bind::traittype &t = *reinterpret_cast<const typename Bind::traittype *>(&desc->traits.d);
auto subtype_a = get_func_a<typename Bind::realtype>(p);
auto subtype_b = get_func_b<typename Bind::realtype>(p);
auto a_minval = get_1st_value<typename Bind::valuetype, double>(t.min);
auto a_maxval = get_2nd_value<typename Bind::valuetype, double>(t.min);
auto b_minval = get_1st_value<typename Bind::valuetype, double>(t.max);
auto b_maxval = get_2nd_value<typename Bind::valuetype, double>(t.max);
printf("a->set_param_range: (%g, %g)\n", a_minval, a_maxval);
printf("b->set_param_range: (%g, %g)\n", b_minval, b_maxval);
(subtype_a)->setValueRange(a_minval, a_maxval);
(subtype_b)->setValueRange(b_minval, b_maxval);
}
return true;
}
};
/* range のとき、スライダの左と右それぞれに限界が設定できるように見えるが、そうではない.
getMin(), getMax() の結果それぞれに range を設定できるように見えて
実際は getMin() には (min, max) のうち min, getMax() には (min, max) のうち max しか有効でないように見える.
このため range に対しても特殊版を用意するハメになった. */
template <>
struct set_param_range_t<tpbind_rng_t, std::true_type, 1> {
static bool set_param_range(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
tpbind_rng_t::realtype *p = reinterpret_cast<tpbind_rng_t::realtype *>(smartptr.getPointer());
if (p) {
const tpbind_rng_t::traittype &t = desc->traits.rd;
auto subtype_a = get_func_a<tpbind_rng_t::realtype>(p);
auto subtype_b = get_func_b<tpbind_rng_t::realtype>(p);
auto a_minval = get_1st_value<tpbind_rng_t::valuetype, double>(t.minmax);
auto a_maxval = get_2nd_value<tpbind_rng_t::valuetype, double>(t.minmax);
(subtype_a)->setValueRange(a_minval, a_maxval);
(subtype_b)->setValueRange(a_minval, a_maxval);
}
return true;
}
};
/*
template <>
struct set_param_range_t< tpbind_pnt_t, std::true_type, 1 > {
static bool set_param_range(Param* param, const toonz_param_desc_t* desc)
{
auto smartptr = param->param();
tpbind_pnt_t::realtype* p = reinterpret_cast< tpbind_pnt_t::realtype* >(smartptr.getPointer());
if (p) {
const tpbind_pnt_t::traittype& t = *reinterpret_cast< const tpbind_pnt_t::traittype* >(&desc->traits.d);
auto subtype_a = get_func_a< tpbind_pnt_t::realtype >(p);
auto subtype_b = get_func_b< tpbind_pnt_t::realtype >(p);
auto a_minval = get_1st_value< tpbind_pnt_t::valuetype, double >(t.min);
auto a_maxval = get_2nd_value< tpbind_pnt_t::valuetype, double >(t.min);
auto b_minval = get_1st_value< tpbind_pnt_t::valuetype, double >(t.max);
auto b_maxval = get_2nd_value< tpbind_pnt_t::valuetype, double >(t.max);
printf("a->set_param_range: pnt(%g, %g)\n", a_minval, a_maxval);
printf("b->set_param_range: pnt(%g, %g)\n", b_minval, b_maxval);
(subtype_a)->setValueRange(a_minval, a_maxval);
(subtype_b)->setValueRange(b_minval, b_maxval);
}
return true;
}
};
*/
/* ranged primitive: */
template <typename Bind>
struct set_param_range_t<Bind, std::false_type, 1> {
static bool set_param_range(Param *param, const toonz_param_desc_t *desc)
{
if (!is_type_of<typename Bind::traittype>(desc))
return false;
auto smartptr = param->param();
typename Bind::realtype *p = reinterpret_cast<typename Bind::realtype *>(smartptr.getPointer());
if (p) {
const typename Bind::traittype &t = *reinterpret_cast<const typename Bind::traittype *>(&desc->traits.d);
printf("p(%p)->set_param_range: typeid:%s desc:%p (%p)\n", p, typeid(typename Bind::traittype).name(), desc, &desc->traits.d);
p->setValueRange(t.min, t.max);
}
return true;
}
};
template <typename Bind>
bool set_param_range(Param *param, const toonz_param_desc_t *desc)
{
if (!is_type_of<typename Bind::traittype>(desc))
return false;
return set_param_range_t<Bind>::set_param_range(param, desc);
}
template <typename Bind, typename Comp = typename std::is_compound<typename Bind::valuetype>::type>
struct set_param_default_t {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
return false;
}
};
/* Default complextype */
/* Point/Range */
template <typename Bind>
struct set_param_default_t<Bind, std::true_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
typename Bind::realtype *p = reinterpret_cast<typename Bind::realtype *>(smartptr.getPointer());
if (p) {
const typename Bind::traittype &t = *reinterpret_cast<const typename Bind::traittype *>(&desc->traits.d);
auto subtype_a = get_func_a<typename Bind::realtype>(p);
auto subtype_b = get_func_b<typename Bind::realtype>(p);
auto a_defval = get_1st_value<typename Bind::valuetype, double>(t.def);
auto b_defval = get_2nd_value<typename Bind::valuetype, double>(t.def);
printf("a->set_param_default: double (%g, %g)\n", a_defval, b_defval);
(subtype_a)->setDefaultValue(a_defval);
(subtype_b)->setDefaultValue(b_defval);
}
return true;
}
};
/* Default Color */
template <>
struct set_param_default_t<tpbind_col_t, std::true_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
tpbind_col_t::realtype *p = reinterpret_cast<tpbind_col_t::realtype *>(smartptr.getPointer());
if (p) {
const tpbind_col_t::traittype &t = *reinterpret_cast<const tpbind_col_t::traittype *>(&desc->traits.d);
p->setDefaultValue(TPixel32(t.def.c0, t.def.c1, t.def.c2, t.def.m));
}
return true;
}
};
/* Default String */
template <>
struct set_param_default_t<tpbind_str_t, std::true_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
tpbind_str_t::realtype *p = reinterpret_cast<tpbind_str_t::realtype *>(smartptr.getPointer());
if (p) {
const tpbind_str_t::traittype &t = *reinterpret_cast<const tpbind_str_t::traittype *>(&desc->traits.d);
printf("a->set_param_default: str\n");
std::wstring wstr = QString::fromStdString(t.def).toStdWString();
p->setDefaultValue(wstr);
p->setValue(wstr, false);
}
return true;
}
};
/* Default Spectrum */
template <>
struct set_param_default_t<tpbind_spc_t, std::true_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
/* unfortunatly, TSpectrumParam's default values must be set within the constructor, for now.
see param_factory_< TSpectrumParam >() */
return false;
}
};
/* Default ToneCurve */
template <>
struct set_param_default_t<tpbind_tcv_t, std::true_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
/*
auto smartptr = param->param();
tpbind_tcv_t::realtype* p = reinterpret_cast< tpbind_tcv_t::realtype* >(smartptr.getPointer());
if (p) {
const tpbind_tcv_t::traittype& t = *reinterpret_cast< const tpbind_tcv_t::traittype* >(&desc->traits.d);
printf("a->set_param_default: spec\n");
QList< TPointD > pt;
for (int i = 0; i < t.cps; i ++) {
pt.push_back(TPointD(t.array[i].x, t.array[i].y));
}
p->setDefaultValue(pt);
p->setIsLinear(!(t.intep));
}*/
return true;
}
};
/* primitive: TDoubleParam */
template <>
struct set_param_default_t<tpbind_dbl_t, std::false_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
tpbind_dbl_t::realtype *p = reinterpret_cast<tpbind_dbl_t::realtype *>(smartptr.getPointer());
if (p) {
const tpbind_dbl_t::traittype &t = *reinterpret_cast<const tpbind_dbl_t::traittype *>(&desc->traits.d);
printf("p(%p)->set_param_default: typeid:%s desc:%p (%p)\n", p, typeid(tpbind_dbl_t::traittype).name(), desc, &desc->traits.d);
p->setDefaultValue(t.def);
}
return true;
}
};
/* primitive: TNotAnimatableParam */
template <typename Bind>
struct set_param_default_t<Bind, std::false_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
typename Bind::realtype *p = reinterpret_cast<typename Bind::realtype *>(smartptr.getPointer());
if (p) {
const typename Bind::traittype &t = *reinterpret_cast<const typename Bind::traittype *>(&desc->traits.d);
printf("p(%p)->set_param_default: typeid:%s desc:%p (%p)\n", p, typeid(typename Bind::traittype).name(), desc, &desc->traits.d);
p->setDefaultValue(t.def);
p->setValue(t.def, false);
}
return true;
}
};
/* Default Enum */
template <>
struct set_param_default_t<tpbind_enm_t, std::false_type> {
static bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
auto smartptr = param->param();
tpbind_enm_t::realtype *p = reinterpret_cast<tpbind_enm_t::realtype *>(smartptr.getPointer());
if (p) {
const tpbind_enm_t::traittype &t = *reinterpret_cast<const tpbind_enm_t::traittype *>(&desc->traits.d);
for (int i = 0; i < t.enums; i++) {
p->addItem(i, t.array[i]);
}
p->setValue(t.def, false);
}
return true;
}
};
template <typename Bind>
bool set_param_default(Param *param, const toonz_param_desc_t *desc)
{
if (!is_type_of<typename Bind::traittype>(desc))
return false;
return set_param_default_t<Bind>::set_param_default(param, desc);
}
template <typename T>
inline T *param_factory_(const toonz_param_desc_t *desc)
{
return new T;
}
template <>
inline TPointParam *param_factory_(const toonz_param_desc_t *desc)
{
return new TPointParam(TPointD(), true /* instanciate from plugin */);
}
template <>
inline TSpectrumParam *param_factory_(const toonz_param_desc_t *desc)
{
const toonz_param_traits_spectrum_t &t = desc->traits.g;
if (t.points) {
std::vector<TSpectrum::ColorKey> keys(t.points);
for (int i = 0; i < t.points; i++) {
keys[i].first = t.array[i].w;
keys[i].second = toPixel32(TPixelD(t.array[i].c0, t.array[i].c1, t.array[i].c2, t.array[i].m));
}
return new TSpectrumParam(t.points, keys.data());
} else {
return new TSpectrumParam(); /* use default constructor: デフォルトでは [black:white] の単純なものが設定される */
}
}
inline TParam *parameter_factory(const toonz_param_desc_t *desc)
{
switch (desc->traits_tag) {
case TOONZ_PARAM_TYPE_DOUBLE:
return param_factory_<TDoubleParam>(desc);
case TOONZ_PARAM_TYPE_RANGE:
return param_factory_<TRangeParam>(desc);
case TOONZ_PARAM_TYPE_PIXEL:
return param_factory_<TPixelParam>(desc);
case TOONZ_PARAM_TYPE_POINT:
return param_factory_<TPointParam>(desc);
case TOONZ_PARAM_TYPE_ENUM:
return param_factory_<TIntEnumParam>(desc);
case TOONZ_PARAM_TYPE_INT:
return param_factory_<TIntParam>(desc);
case TOONZ_PARAM_TYPE_BOOL:
return param_factory_<TBoolParam>(desc);
case TOONZ_PARAM_TYPE_SPECTRUM:
return param_factory_<TSpectrumParam>(desc);
case TOONZ_PARAM_TYPE_STRING:
return param_factory_<TStringParam>(desc);
case TOONZ_PARAM_TYPE_TONECURVE:
return param_factory_<TToneCurveParam>(desc);
default:
break;
}
return NULL;
}
template <typename T>
inline int check_pollution_(const T &t)
{
if (t.reserved_)
return TOONZ_PARAM_ERROR_POLLUTED;
return 0;
}
template <typename T>
inline int check_traits_sanity_(const toonz_param_desc_t *desc)
{
const T &t = reinterpret_cast<const T &>(desc->traits.d);
return check_pollution_<T>(t);
}
template <>
inline int check_traits_sanity_<toonz_param_traits_double_t>(const toonz_param_desc_t *desc)
{
int err = 0;
const toonz_param_traits_double_t &t = desc->traits.d;
err |= check_pollution_(t);
if (t.min > t.max)
err |= TOONZ_PARAM_ERROR_MIN_MAX;
return 0;
}
template <>
inline int check_traits_sanity_<toonz_param_traits_range_t>(const toonz_param_desc_t *desc)
{
int err = 0;
const toonz_param_traits_range_t &t = desc->traits.rd;
err |= check_pollution_(t);
if (t.minmax.a == 0 && t.minmax.b == 0)
return err; /* range に興味がない場合の 0,0 を許す */
if (t.minmax.a > t.minmax.b)
err |= TOONZ_PARAM_ERROR_MIN_MAX;
return err;
}
template <>
inline int check_traits_sanity_<toonz_param_traits_enum_t>(const toonz_param_desc_t *desc)
{
int err = 0;
const toonz_param_traits_enum_t &t = desc->traits.e;
err |= check_pollution_(t);
if (t.enums == 0)
return err;
if (t.enums < 0)
err |= TOONZ_PARAM_ERROR_ARRAY_NUM;
if (t.array == NULL)
err |= TOONZ_PARAM_ERROR_ARRAY;
return err;
}
template <>
inline int check_traits_sanity_<toonz_param_traits_spectrum_t>(const toonz_param_desc_t *desc)
{
int err = 0;
const toonz_param_traits_spectrum_t &t = desc->traits.g;
err |= check_pollution_(t);
if (t.points == 0)
return err;
if (t.points < 0)
err |= TOONZ_PARAM_ERROR_ARRAY_NUM;
if (t.array == NULL)
err |= TOONZ_PARAM_ERROR_ARRAY;
return err;
}
/*
template <> int check_traits_sanity_< toonz_param_traits_tonecurve_t >(const toonz_param_desc_t* desc)
{
int err = 0;
const toonz_param_traits_tonecurve_t& t = desc->traits.tcv;
err |= check_pollution_(t);
if (t.points == 0)
return err;
if (t.points < 0)
err |= TOONZ_PARAM_ERROR_ARRAY_NUM;
if (t.array == NULL)
err |= TOONZ_PARAM_ERROR_ARRAY;
return err;
}
*/
inline int check_traits_sanity(const toonz_param_desc_t *desc)
{
int err = 0;
switch (desc->traits_tag) {
case TOONZ_PARAM_TYPE_DOUBLE:
err = check_traits_sanity_<toonz_param_traits_double_t>(desc);
break;
case TOONZ_PARAM_TYPE_RANGE:
err = check_traits_sanity_<toonz_param_traits_range_t>(desc);
break;
case TOONZ_PARAM_TYPE_PIXEL:
err = check_traits_sanity_<toonz_param_traits_color_t>(desc);
break;
case TOONZ_PARAM_TYPE_POINT:
err = check_traits_sanity_<toonz_param_traits_point_t>(desc);
break;
case TOONZ_PARAM_TYPE_ENUM:
err = check_traits_sanity_<toonz_param_traits_enum_t>(desc);
break;
case TOONZ_PARAM_TYPE_INT:
err = check_traits_sanity_<toonz_param_traits_int_t>(desc);
break;
case TOONZ_PARAM_TYPE_BOOL:
err = check_traits_sanity_<toonz_param_traits_bool_t>(desc);
break;
case TOONZ_PARAM_TYPE_SPECTRUM:
err = check_traits_sanity_<toonz_param_traits_spectrum_t>(desc);
break;
case TOONZ_PARAM_TYPE_STRING:
err = check_traits_sanity_<toonz_param_traits_string_t>(desc);
break;
case TOONZ_PARAM_TYPE_TONECURVE:
err = check_traits_sanity_<toonz_param_traits_tonecurve_t>(desc);
break;
default:
err = TOONZ_PARAM_ERROR_TRAITS;
break;
}
return err;
}
template <typename T>
inline bool param_type_check_(TParam *p, const toonz_param_desc_t *desc, size_t &vsz)
{
if (typename T::realtype *d = dynamic_cast<typename T::realtype *>(p)) {
if (is_type_of<typename T::traittype>(desc)) {
vsz = sizeof(typename T::traittype::iovaluetype);
return true;
}
}
return false;
}
inline bool parameter_type_check(TParam *p, const toonz_param_desc_t *desc, size_t &vsz)
{
switch (desc->traits_tag) {
case TOONZ_PARAM_TYPE_DOUBLE:
return param_type_check_<tpbind_dbl_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_RANGE:
return param_type_check_<tpbind_rng_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_PIXEL:
return param_type_check_<tpbind_col_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_POINT:
return param_type_check_<tpbind_pnt_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_ENUM:
return param_type_check_<tpbind_enm_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_INT:
return param_type_check_<tpbind_int_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_BOOL:
return param_type_check_<tpbind_bool_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_SPECTRUM:
return param_type_check_<tpbind_spc_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_STRING:
return param_type_check_<tpbind_str_t>(p, desc, vsz);
case TOONZ_PARAM_TYPE_TONECURVE:
return param_type_check_<tpbind_tcv_t>(p, desc, vsz);
default:
break;
}
return false;
}
template <typename T>
inline bool param_read_value_(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
/* isize は iovaluetype の size でなく count になったのでサイズチェックは無効 */
//if (isize == sizeof(typename T::traittype::iovaluetype)) {
auto r = reinterpret_cast<typename T::realtype *>(p);
auto v = r->getValue();
*reinterpret_cast<typename T::traittype::iovaluetype *>(ptr) = v;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_dbl_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
//if (isize == sizeof(tpbind_dbl_t::traittype::iovaluetype)) {
auto r = reinterpret_cast<tpbind_dbl_t::realtype *>(p);
auto v = r->getValue(frame);
*reinterpret_cast<tpbind_dbl_t::traittype::iovaluetype *>(ptr) = v;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_str_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
auto r = reinterpret_cast<tpbind_str_t::realtype *>(p);
const std::string str = QString::fromStdWString(r->getValue()).toStdString();
std::size_t len = str.length() + 1;
/* get_type() の返す大きさも文字列長+1 を含んでいる */
if (isize < len)
len = isize; /* 要求サイズが実際の長さより短くても良いが切り詰める(ただし 1 以上であること) */
if (len > 0) {
auto dst = reinterpret_cast<char *>(ptr);
strncpy(dst, str.c_str(), len - 1);
dst[len - 1] = '\0';
osize = len;
return true;
}
return false;
}
template <>
inline bool param_read_value_<tpbind_rng_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
//if (isize == sizeof(tpbind_rng_t::traittype::iovaluetype)) {
auto r = reinterpret_cast<tpbind_rng_t::realtype *>(p);
auto v = r->getValue(frame);
auto dst = reinterpret_cast<tpbind_rng_t::traittype::iovaluetype *>(ptr);
dst->a = v.first;
dst->b = v.second;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_col_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
//if (isize == sizeof(tpbind_col_t::traittype::iovaluetype)) {
auto r = reinterpret_cast<tpbind_col_t::realtype *>(p);
/* getValueD() だと 16bit * 4 が返る */
//auto v = r->getValueD(frame);
auto v = r->getValue(frame);
auto dst = reinterpret_cast<tpbind_col_t::traittype::iovaluetype *>(ptr);
dst->c0 = v.r;
dst->c1 = v.g;
dst->c2 = v.b;
dst->m = v.m;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_pnt_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
//if (isize == sizeof(tpbind_pnt_t::traittype::iovaluetype)) {
auto r = reinterpret_cast<tpbind_pnt_t::realtype *>(p);
auto v = r->getValue(frame);
auto dst = reinterpret_cast<tpbind_pnt_t::traittype::iovaluetype *>(ptr);
dst->x = v.x;
dst->y = v.y;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_spc_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
//if (isize == sizeof(tpbind_spc_t::traittype::iovaluetype)) {
auto r = reinterpret_cast<tpbind_spc_t::realtype *>(p);
auto dst = reinterpret_cast<tpbind_spc_t::traittype::iovaluetype *>(ptr);
/* getValue64() だと 1channle 16bit が返るがデフォルト型に合わせる */
auto v = r->getValue(frame).getValue(dst->w);
dst->c0 = v.r;
dst->c1 = v.g;
dst->c2 = v.b;
dst->m = v.m;
osize = 1;
return true;
//}
return false;
}
template <>
inline bool param_read_value_<tpbind_tcv_t>(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
auto r = reinterpret_cast<tpbind_tcv_t::realtype *>(p);
QList<TPointD> points = r->getValue(frame);
size_t ps = points.size();
/* コントロールポイントのリストしか戻って来ない! */
if (isize >= ps) {
int channel = r->getCurrentChannel();
int interp = !r->isLinear();
int c = isize < points.size() ? isize : points.size();
for (int i = 0; i < c; i++) {
auto dst = reinterpret_cast<tpbind_tcv_t::traittype::iovaluetype *>(ptr);
dst[i].x = points[i].x;
dst[i].y = points[i].y;
dst[i].channel = channel;
dst[i].interp = interp;
}
osize = c;
return true;
}
return false;
}
inline bool parameter_read_value(TParam *p, const toonz_param_desc_t *desc, void *ptr, double frame, size_t isize, size_t &osize)
{
size_t sz = 0;
if (!parameter_type_check(p, desc, sz)) { // typecheck
return false;
}
switch (desc->traits_tag) {
case TOONZ_PARAM_TYPE_DOUBLE:
return param_read_value_<tpbind_dbl_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_RANGE:
return param_read_value_<tpbind_rng_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_PIXEL:
return param_read_value_<tpbind_col_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_POINT:
return param_read_value_<tpbind_pnt_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_ENUM:
return param_read_value_<tpbind_enm_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_INT:
return param_read_value_<tpbind_int_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_BOOL:
return param_read_value_<tpbind_bool_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_SPECTRUM:
return param_read_value_<tpbind_spc_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_STRING:
return param_read_value_<tpbind_str_t>(p, desc, ptr, frame, isize, osize);
case TOONZ_PARAM_TYPE_TONECURVE:
return param_read_value_<tpbind_tcv_t>(p, desc, ptr, frame, isize, osize);
default:
break;
}
return false;
}
#endif