bezier.js

  1. import { Tweenable } from './tweenable'
  2. /** @typedef {import(".").shifty.easingFunction} shifty.easingFunction */
  3. /**
  4. * The Bezier magic in this file is adapted/copied almost wholesale from
  5. * [Scripty2](https://github.com/madrobby/scripty2/blob/master/src/effects/transitions/cubic-bezier.js),
  6. * which was adapted from Apple code (which probably came from
  7. * [here](http://opensource.apple.com/source/WebCore/WebCore-955.66/platform/graphics/UnitBezier.h)).
  8. * Special thanks to Apple and Thomas Fuchs for much of this code.
  9. */
  10. /**
  11. * Copyright (c) 2006 Apple Computer, Inc. All rights reserved.
  12. *
  13. * Redistribution and use in source and binary forms, with or without
  14. * modification, are permitted provided that the following conditions are met:
  15. *
  16. * 1. Redistributions of source code must retain the above copyright notice,
  17. * this list of conditions and the following disclaimer.
  18. *
  19. * 2. Redistributions in binary form must reproduce the above copyright notice,
  20. * this list of conditions and the following disclaimer in the documentation
  21. * and/or other materials provided with the distribution.
  22. *
  23. * 3. Neither the name of the copyright holder(s) nor the names of any
  24. * contributors may be used to endorse or promote products derived from
  25. * this software without specific prior written permission.
  26. *
  27. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  28. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  29. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  30. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
  31. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  32. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  33. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  34. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  35. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  36. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  37. * POSSIBILITY OF SUCH DAMAGE.
  38. */
  39. // port of webkit cubic bezier handling by http://www.netzgesta.de/dev/
  40. /**
  41. * @param {number} t
  42. * @param {number} p1x
  43. * @param {number} p1y
  44. * @param {number} p2x
  45. * @param {number} p2y
  46. * @param {number} duration
  47. * @returns {Function}
  48. * @private
  49. */
  50. function cubicBezierAtTime(t, p1x, p1y, p2x, p2y, duration) {
  51. let ax = 0,
  52. bx = 0,
  53. cx = 0,
  54. ay = 0,
  55. by = 0,
  56. cy = 0
  57. const sampleCurveX = t => ((ax * t + bx) * t + cx) * t
  58. const sampleCurveY = t => ((ay * t + by) * t + cy) * t
  59. const sampleCurveDerivativeX = t => (3 * ax * t + 2 * bx) * t + cx
  60. const solveEpsilon = duration => 1 / (200 * duration)
  61. const fabs = n => (n >= 0 ? n : 0 - n)
  62. const solveCurveX = (x, epsilon) => {
  63. let t0, t1, t2, x2, d2, i
  64. for (t2 = x, i = 0; i < 8; i++) {
  65. x2 = sampleCurveX(t2) - x
  66. if (fabs(x2) < epsilon) {
  67. return t2
  68. }
  69. d2 = sampleCurveDerivativeX(t2)
  70. if (fabs(d2) < 1e-6) {
  71. break
  72. }
  73. t2 = t2 - x2 / d2
  74. }
  75. t0 = 0
  76. t1 = 1
  77. t2 = x
  78. if (t2 < t0) {
  79. return t0
  80. }
  81. if (t2 > t1) {
  82. return t1
  83. }
  84. while (t0 < t1) {
  85. x2 = sampleCurveX(t2)
  86. if (fabs(x2 - x) < epsilon) {
  87. return t2
  88. }
  89. if (x > x2) {
  90. t0 = t2
  91. } else {
  92. t1 = t2
  93. }
  94. t2 = (t1 - t0) * 0.5 + t0
  95. }
  96. return t2 // Failure.
  97. }
  98. const solve = (x, epsilon) => sampleCurveY(solveCurveX(x, epsilon))
  99. cx = 3 * p1x
  100. bx = 3 * (p2x - p1x) - cx
  101. ax = 1 - cx - bx
  102. cy = 3 * p1y
  103. by = 3 * (p2y - p1y) - cy
  104. ay = 1 - cy - by
  105. return solve(t, solveEpsilon(duration))
  106. }
  107. // End ported code
  108. /**
  109. * GetCubicBezierTransition(x1, y1, x2, y2) -> Function.
  110. *
  111. * Generates a transition easing function that is compatible
  112. * with WebKit's CSS transitions `-webkit-transition-timing-function`
  113. * CSS property.
  114. *
  115. * The W3C has more information about CSS3 transition timing functions:
  116. * http://www.w3.org/TR/css3-transitions/#transition-timing-function_tag
  117. *
  118. * @param {number} [x1]
  119. * @param {number} [y1]
  120. * @param {number} [x2]
  121. * @param {number} [y2]
  122. * @return {Function}
  123. */
  124. export const getCubicBezierTransition = (
  125. x1 = 0.25,
  126. y1 = 0.25,
  127. x2 = 0.75,
  128. y2 = 0.75
  129. ) => pos => cubicBezierAtTime(pos, x1, y1, x2, y2, 1)
  130. /**
  131. * Create a Bezier easing function and attach it to {@link
  132. * Tweenable.formulas}. This function gives you total control over the
  133. * easing curve. Matthew Lein's [Ceaser](http://matthewlein.com/ceaser/) is a
  134. * useful tool for visualizing the curves you can make with this function.
  135. * @method shifty.setBezierFunction
  136. * @param {string} name The name of the easing curve. Overwrites the old
  137. * easing function on {@link Tweenable.formulas} if it exists.
  138. * @param {number} x1
  139. * @param {number} y1
  140. * @param {number} x2
  141. * @param {number} y2
  142. * @return {shifty.easingFunction} The {@link shifty.easingFunction} that was
  143. * attached to {@link Tweenable.formulas}.
  144. */
  145. export const setBezierFunction = (name, x1, y1, x2, y2) => {
  146. const cubicBezierTransition = getCubicBezierTransition(x1, y1, x2, y2)
  147. cubicBezierTransition.displayName = name
  148. cubicBezierTransition.x1 = x1
  149. cubicBezierTransition.y1 = y1
  150. cubicBezierTransition.x2 = x2
  151. cubicBezierTransition.y2 = y2
  152. return (Tweenable.formulas[name] = cubicBezierTransition)
  153. }
  154. /**
  155. * `delete` an easing function from {@link Tweenable.formulas}. Be
  156. * careful with this method, as it `delete`s whatever easing formula matches
  157. * `name` (which means you can delete standard Shifty easing functions).
  158. * @method shifty.unsetBezierFunction
  159. * @param {string} name The name of the easing function to delete.
  160. * @return {boolean} Whether or not the functions was `delete`d.
  161. */
  162. export const unsetBezierFunction = name => delete Tweenable.formulas[name]