moreupdates
[synfig.git] / synfig-core / trunk / src / synfig / valuenode_segcalctangent.cpp
1 /* === S Y N F I G ========================================================= */
2 /*!     \file valuenode_segcalctangent.cpp
3 **      \brief Template File
4 **
5 **      $Id: valuenode_segcalctangent.cpp,v 1.1.1.1 2005/01/04 01:23:15 darco Exp $
6 **
7 **      \legal
8 **      Copyright (c) 2002 Robert B. Quattlebaum Jr.
9 **
10 **      This software and associated documentation
11 **      are CONFIDENTIAL and PROPRIETARY property of
12 **      the above-mentioned copyright holder.
13 **
14 **      You may not copy, print, publish, or in any
15 **      other way distribute this software without
16 **      a prior written agreement with
17 **      the copyright holder.
18 **      \endlegal
19 */
20 /* ========================================================================= */
21
22 /* === H E A D E R S ======================================================= */
23
24 #ifdef USING_PCH
25 #       include "pch.h"
26 #else
27 #ifdef HAVE_CONFIG_H
28 #       include <config.h>
29 #endif
30
31 #include "valuenode_segcalctangent.h"
32 #include "valuenode_const.h"
33 #include "valuenode_composite.h"
34 #include "general.h"
35 #include "exception.h"
36 #include <ETL/hermite>
37 #include <ETL/calculus>
38 #include "segment.h"
39
40 #endif
41
42 /* === U S I N G =========================================================== */
43
44 using namespace std;
45 using namespace etl;
46 using namespace synfig;
47
48 /* === M A C R O S ========================================================= */
49
50 /* === G L O B A L S ======================================================= */
51
52 /* === P R O C E D U R E S ================================================= */
53
54 /* === M E T H O D S ======================================================= */
55
56 ValueNode_SegCalcTangent::ValueNode_SegCalcTangent(const ValueBase::Type &x):
57         LinkableValueNode(x)
58 {
59         if(x!=ValueBase::TYPE_VECTOR)
60                 throw Exception::BadType(ValueBase::type_name(x));
61         
62         segment_=ValueNode_Composite::create(ValueBase::TYPE_SEGMENT);
63         amount_=ValueNode_Const::create(Real(0.5));
64 }
65
66 ValueNode_SegCalcTangent*
67 ValueNode_SegCalcTangent::create(const ValueBase &x)
68 {
69         return new ValueNode_SegCalcTangent(x.get_type());
70 }
71
72 ValueNode_SegCalcTangent::~ValueNode_SegCalcTangent()
73 {
74         unlink_all();
75 }
76
77 ValueBase
78 ValueNode_SegCalcTangent::operator()(Time t)const
79 {
80         Segment segment((*segment_)(t).get(Segment()));
81
82         etl::hermite<Vector> curve(segment.p1,segment.p2,segment.t1,segment.t2);
83         etl::derivative< etl::hermite<Vector> > deriv(curve);
84         
85 #ifdef ETL_FIXED_DERIVATIVE
86         return deriv((*amount_)(t).get(Real()))*(0.5);
87 #else
88         return deriv((*amount_)(t).get(Real()))*(-0.5);
89 #endif
90         
91 }
92
93
94 String
95 ValueNode_SegCalcTangent::get_name()const
96 {
97         return "segcalctangent";
98 }
99
100 String
101 ValueNode_SegCalcTangent::get_local_name()const
102 {
103         return _("SegCalcTangent");
104 }
105                 
106 bool
107 ValueNode_SegCalcTangent::check_type(ValueBase::Type type)
108 {
109         return type==ValueBase::TYPE_VECTOR;
110 }
111
112 bool
113 ValueNode_SegCalcTangent::set_link_vfunc(int i,ValueNode::Handle x)
114 {
115         assert(i==0 || i==1);
116         if(i==0)
117         {
118                 segment_=x;
119                 signal_child_changed()(i);signal_value_changed()();
120                 return true;
121         }
122         if(i==1)
123         {
124                 amount_=x;
125                 signal_child_changed()(i);signal_value_changed()();
126                 return true;
127         }
128         return false;
129 }
130
131 ValueNode::LooseHandle
132 ValueNode_SegCalcTangent::get_link_vfunc(int i)const
133 {
134         assert(i==0 || i==1);
135         if(i==0)
136                 return segment_;
137         if(i==1)
138                 return amount_;
139
140         return 0;
141 }
142
143 int
144 ValueNode_SegCalcTangent::link_count()const
145 {
146         return 2;
147 }
148
149 String
150 ValueNode_SegCalcTangent::link_name(int i)const
151 {
152         assert(i==0 || i==1);
153         if(i==0)
154                 return "segment";
155         if(i==1)
156                 return "amount";
157         return String();
158 }
159
160 String
161 ValueNode_SegCalcTangent::link_local_name(int i)const
162 {
163         assert(i==0 || i==1);
164         if(i==0)
165                 return _("Segment");
166         if(i==1)
167                 return _("Amount");
168         return String();
169 }
170
171 int
172 ValueNode_SegCalcTangent::get_link_index_from_name(const String &name)const
173 {
174         if(name=="segment")
175                 return 0;
176         if(name=="amount")
177                 return 1;
178         
179         throw Exception::BadLinkName(name);
180 }
181
182 LinkableValueNode*
183 ValueNode_SegCalcTangent::create_new()const
184 {
185         return new ValueNode_SegCalcTangent(ValueBase::TYPE_VECTOR);
186 }