Curve.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. // MIT License - Copyright (C) The Mono.Xna Team
  2. // This file is subject to the terms and conditions defined in
  3. // file 'LICENSE.txt', which is part of this source code package.
  4. using System;
  5. using System.ComponentModel;
  6. using System.Runtime.Serialization;
  7. namespace CommonLang.Geometry
  8. {
  9. /// <summary>
  10. /// Contains a collection of <see cref="CurveKey"/> points in 2D space and provides methods for evaluating features of the curve they define.
  11. /// </summary>
  12. // TODO : [TypeConverter(typeof(ExpandableObjectConverter))]
  13. public class Curve
  14. {
  15. #region Private Fields
  16. private CurveKeyCollection _keys;
  17. private CurveLoopType _postLoop;
  18. private CurveLoopType _preLoop;
  19. #endregion
  20. #region Public Properties
  21. /// <summary>
  22. /// Returns <c>true</c> if this curve is constant (has zero or one points); <c>false</c> otherwise.
  23. /// </summary>
  24. public bool IsConstant
  25. {
  26. get { return this._keys.Count <= 1; }
  27. }
  28. /// <summary>
  29. /// The collection of curve keys.
  30. /// </summary>
  31. public CurveKeyCollection Keys
  32. {
  33. get { return this._keys; }
  34. }
  35. /// <summary>
  36. /// Defines how to handle weighting values that are greater than the last control point in the curve.
  37. /// </summary>
  38. public CurveLoopType PostLoop
  39. {
  40. get { return this._postLoop; }
  41. set { this._postLoop = value; }
  42. }
  43. /// <summary>
  44. /// Defines how to handle weighting values that are less than the first control point in the curve.
  45. /// </summary>
  46. public CurveLoopType PreLoop
  47. {
  48. get { return this._preLoop; }
  49. set { this._preLoop = value; }
  50. }
  51. #endregion
  52. #region Public Constructors
  53. /// <summary>
  54. /// Constructs a curve.
  55. /// </summary>
  56. public Curve()
  57. {
  58. this._keys = new CurveKeyCollection();
  59. }
  60. #endregion
  61. #region Public Methods
  62. /// <summary>
  63. /// Creates a copy of this curve.
  64. /// </summary>
  65. /// <returns>A copy of this curve.</returns>
  66. public Curve Clone()
  67. {
  68. Curve curve = new Curve();
  69. curve._keys = this._keys.Clone();
  70. curve._preLoop = this._preLoop;
  71. curve._postLoop = this._postLoop;
  72. return curve;
  73. }
  74. /// <summary>
  75. /// Evaluate the value at a position of this <see cref="Curve"/>.
  76. /// </summary>
  77. /// <param name="position">The position on this <see cref="Curve"/>.</param>
  78. /// <returns>Value at the position on this <see cref="Curve"/>.</returns>
  79. public float Evaluate(float position)
  80. {
  81. CurveKey first = _keys[0];
  82. CurveKey last = _keys[_keys.Count - 1];
  83. if (position < first.Position)
  84. {
  85. switch (this.PreLoop)
  86. {
  87. case CurveLoopType.Constant:
  88. //constant
  89. return first.Value;
  90. case CurveLoopType.Linear:
  91. // linear y = a*x +b with a tangeant of last point
  92. return first.Value - first.TangentIn * (first.Position - position);
  93. case CurveLoopType.Cycle:
  94. //start -> end / start -> end
  95. int cycle = GetNumberOfCycle(position);
  96. float virtualPos = position - (cycle * (last.Position - first.Position));
  97. return GetCurvePosition(virtualPos);
  98. case CurveLoopType.CycleOffset:
  99. //make the curve continue (with no step) so must up the curve each cycle of delta(value)
  100. cycle = GetNumberOfCycle(position);
  101. virtualPos = position - (cycle * (last.Position - first.Position));
  102. return (GetCurvePosition(virtualPos) + cycle * (last.Value - first.Value));
  103. case CurveLoopType.Oscillate:
  104. //go back on curve from end and target start
  105. // start-> end / end -> start
  106. cycle = GetNumberOfCycle(position);
  107. if (0 == cycle % 2f)//if pair
  108. virtualPos = position - (cycle * (last.Position - first.Position));
  109. else
  110. virtualPos = last.Position - position + first.Position + (cycle * (last.Position - first.Position));
  111. return GetCurvePosition(virtualPos);
  112. }
  113. }
  114. else if (position > last.Position)
  115. {
  116. int cycle;
  117. switch (this.PostLoop)
  118. {
  119. case CurveLoopType.Constant:
  120. //constant
  121. return last.Value;
  122. case CurveLoopType.Linear:
  123. // linear y = a*x +b with a tangeant of last point
  124. return last.Value + first.TangentOut * (position - last.Position);
  125. case CurveLoopType.Cycle:
  126. //start -> end / start -> end
  127. cycle = GetNumberOfCycle(position);
  128. float virtualPos = position - (cycle * (last.Position - first.Position));
  129. return GetCurvePosition(virtualPos);
  130. case CurveLoopType.CycleOffset:
  131. //make the curve continue (with no step) so must up the curve each cycle of delta(value)
  132. cycle = GetNumberOfCycle(position);
  133. virtualPos = position - (cycle * (last.Position - first.Position));
  134. return (GetCurvePosition(virtualPos) + cycle * (last.Value - first.Value));
  135. case CurveLoopType.Oscillate:
  136. //go back on curve from end and target start
  137. // start-> end / end -> start
  138. cycle = GetNumberOfCycle(position);
  139. virtualPos = position - (cycle * (last.Position - first.Position));
  140. if (0 == cycle % 2f)//if pair
  141. virtualPos = position - (cycle * (last.Position - first.Position));
  142. else
  143. virtualPos = last.Position - position + first.Position + (cycle * (last.Position - first.Position));
  144. return GetCurvePosition(virtualPos);
  145. }
  146. }
  147. //in curve
  148. return GetCurvePosition(position);
  149. }
  150. /// <summary>
  151. /// Computes tangents for all keys in the collection.
  152. /// </summary>
  153. /// <param name="tangentType">The tangent type for both in and out.</param>
  154. public void ComputeTangents (CurveTangent tangentType)
  155. {
  156. ComputeTangents(tangentType, tangentType);
  157. }
  158. /// <summary>
  159. /// Computes tangents for all keys in the collection.
  160. /// </summary>
  161. /// <param name="tangentInType">The tangent in-type. <see cref="CurveKey.TangentIn"/> for more details.</param>
  162. /// <param name="tangentOutType">The tangent out-type. <see cref="CurveKey.TangentOut"/> for more details.</param>
  163. public void ComputeTangents(CurveTangent tangentInType, CurveTangent tangentOutType)
  164. {
  165. for (var i = 0; i < Keys.Count; ++i)
  166. {
  167. ComputeTangent(i, tangentInType, tangentOutType);
  168. }
  169. }
  170. /// <summary>
  171. /// Computes tangent for the specific key in the collection.
  172. /// </summary>
  173. /// <param name="keyIndex">The index of a key in the collection.</param>
  174. /// <param name="tangentType">The tangent type for both in and out.</param>
  175. public void ComputeTangent(int keyIndex, CurveTangent tangentType)
  176. {
  177. ComputeTangent(keyIndex, tangentType, tangentType);
  178. }
  179. /// <summary>
  180. /// Computes tangent for the specific key in the collection.
  181. /// </summary>
  182. /// <param name="keyIndex">The index of key in the collection.</param>
  183. /// <param name="tangentInType">The tangent in-type. <see cref="CurveKey.TangentIn"/> for more details.</param>
  184. /// <param name="tangentOutType">The tangent out-type. <see cref="CurveKey.TangentOut"/> for more details.</param>
  185. public void ComputeTangent(int keyIndex, CurveTangent tangentInType, CurveTangent tangentOutType)
  186. {
  187. // See http://msdn.microsoft.com/en-us/library/microsoft.xna.framework.curvetangent.aspx
  188. var key = _keys[keyIndex];
  189. float p0, p, p1;
  190. p0 = p = p1 = key.Position;
  191. float v0, v, v1;
  192. v0 = v = v1 = key.Value;
  193. if ( keyIndex > 0 )
  194. {
  195. p0 = _keys[keyIndex - 1].Position;
  196. v0 = _keys[keyIndex - 1].Value;
  197. }
  198. if (keyIndex < _keys.Count-1)
  199. {
  200. p1 = _keys[keyIndex + 1].Position;
  201. v1 = _keys[keyIndex + 1].Value;
  202. }
  203. switch (tangentInType)
  204. {
  205. case CurveTangent.Flat:
  206. key.TangentIn = 0;
  207. break;
  208. case CurveTangent.Linear:
  209. key.TangentIn = v - v0;
  210. break;
  211. case CurveTangent.Smooth:
  212. var pn = p1 - p0;
  213. if (Math.Abs(pn) < float.Epsilon)
  214. key.TangentIn = 0;
  215. else
  216. key.TangentIn = (v1 - v0) * ((p - p0) / pn);
  217. break;
  218. }
  219. switch (tangentOutType)
  220. {
  221. case CurveTangent.Flat:
  222. key.TangentOut = 0;
  223. break;
  224. case CurveTangent.Linear:
  225. key.TangentOut = v1 - v;
  226. break;
  227. case CurveTangent.Smooth:
  228. var pn = p1 - p0;
  229. if (Math.Abs(pn) < float.Epsilon)
  230. key.TangentOut = 0;
  231. else
  232. key.TangentOut = (v1 - v0) * ((p1 - p) / pn);
  233. break;
  234. }
  235. }
  236. #endregion
  237. #region Private Methods
  238. private int GetNumberOfCycle(float position)
  239. {
  240. float cycle = (position - _keys[0].Position) / (_keys[_keys.Count - 1].Position - _keys[0].Position);
  241. if (cycle < 0f)
  242. cycle--;
  243. return (int)cycle;
  244. }
  245. private float GetCurvePosition(float position)
  246. {
  247. //only for position in curve
  248. CurveKey prev = this._keys[0];
  249. CurveKey next;
  250. for (int i = 1; i < this._keys.Count; ++i)
  251. {
  252. next = this.Keys[i];
  253. if (next.Position >= position)
  254. {
  255. if (prev.Continuity == CurveContinuity.Step)
  256. {
  257. if (position >= 1f)
  258. {
  259. return next.Value;
  260. }
  261. return prev.Value;
  262. }
  263. float t = (position - prev.Position) / (next.Position - prev.Position);//to have t in [0,1]
  264. float ts = t * t;
  265. float tss = ts * t;
  266. //After a lot of search on internet I have found all about spline function
  267. // and bezier (phi'sss ancien) but finaly use hermite curve
  268. //http://en.wikipedia.org/wiki/Cubic_Hermite_spline
  269. //P(t) = (2*t^3 - 3t^2 + 1)*P0 + (t^3 - 2t^2 + t)m0 + (-2t^3 + 3t^2)P1 + (t^3-t^2)m1
  270. //with P0.value = prev.value , m0 = prev.tangentOut, P1= next.value, m1 = next.TangentIn
  271. return (2 * tss - 3 * ts + 1f) * prev.Value + (tss - 2 * ts + t) * prev.TangentOut + (3 * ts - 2 * tss) * next.Value + (tss - ts) * next.TangentIn;
  272. }
  273. prev = next;
  274. }
  275. return 0f;
  276. }
  277. #endregion
  278. }
  279. }