UNPKG

19.9 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3var index_1 = require("../index");
4var concat_1 = require("./concat");
5function interpolate(y, from, to) {
6 return (from * (1 - y) + to * y);
7}
8function flip(fn) {
9 return function (x) { return 1 - fn(1 - x); };
10}
11function createEasing(fn) {
12 var fnFlipped = flip(fn);
13 return {
14 easeIn: function (x, from, to) {
15 return interpolate(fn(x), from, to);
16 },
17 easeOut: function (x, from, to) {
18 return interpolate(fnFlipped(x), from, to);
19 },
20 easeInOut: function (x, from, to) {
21 var y = (x < 0.5) ?
22 (fn(2 * x) * 0.5) :
23 (0.5 + fnFlipped(2 * (x - 0.5)) * 0.5);
24 return interpolate(y, from, to);
25 }
26 };
27}
28var easingPower2 = createEasing(function (x) { return x * x; });
29var easingPower3 = createEasing(function (x) { return x * x * x; });
30var easingPower4 = createEasing(function (x) {
31 var xx = x * x;
32 return xx * xx;
33});
34var EXP_WEIGHT = 6;
35var EXP_MAX = Math.exp(EXP_WEIGHT) - 1;
36function expFn(x) {
37 return (Math.exp(x * EXP_WEIGHT) - 1) / EXP_MAX;
38}
39var easingExponential = createEasing(expFn);
40var OVERSHOOT = 1.70158;
41var easingBack = createEasing(function (x) { return x * x * ((OVERSHOOT + 1) * x - OVERSHOOT); });
42var PARAM1 = 7.5625;
43var PARAM2 = 2.75;
44function easeOutFn(x) {
45 var z = x;
46 if (z < 1 / PARAM2) {
47 return (PARAM1 * z * z);
48 }
49 else if (z < 2 / PARAM2) {
50 return (PARAM1 * (z -= 1.5 / PARAM2) * z + 0.75);
51 }
52 else if (z < 2.5 / PARAM2) {
53 return (PARAM1 * (z -= 2.25 / PARAM2) * z + 0.9375);
54 }
55 else {
56 return (PARAM1 * (z -= 2.625 / PARAM2) * z + 0.984375);
57 }
58}
59var easingBounce = createEasing(function (x) { return 1 - easeOutFn(1 - x); });
60var easingCirc = createEasing(function (x) { return -(Math.sqrt(1 - x * x) - 1); });
61var PERIOD = 0.3;
62var OVERSHOOT_ELASTIC = PERIOD / 4;
63var AMPLITUDE = 1;
64function elasticIn(x) {
65 var z = x;
66 if (z <= 0) {
67 return 0;
68 }
69 else if (z >= 1) {
70 return 1;
71 }
72 else {
73 z -= 1;
74 return -(AMPLITUDE * Math.pow(2, 10 * z))
75 * Math.sin((z - OVERSHOOT_ELASTIC) * (2 * Math.PI) / PERIOD);
76 }
77}
78var easingElastic = createEasing(elasticIn);
79var HALF_PI = Math.PI * 0.5;
80var easingSine = createEasing(function (x) { return 1 - Math.cos(x * HALF_PI); });
81var DEFAULT_INTERVAL = 15;
82/**
83 * Creates a stream of numbers emitted in a quick burst, following a numeric
84 * function like sine or elastic or quadratic. tween() is meant for creating
85 * streams for animations.
86 *
87 * Example:
88 *
89 * ```js
90 * import tween from 'xstream/extra/tween'
91 *
92 * const stream = tween({
93 * from: 20,
94 * to: 100,
95 * ease: tween.exponential.easeIn,
96 * duration: 1000, // milliseconds
97 * })
98 *
99 * stream.addListener({
100 * next: (x) => console.log(x),
101 * error: (err) => console.error(err),
102 * complete: () => console.log('concat completed'),
103 * })
104 * ```
105 *
106 * The stream would behave like the plot below:
107 *
108 * ```text
109 * 100 #
110 * |
111 * |
112 * |
113 * |
114 * 80 #
115 * |
116 * |
117 * |
118 * | #
119 * 60
120 * |
121 * | #
122 * |
123 * | #
124 * 40
125 * | #
126 * | #
127 * | ##
128 * | ###
129 * 20########
130 * +---------------------> time
131 * ```
132 *
133 * Provide a configuration object with **from**, **to**, **duration**, **ease**,
134 * **interval** (optional), and this factory function will return a stream of
135 * numbers following that pattern. The first number emitted will be `from`, and
136 * the last number will be `to`. The numbers in between follow the easing
137 * function you specify in `ease`, and the stream emission will last in total
138 * `duration` milliseconds.
139 *
140 * The easing functions are attached to `tween` too, such as
141 * `tween.linear.ease`, `tween.power2.easeIn`, `tween.exponential.easeOut`, etc.
142 * Here is a list of all the available easing options:
143 *
144 * - `tween.linear` with ease
145 * - `tween.power2` with easeIn, easeOut, easeInOut
146 * - `tween.power3` with easeIn, easeOut, easeInOut
147 * - `tween.power4` with easeIn, easeOut, easeInOut
148 * - `tween.exponential` with easeIn, easeOut, easeInOut
149 * - `tween.back` with easeIn, easeOut, easeInOut
150 * - `tween.bounce` with easeIn, easeOut, easeInOut
151 * - `tween.circular` with easeIn, easeOut, easeInOut
152 * - `tween.elastic` with easeIn, easeOut, easeInOut
153 * - `tween.sine` with easeIn, easeOut, easeInOut
154 *
155 * @factory true
156 * @param {TweenConfig} config An object with properties `from: number`,
157 * `to: number`, `duration: number`, `ease: function` (optional, defaults to
158 * linear), `interval: number` (optional, defaults to 15).
159 * @return {Stream}
160 */
161function tween(_a) {
162 var from = _a.from, to = _a.to, duration = _a.duration, _b = _a.ease, ease = _b === void 0 ? tweenFactory.linear.ease : _b, _c = _a.interval, interval = _c === void 0 ? DEFAULT_INTERVAL : _c;
163 var totalTicks = Math.round(duration / interval);
164 return index_1.Stream.periodic(interval)
165 .take(totalTicks)
166 .map(function (tick) { return ease(tick / totalTicks, from, to); })
167 .compose(function (s) { return concat_1.default(s, index_1.Stream.of(to)); });
168}
169var tweenFactory = tween;
170tweenFactory.linear = { ease: interpolate };
171tweenFactory.power2 = easingPower2;
172tweenFactory.power3 = easingPower3;
173tweenFactory.power4 = easingPower4;
174tweenFactory.exponential = easingExponential;
175tweenFactory.back = easingBack;
176tweenFactory.bounce = easingBounce;
177tweenFactory.circular = easingCirc;
178tweenFactory.elastic = easingElastic;
179tweenFactory.sine = easingSine;
180exports.default = tweenFactory;
181//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tween.js","sourceRoot":"","sources":["../src/extra/tween.ts"],"names":[],"mappings":"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","sourcesContent":["import {Stream} from '../index';\nimport concat from './concat';\n\nexport type Ease = (x: number, from: number, to: number) => number;\nexport type Easings = {\n  easeIn: Ease;\n  easeOut: Ease;\n  easeInOut: Ease;\n};\n\nexport type NumericFunction = (input: number) => number;\n\nexport interface TweenConfig {\n  from: number;\n  to: number;\n  duration: number;\n  ease?: Ease;\n  interval?: number;\n}\n\nfunction interpolate(y: number, from: number, to: number): number {\n  return (from * (1 - y) + to * y);\n}\n\nfunction flip(fn: NumericFunction): NumericFunction {\n  return x => 1 - fn(1 - x);\n}\n\nfunction createEasing(fn: NumericFunction): Easings {\n  let fnFlipped = flip(fn);\n  return {\n    easeIn(x, from, to) {\n      return interpolate(fn(x), from, to);\n    },\n    easeOut(x, from, to) {\n      return interpolate(fnFlipped(x), from, to);\n    },\n    easeInOut(x, from, to) {\n      const y = (x < 0.5) ?\n        (fn(2 * x) * 0.5) :\n        (0.5 + fnFlipped(2 * (x - 0.5)) * 0.5);\n      return interpolate(y, from, to);\n    }\n  };\n}\n\nlet easingPower2 = createEasing(x => x * x);\nlet easingPower3 = createEasing(x => x * x * x);\nlet easingPower4 = createEasing(x => {\n  const xx = x * x;\n  return xx * xx;\n});\n\nconst EXP_WEIGHT = 6;\nconst EXP_MAX = Math.exp(EXP_WEIGHT) - 1;\nfunction expFn(x: number): number {\n  return (Math.exp(x * EXP_WEIGHT) - 1) / EXP_MAX;\n}\nlet easingExponential = createEasing(expFn);\n\nconst OVERSHOOT = 1.70158;\nlet easingBack = createEasing(x => x * x * ((OVERSHOOT + 1) * x - OVERSHOOT));\n\nconst PARAM1 = 7.5625;\nconst PARAM2 = 2.75;\nfunction easeOutFn(x: number): number {\n  let z = x;\n  if (z < 1 / PARAM2) {\n    return (PARAM1 * z * z);\n  } else if (z < 2 / PARAM2) {\n    return (PARAM1 * (z -= 1.5 / PARAM2) * z + 0.75);\n  } else if (z < 2.5 / PARAM2) {\n    return (PARAM1 * (z -= 2.25 / PARAM2) * z + 0.9375);\n  } else {\n    return (PARAM1 * (z -= 2.625 / PARAM2) * z + 0.984375);\n  }\n}\nlet easingBounce = createEasing(x => 1 - easeOutFn(1 - x));\n\nlet easingCirc = createEasing(x => -(Math.sqrt(1 - x * x) - 1));\n\nconst PERIOD = 0.3;\nconst OVERSHOOT_ELASTIC = PERIOD / 4;\nconst AMPLITUDE = 1;\nfunction elasticIn(x: number): number {\n  let z = x;\n  if (z <= 0) {\n    return 0;\n  } else if (z >= 1) {\n    return 1;\n  } else {\n    z -= 1;\n    return -(AMPLITUDE * Math.pow(2, 10 * z))\n      * Math.sin((z - OVERSHOOT_ELASTIC) * (2 * Math.PI) / PERIOD);\n  }\n}\nlet easingElastic = createEasing(elasticIn);\n\nconst HALF_PI = Math.PI * 0.5;\nlet easingSine = createEasing(x => 1 - Math.cos(x * HALF_PI));\n\nconst DEFAULT_INTERVAL: number = 15;\n\nexport interface TweenFactory {\n  (config: TweenConfig): Stream<number>;\n  linear: { ease: Ease };\n  power2: Easings;\n  power3: Easings;\n  power4: Easings;\n  exponential: Easings;\n  back: Easings;\n  bounce: Easings;\n  circular: Easings;\n  elastic: Easings;\n  sine: Easings;\n}\n\n/**\n * Creates a stream of numbers emitted in a quick burst, following a numeric\n * function like sine or elastic or quadratic. tween() is meant for creating\n * streams for animations.\n *\n * Example:\n *\n * ```js\n * import tween from 'xstream/extra/tween'\n *\n * const stream = tween({\n *   from: 20,\n *   to: 100,\n *   ease: tween.exponential.easeIn,\n *   duration: 1000, // milliseconds\n * })\n *\n * stream.addListener({\n *   next: (x) => console.log(x),\n *   error: (err) => console.error(err),\n *   complete: () => console.log('concat completed'),\n * })\n * ```\n *\n * The stream would behave like the plot below:\n *\n * ```text\n * 100                  #\n * |\n * |\n * |\n * |\n * 80                  #\n * |\n * |\n * |\n * |                  #\n * 60\n * |\n * |                 #\n * |\n * |                #\n * 40\n * |               #\n * |              #\n * |            ##\n * |         ###\n * 20########\n * +---------------------> time\n * ```\n *\n * Provide a configuration object with **from**, **to**, **duration**, **ease**,\n * **interval** (optional), and this factory function will return a stream of\n * numbers following that pattern. The first number emitted will be `from`, and\n * the last number will be `to`. The numbers in between follow the easing\n * function you specify in `ease`, and the stream emission will last in total\n * `duration` milliseconds.\n *\n * The easing functions are attached to `tween` too, such as\n * `tween.linear.ease`, `tween.power2.easeIn`, `tween.exponential.easeOut`, etc.\n * Here is a list of all the available easing options:\n *\n * - `tween.linear` with ease\n * - `tween.power2` with easeIn, easeOut, easeInOut\n * - `tween.power3` with easeIn, easeOut, easeInOut\n * - `tween.power4` with easeIn, easeOut, easeInOut\n * - `tween.exponential` with easeIn, easeOut, easeInOut\n * - `tween.back` with easeIn, easeOut, easeInOut\n * - `tween.bounce` with easeIn, easeOut, easeInOut\n * - `tween.circular` with easeIn, easeOut, easeInOut\n * - `tween.elastic` with easeIn, easeOut, easeInOut\n * - `tween.sine` with easeIn, easeOut, easeInOut\n *\n * @factory true\n * @param {TweenConfig} config An object with properties `from: number`,\n * `to: number`, `duration: number`, `ease: function` (optional, defaults to\n * linear), `interval: number` (optional, defaults to 15).\n * @return {Stream}\n */\nfunction tween({\n  from,\n  to,\n  duration,\n  ease = tweenFactory.linear.ease,\n  interval = DEFAULT_INTERVAL\n}: TweenConfig): Stream<number> {\n  const totalTicks = Math.round(duration / interval);\n  return Stream.periodic(interval)\n    .take(totalTicks)\n    .map(tick => ease(tick / totalTicks, from, to))\n    .compose(s => concat<number>(s, Stream.of(to)));\n}\n\nconst tweenFactory: TweenFactory = <TweenFactory> tween;\n\ntweenFactory.linear = { ease: interpolate };\ntweenFactory.power2 = easingPower2;\ntweenFactory.power3 = easingPower3;\ntweenFactory.power4 = easingPower4;\ntweenFactory.exponential = easingExponential;\ntweenFactory.back = easingBack;\ntweenFactory.bounce = easingBounce;\ntweenFactory.circular = easingCirc;\ntweenFactory.elastic = easingElastic;\ntweenFactory.sine = easingSine;\n\nexport default tweenFactory;\n"]}
\No newline at end of file