754 lines
26 KiB
C
754 lines
26 KiB
C
|
#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
|