--- /dev/null
+/* === S Y N F I G ========================================================= */
+/*! \file valuenode_composite.cpp
+** \brief Implementation of the "Composite" valuenode conversion.
+**
+** $Id$
+**
+** \legal
+** Copyright (c) 2002-2005 Robert B. Quattlebaum Jr., Adrian Bentley
+** Copyright (c) 2008 Chris Moore
+**
+** This package is free software; you can redistribute it and/or
+** modify it under the terms of the GNU General Public License as
+** published by the Free Software Foundation; either version 2 of
+** the License, or (at your option) any later version.
+**
+** This package is distributed in the hope that it will be useful,
+** but WITHOUT ANY WARRANTY; without even the implied warranty of
+** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+** General Public License for more details.
+** \endlegal
+*/
+/* ========================================================================= */
+
+/* === H E A D E R S ======================================================= */
+
+#ifdef USING_PCH
+# include "pch.h"
+#else
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
+
+#include "valuenode_composite.h"
+#include "valuenode_const.h"
+#include <stdexcept>
+#include "general.h"
+#include "valuenode_radialcomposite.h"
+#include "vector.h"
+#include "color.h"
+#include "segment.h"
+#include "savecanvas.h"
+
+#endif
+
+/* === U S I N G =========================================================== */
+
+using namespace std;
+using namespace etl;
+using namespace synfig;
+
+/* === M A C R O S ========================================================= */
+
+/* === G L O B A L S ======================================================= */
+
+/* === P R O C E D U R E S ================================================= */
+
+/* === M E T H O D S ======================================================= */
+
+synfig::ValueNode_Composite::ValueNode_Composite(const ValueBase &value):
+ LinkableValueNode(value.get_type())
+{
+ switch(get_type())
+ {
+ case ValueBase::TYPE_VECTOR:
+ set_link("x",ValueNode_Const::create(value.get(Vector())[0]));
+ set_link("y",ValueNode_Const::create(value.get(Vector())[1]));
+ break;
+ case ValueBase::TYPE_COLOR:
+ set_link("r",ValueNode_Const::create(value.get(Color()).get_r()));
+ set_link("g",ValueNode_Const::create(value.get(Color()).get_g()));
+ set_link("b",ValueNode_Const::create(value.get(Color()).get_b()));
+ set_link("a",ValueNode_Const::create(value.get(Color()).get_a()));
+ break;
+ case ValueBase::TYPE_SEGMENT:
+ set_link("p1",ValueNode_Const::create(value.get(Segment()).p1));
+ set_link("t1",ValueNode_Const::create(value.get(Segment()).t1));
+ set_link("p2",ValueNode_Const::create(value.get(Segment()).p2));
+ set_link("t2",ValueNode_Const::create(value.get(Segment()).t2));
+ break;
+ case ValueBase::TYPE_BLINEPOINT:
+ {
+ BLinePoint bline_point(value);
+ set_link(0,ValueNode_Const::create(bline_point.get_vertex()));
+ set_link(1,ValueNode_Const::create(bline_point.get_width()));
+ set_link(2,ValueNode_Const::create(bline_point.get_origin()));
+ set_link(3,ValueNode_Const::create(bline_point.get_split_tangent_flag()));
+ set_link(4,ValueNode_RadialComposite::create(bline_point.get_tangent1()));
+ set_link(5,ValueNode_RadialComposite::create(bline_point.get_tangent2()));
+ break;
+ }
+ default:
+ assert(0);
+ throw Exception::BadType(ValueBase::type_local_name(get_type()));
+ }
+}
+
+ValueNode_Composite::~ValueNode_Composite()
+{
+ unlink_all();
+}
+
+ValueNode_Composite*
+ValueNode_Composite::create(const ValueBase &value)
+{
+ return new ValueNode_Composite(value);
+}
+
+LinkableValueNode*
+ValueNode_Composite::create_new()const
+{
+ return new ValueNode_Composite(ValueBase(get_type()));
+}
+
+ValueBase
+synfig::ValueNode_Composite::operator()(Time t)const
+{
+ if (getenv("SYNFIG_DEBUG_VALUENODE_OPERATORS"))
+ printf("%s:%d operator()\n", __FILE__, __LINE__);
+
+ switch(get_type())
+ {
+ case ValueBase::TYPE_VECTOR:
+ {
+ Vector vect;
+ assert(components[0] && components[1]);
+ vect[0]=(*components[0])(t).get(Vector::value_type());
+ vect[1]=(*components[1])(t).get(Vector::value_type());
+ return vect;
+ }
+ case ValueBase::TYPE_COLOR:
+ {
+ Color color;
+ assert(components[0] && components[1] && components[2] && components[3]);
+ color.set_r((*components[0])(t).get(Vector::value_type()));
+ color.set_g((*components[1])(t).get(Vector::value_type()));
+ color.set_b((*components[2])(t).get(Vector::value_type()));
+ color.set_a((*components[3])(t).get(Vector::value_type()));
+ return color;
+ }
+ case ValueBase::TYPE_SEGMENT:
+ {
+ Segment seg;
+ assert(components[0] && components[1] && components[2] && components[3]);
+ seg.p1=(*components[0])(t).get(Point());
+ seg.t1=(*components[1])(t).get(Vector());
+ seg.p2=(*components[2])(t).get(Point());
+ seg.t2=(*components[3])(t).get(Vector());
+ return seg;
+ }
+ case ValueBase::TYPE_BLINEPOINT:
+ {
+ BLinePoint ret;
+ assert(components[0] && components[1] && components[2] && components[3] && components[4] && components[5]);
+ ret.set_vertex((*components[0])(t).get(Point()));
+ ret.set_width((*components[1])(t).get(Real()));
+ ret.set_origin((*components[2])(t).get(Real()));
+ ret.set_split_tangent_flag((*components[3])(t).get(bool()));
+ ret.set_tangent1((*components[4])(t).get(Vector()));
+ if(ret.get_split_tangent_flag())
+ ret.set_tangent2((*components[5])(t).get(Vector()));
+ return ret;
+ }
+ default:
+ synfig::error(string("ValueNode_Composite::operator():")+_("Bad type for composite"));
+ assert(components[0]);
+ return (*components[0])(t);
+ }
+}
+
+int
+ValueNode_Composite::link_count()const
+{
+ switch(get_type())
+ {
+ case ValueBase::TYPE_VECTOR:
+ return 2;
+ case ValueBase::TYPE_COLOR:
+ return 4;
+ case ValueBase::TYPE_SEGMENT:
+ return 4;
+ case ValueBase::TYPE_BLINEPOINT:
+ return 6;
+ default:
+ synfig::warning(string("ValueNode_Composite::component_count():")+_("Bad type for composite"));
+ return 1;
+ }
+}
+
+bool
+ValueNode_Composite::set_link_vfunc(int i,ValueNode::Handle x)
+{
+ assert(i>=0 && i<link_count());
+
+ if(PlaceholderValueNode::Handle::cast_dynamic(x))
+ {
+ components[i]=x;
+ return true;
+ }
+
+ switch(get_type())
+ {
+ case ValueBase::TYPE_VECTOR:
+ if(x->get_type()==ValueBase(Real()).get_type() || PlaceholderValueNode::Handle::cast_dynamic(x))
+ {
+ components[i]=x;
+ return true;
+ }
+ break;
+
+ case ValueBase::TYPE_COLOR:
+ if(x->get_type()==ValueBase(Real()).get_type() || PlaceholderValueNode::Handle::cast_dynamic(x))
+ {
+ components[i]=x;
+ return true;
+ }
+ break;
+
+ case ValueBase::TYPE_SEGMENT:
+ if(x->get_type()==ValueBase(Point()).get_type() || PlaceholderValueNode::Handle::cast_dynamic(x))
+ {
+ components[i]=x;
+ return true;
+ }
+ break;
+
+ case ValueBase::TYPE_BLINEPOINT:
+ if((i==0 || i==4 || i==5) && x->get_type()==ValueBase(Point()).get_type())
+ {
+ components[i]=x;
+ return true;
+ }
+ if((i==1 || i==2) && x->get_type()==ValueBase(Real()).get_type())
+ {
+ components[i]=x;
+ return true;
+ }
+ if(i==3 && x->get_type()==ValueBase(bool()).get_type())
+ {
+ components[i]=x;
+ return true;
+ }
+ break;
+
+ default:
+ break;
+ }
+ return false;
+}
+
+ValueNode::LooseHandle
+ValueNode_Composite::get_link_vfunc(int i)const
+{
+ assert(i>=0 && i<link_count());
+
+ return components[i];
+}
+
+String
+ValueNode_Composite::link_local_name(int i)const
+{
+ assert(i>=0 && i<link_count());
+
+ switch(get_type())
+ {
+ case ValueBase::TYPE_VECTOR:
+ return strprintf("%c-Axis",'X'+i);
+
+ case ValueBase::TYPE_COLOR:
+ if(i==0)
+ return _("Red");
+ else if(i==1)
+ return _("Green");
+ else if(i==2)
+ return _("Blue");
+ else if(i==3)
+ return _("Alpha");
+
+ case ValueBase::TYPE_SEGMENT:
+ if(i==0)
+ return _("Vertex 1");
+ else if(i==1)
+ return _("Tangent 1");
+ else if(i==2)
+ return _("Vertex 2");
+ else if(i==3)
+ return _("Tangent 2");
+
+ case ValueBase::TYPE_BLINEPOINT:
+ if(i==0)
+ return _("Vertex");
+ else if(i==1)
+ return _("Width");
+ else if(i==2)
+ return _("Origin");
+ else if(i==3)
+ return _("Split Tangents");
+ else if(i==4)
+ return _("Tangent 1");
+ else if(i==5)
+ return _("Tangent 2");
+
+ default:
+ assert(0);
+ // notice that Composite counts from 1 and Radial Composite counts
+ // from 0! we need to keep it like that to correctly load old
+ // animations, but let's not save "c%d" format link names in future
+ return etl::strprintf(_("C%d"),i+1);
+ }
+}
+
+
+String
+ValueNode_Composite::link_name(int i)const
+{
+ assert(i>=0 && i<link_count());
+
+ if (get_file_version() < RELEASE_VERSION_0_61_08)
+ return strprintf("c%d",i+1);
+
+ switch(get_type())
+ {
+ case ValueBase::TYPE_COLOR:
+ switch(i)
+ {
+ case 0: return "red";
+ case 1: return "green";
+ case 2: return "blue";
+ case 3: return "alpha";
+ }
+ break;
+ case ValueBase::TYPE_SEGMENT:
+ switch(i)
+ {
+ case 0: return "p1";
+ case 1: return "t1";
+ case 2: return "p2";
+ case 3: return "t2";
+ }
+ break;
+ case ValueBase::TYPE_VECTOR:
+ switch(i)
+ {
+ case 0: return "x";
+ case 1: return "y";
+ }
+ break;
+ case ValueBase::TYPE_BLINEPOINT:
+ switch(i)
+ {
+ case 0: return "point";
+ case 1: return "width";
+ case 2: return "origin";
+ case 3: return "split";
+ case 4: return "t1";
+ case 5: return "t2";
+ }
+ break;
+ default:
+ break;
+ }
+
+ assert(0);
+ // notice that Composite counts from 1 and Radial Composite counts
+ // from 0! we need to keep it like that to correctly load old
+ // animations, but let's not save "c%d" format link names in future
+ return strprintf("c%d",i+1);
+}
+
+int
+ValueNode_Composite::get_link_index_from_name(const String &name)const
+{
+ if(name.empty())
+ throw Exception::BadLinkName(name);
+
+ if(name[0]=='c' && name.size() == 2 && name[1]-'1' >= 0 && name[1]-'1' < link_count())
+ return name[1]-'1';
+
+ switch(get_type())
+ {
+ case ValueBase::TYPE_COLOR:
+ if(name[0]=='r')
+ return 0;
+ if(name[0]=='g')
+ return 1;
+ if(name[0]=='b')
+ return 2;
+ if(name[0]=='a')
+ return 3;
+ case ValueBase::TYPE_SEGMENT:
+ if(name=="p1")
+ return 0;
+ if(name=="t1")
+ return 1;
+ if(name=="p2")
+ return 2;
+ if(name=="t2")
+ return 3;
+ case ValueBase::TYPE_VECTOR:
+ if(name[0]=='x')
+ return 0;
+ if(name[0]=='y')
+ return 1;
+ if(name[0]=='z') // \todo "z"? really?
+ return 2;
+ case ValueBase::TYPE_BLINEPOINT:
+ if(name[0]=='p' || name=="v1" || name=="p1")
+ return 0;
+ if(name=="w" || name=="width")
+ return 1;
+ if(name=="o" || name=="origin")
+ return 2;
+ if(name=="split")
+ return 3;
+ if(name=="t1")
+ return 4;
+ if(name=="t2")
+ return 5;
+ default:
+ break;
+ }
+
+ throw Exception::BadLinkName(name);
+}
+
+String
+ValueNode_Composite::get_name()const
+{
+ return "composite";
+}
+
+String
+ValueNode_Composite::get_local_name()const
+{
+ return _("Composite");
+}
+
+bool
+ValueNode_Composite::check_type(ValueBase::Type type)
+{
+ return
+ type==ValueBase::TYPE_SEGMENT ||
+ type==ValueBase::TYPE_VECTOR ||
+ type==ValueBase::TYPE_COLOR ||
+ type==ValueBase::TYPE_BLINEPOINT;
+}