easing.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. #include "pocketpy/easing.h"
  2. namespace pkpy{
  3. // https://easings.net/
  4. const double kPi = 3.1415926545;
  5. static double easeLinear( double x ) {
  6. return x;
  7. }
  8. static double easeInSine( double x ) {
  9. return 1.0 - std::cos( x * kPi / 2 );
  10. }
  11. static double easeOutSine( double x ) {
  12. return std::sin( x * kPi / 2 );
  13. }
  14. static double easeInOutSine( double x ) {
  15. return -( std::cos( kPi * x ) - 1 ) / 2;
  16. }
  17. static double easeInQuad( double x ) {
  18. return x * x;
  19. }
  20. static double easeOutQuad( double x ) {
  21. return 1 - std::pow( 1 - x, 2 );
  22. }
  23. static double easeInOutQuad( double x ) {
  24. if( x < 0.5 ) {
  25. return 2 * x * x;
  26. } else {
  27. return 1 - std::pow( -2 * x + 2, 2 ) / 2;
  28. }
  29. }
  30. static double easeInCubic( double x ) {
  31. return x * x * x;
  32. }
  33. static double easeOutCubic( double x ) {
  34. return 1 - std::pow( 1 - x, 3 );
  35. }
  36. static double easeInOutCubic( double x ) {
  37. if( x < 0.5 ) {
  38. return 4 * x * x * x;
  39. } else {
  40. return 1 - std::pow( -2 * x + 2, 3 ) / 2;
  41. }
  42. }
  43. static double easeInQuart( double x ) {
  44. return std::pow( x, 4 );
  45. }
  46. static double easeOutQuart( double x ) {
  47. return 1 - std::pow( 1 - x, 4 );
  48. }
  49. static double easeInOutQuart( double x ) {
  50. if( x < 0.5 ) {
  51. return 8 * std::pow( x, 4 );
  52. } else {
  53. return 1 - std::pow( -2 * x + 2, 4 ) / 2;
  54. }
  55. }
  56. static double easeInQuint( double x ) {
  57. return std::pow( x, 5 );
  58. }
  59. static double easeOutQuint( double x ) {
  60. return 1 - std::pow( 1 - x, 5 );
  61. }
  62. static double easeInOutQuint( double x ) {
  63. if( x < 0.5 ) {
  64. return 16 * std::pow( x, 5 );
  65. } else {
  66. return 1 - std::pow( -2 * x + 2, 5 ) / 2;
  67. }
  68. }
  69. static double easeInExpo( double x ) {
  70. return x == 0 ? 0 : std::pow( 2, 10 * x - 10 );
  71. }
  72. static double easeOutExpo( double x ) {
  73. return x == 1 ? 1 : 1 - std::pow( 2, -10 * x );
  74. }
  75. static double easeInOutExpo( double x ) {
  76. if( x == 0 ) {
  77. return 0;
  78. } else if( x == 1 ) {
  79. return 1;
  80. } else if( x < 0.5 ) {
  81. return std::pow( 2, 20 * x - 10 ) / 2;
  82. } else {
  83. return (2 - std::pow( 2, -20 * x + 10 )) / 2;
  84. }
  85. }
  86. static double easeInCirc( double x ) {
  87. return 1 - std::sqrt( 1 - std::pow( x, 2 ) );
  88. }
  89. static double easeOutCirc( double x ) {
  90. return std::sqrt( 1 - std::pow( x - 1, 2 ) );
  91. }
  92. static double easeInOutCirc( double x ) {
  93. if( x < 0.5 ) {
  94. return (1 - std::sqrt( 1 - std::pow( 2 * x, 2 ) )) / 2;
  95. } else {
  96. return (std::sqrt( 1 - std::pow( -2 * x + 2, 2 ) ) + 1) / 2;
  97. }
  98. }
  99. static double easeInBack( double x ) {
  100. const double c1 = 1.70158;
  101. const double c3 = c1 + 1;
  102. return c3 * x * x * x - c1 * x * x;
  103. }
  104. static double easeOutBack( double x ) {
  105. const double c1 = 1.70158;
  106. const double c3 = c1 + 1;
  107. return 1 + c3 * std::pow( x - 1, 3 ) + c1 * std::pow( x - 1, 2 );
  108. }
  109. static double easeInOutBack( double x ) {
  110. const double c1 = 1.70158;
  111. const double c2 = c1 * 1.525;
  112. if( x < 0.5 ) {
  113. return (std::pow( 2 * x, 2 ) * ((c2 + 1) * 2 * x - c2)) / 2;
  114. } else {
  115. return (std::pow( 2 * x - 2, 2 ) * ((c2 + 1) * (x * 2 - 2) + c2) + 2) / 2;
  116. }
  117. }
  118. static double easeInElastic( double x ) {
  119. const double c4 = (2 * kPi) / 3;
  120. if( x == 0 ) {
  121. return 0;
  122. } else if( x == 1 ) {
  123. return 1;
  124. } else {
  125. return -std::pow( 2, 10 * x - 10 ) * std::sin( (x * 10 - 10.75) * c4 );
  126. }
  127. }
  128. static double easeOutElastic( double x ) {
  129. const double c4 = (2 * kPi) / 3;
  130. if( x == 0 ) {
  131. return 0;
  132. } else if( x == 1 ) {
  133. return 1;
  134. } else {
  135. return std::pow( 2, -10 * x ) * std::sin( (x * 10 - 0.75) * c4 ) + 1;
  136. }
  137. }
  138. static double easeInOutElastic( double x ) {
  139. const double c5 = (2 * kPi) / 4.5;
  140. if( x == 0 ) {
  141. return 0;
  142. } else if( x == 1 ) {
  143. return 1;
  144. } else if( x < 0.5 ) {
  145. return -(std::pow( 2, 20 * x - 10 ) * std::sin( (20 * x - 11.125) * c5 )) / 2;
  146. } else {
  147. return (std::pow( 2, -20 * x + 10 ) * std::sin( (20 * x - 11.125) * c5 )) / 2 + 1;
  148. }
  149. }
  150. static double easeOutBounce( double x ) {
  151. const double n1 = 7.5625;
  152. const double d1 = 2.75;
  153. if( x < 1 / d1 ) {
  154. return n1 * x * x;
  155. } else if( x < 2 / d1 ) {
  156. x -= 1.5 / d1;
  157. return n1 * x * x + 0.75;
  158. } else if( x < 2.5 / d1 ) {
  159. x -= 2.25 / d1;
  160. return n1 * x * x + 0.9375;
  161. } else {
  162. x -= 2.625 / d1;
  163. return n1 * x * x + 0.984375;
  164. }
  165. }
  166. static double easeInBounce( double x ) {
  167. return 1 - easeOutBounce(1 - x);
  168. }
  169. static double easeInOutBounce( double x ) {
  170. return x < 0.5
  171. ? (1 - easeOutBounce(1 - 2 * x)) / 2
  172. : (1 + easeOutBounce(2 * x - 1)) / 2;
  173. }
  174. void add_module_easing(VM* vm){
  175. PyObject* mod = vm->new_module("easing");
  176. #define EASE(name) \
  177. vm->bind_func(mod, #name, 1, [](VM* vm, ArgsView args){ \
  178. f64 t = CAST(f64, args[0]); \
  179. return VAR(ease##name(t)); \
  180. });
  181. EASE(Linear)
  182. EASE(InSine)
  183. EASE(OutSine)
  184. EASE(InOutSine)
  185. EASE(InQuad)
  186. EASE(OutQuad)
  187. EASE(InOutQuad)
  188. EASE(InCubic)
  189. EASE(OutCubic)
  190. EASE(InOutCubic)
  191. EASE(InQuart)
  192. EASE(OutQuart)
  193. EASE(InOutQuart)
  194. EASE(InQuint)
  195. EASE(OutQuint)
  196. EASE(InOutQuint)
  197. EASE(InExpo)
  198. EASE(OutExpo)
  199. EASE(InOutExpo)
  200. EASE(InCirc)
  201. EASE(OutCirc)
  202. EASE(InOutCirc)
  203. EASE(InBack)
  204. EASE(OutBack)
  205. EASE(InOutBack)
  206. EASE(InElastic)
  207. EASE(OutElastic)
  208. EASE(InOutElastic)
  209. EASE(InBounce)
  210. EASE(OutBounce)
  211. EASE(InOutBounce)
  212. #undef EASE
  213. }
  214. } // namespace pkpy