1. /**
  2. * @module meteoJS/timeline/animation
  3. */
  4. import $ from 'jquery';
  5. import addEventFunctions from '../Events.js';
  6. import Timeline from '../Timeline.js';
  7. /**
  8. * Options for animation constructor.
  9. *
  10. * @typedef {Object} module:meteoJS/timeline/animation~options
  11. * @param {module:meteoJS/timeline.Timeline} timeline - Timeline to animate.
  12. * @param {number} [restartPause=1.8]
  13. * Time in seconds to pause before the animation restart.
  14. * @param {number} [imagePeriod=0.2]
  15. * Time in seconds between animation of two images.
  16. * Ignored, if imageFrequency is specified.
  17. * @param {number|undefined} [imageFrequency]
  18. * Time of images during one second.
  19. * @param {boolean} [enabledStepsOnly=true] - Use only enabled times.
  20. * @param {boolean} [allEnabledStepsOnly=false]
  21. * Use only times that are enabled by all sets of time.
  22. */
  23. /**
  24. * Event on animation start.
  25. *
  26. * @event module:meteoJS/timeline/animation#start:animation
  27. */
  28. /**
  29. * Event on animation stop.
  30. *
  31. * @event module:meteoJS/timeline/animation#stop:animation
  32. */
  33. /**
  34. * Event on reaching last timestamp.
  35. *
  36. * @event module:meteoJS/timeline/animation#end:animation
  37. */
  38. /**
  39. * Event triggered immediatly before restart of animation.
  40. *
  41. * @event module:meteoJS/timeline/animation#restart:animation
  42. */
  43. /**
  44. * Event triggered when imageFrequency/imagePeriod is changed.
  45. *
  46. * @event module:meteoJS/timeline/animation#change:imageFrequency
  47. */
  48. /**
  49. * Event triggered when restartPause is changed.
  50. *
  51. * @event module:meteoJS/timeline/animation#change:restartPause
  52. */
  53. /**
  54. * Object to animate {@link module:meteoJS/timeline.Timeline}.
  55. *
  56. * <pre><code>import Animation from 'meteojs/timeline/Animation';</code></pre>
  57. */
  58. export class Animation {
  59. /**
  60. * @param {module:meteoJS/timeline/animation~options} options - Options.
  61. */
  62. constructor({ timeline,
  63. restartPause = 1.8,
  64. imagePeriod = 0.2,
  65. imageFrequency = undefined,
  66. enabledStepsOnly = true,
  67. allEnabledStepsOnly = false } = {}) {
  68. /**
  69. * @type module:meteoJS/timeline/animation~options
  70. * @private
  71. */
  72. this.options = {
  73. timeline,
  74. restartPause,
  75. imagePeriod,
  76. imageFrequency,
  77. enabledStepsOnly,
  78. allEnabledStepsOnly
  79. };
  80. // Normalize options
  81. if (this.options.timeline === undefined)
  82. this.options.timeline = new Timeline();
  83. if (this.options.imageFrequency !== undefined &&
  84. this.options.imageFrequency != 0)
  85. this.options.imagePeriod = 1/this.options.imageFrequency;
  86. /**
  87. * ID to window.setInterval() of the animation.
  88. * If undefined, there is no started animation.
  89. * @type undefined|number
  90. * @private
  91. */
  92. this.animationIntervalID = undefined;
  93. /**
  94. * ID to window.setTimeout() ot the animation (used for restart-pause).
  95. * If undefined, there is no started setTimeout (i.e. no restart-pause).
  96. * @type undefined|number
  97. * @private
  98. */
  99. this.animationTimeoutID = undefined;
  100. /**
  101. * Current position in this.times in the animation.
  102. * @type integer
  103. * @private
  104. */
  105. this.animationStep = 0;
  106. /**
  107. * Hash with timestamps-valueOf's as keys and index in this.times as values.
  108. * @type Object
  109. * @private
  110. */
  111. this.timesHash = {};
  112. /**
  113. * List of timestamps. Current list of times of the timeline to animate over.
  114. * @type Date[]
  115. * @private
  116. */
  117. this.times = [];
  118. // Timeline initialisieren
  119. let onChangeTimes = () => {
  120. this.times = this.options.timeline[this._getTimelineTimesMethod()]();
  121. this.timesHash = {};
  122. this.times.forEach((time, i) => this.timesHash[time.valueOf()] = i);
  123. };
  124. this.options.timeline.on(this._getTimelineChangeTimesEvent(), onChangeTimes);
  125. onChangeTimes();
  126. }
  127. /**
  128. * Returns time period between two animation steps (in s).
  129. *
  130. * @returns {number} Time period.
  131. */
  132. getImagePeriod() {
  133. return this.options.imagePeriod;
  134. }
  135. /**
  136. * Sets time period between to animation steps (in s)
  137. *
  138. * @param {number} imagePeriod - Time period.
  139. * @returns {module:meteoJS/timeline/animation.Animation} This.
  140. */
  141. setImagePeriod(imagePeriod) {
  142. this.options.imagePeriod = imagePeriod;
  143. if (this.isStarted())
  144. this._updateAnimation();
  145. this.trigger('change:imageFrequency');
  146. return this;
  147. }
  148. /**
  149. * Returns time frequency of animation steps (in 1/s).
  150. *
  151. * @returns {number} Time frequency.
  152. */
  153. getImageFrequency() {
  154. return 1/this.options.imagePeriod;
  155. }
  156. /**
  157. * Sets time frequency of animation steps (in 1/s).
  158. *
  159. * @param {number} imageFrequency - Time frequency.
  160. * @returns {module:meteoJS/timeline/animation.Animation} This.
  161. */
  162. setImageFrequency(imageFrequency) {
  163. if (imageFrequency != 0)
  164. this.setImagePeriod(1/imageFrequency);
  165. return this;
  166. }
  167. /**
  168. * Returns time duration before a restart (jump from end to beginning, in s).
  169. *
  170. * @returns {number} Time duration.
  171. */
  172. getRestartPause() {
  173. return this.options.restartPause;
  174. }
  175. /**
  176. * Sets time duration before a restart (in s).
  177. *
  178. * @param {number} restartPause - Time duration.
  179. * @returns {module:meteoJS/timeline/animation.Animation} This.
  180. */
  181. setRestartPause(restartPause) {
  182. this.options.restartPause = Number(restartPause); // Convert string to number
  183. this.trigger('change:restartPause');
  184. return this;
  185. }
  186. /**
  187. * Is animation started.
  188. *
  189. * @returns {boolean}
  190. */
  191. isStarted() {
  192. return this.animationIntervalID !== undefined ||
  193. this.animationTimeoutID !== undefined;
  194. }
  195. /**
  196. * Starts the animation.
  197. *
  198. * @returns {module:meteoJS/timeline/animation.Animation} This.
  199. * @fires module:meteoJS/timeline/animation#start:animation
  200. */
  201. start() {
  202. if (this.options.timeline.getSelectedTime().valueOf() in this.timesHash)
  203. this._setStep(this.timesHash[this.options.timeline.getSelectedTime().valueOf()]);
  204. if (!this.isStarted())
  205. this._updateAnimation();
  206. this.trigger('start:animation');
  207. }
  208. /**
  209. * Stops the animation.
  210. *
  211. * @returns {module:meteoJS/timeline/animation.Animation} This.
  212. * @fires module:meteoJS/timeline/animation#stop:animation
  213. */
  214. stop() {
  215. this._clearAnimation();
  216. this.trigger('stop:animation');
  217. }
  218. /**
  219. * Toggles the animation.
  220. *
  221. * @returns {module:meteoJS/timeline/animation.Animation} This.
  222. */
  223. toggle() {
  224. if (this.isStarted())
  225. this.stop();
  226. else
  227. this.start();
  228. }
  229. /**
  230. * Setzt Schritt der Animation
  231. * @private
  232. * @param {number} step
  233. */
  234. _setStep(step) {
  235. if (0 <= step && step < this._getCount())
  236. this.animationStep = step;
  237. }
  238. /**
  239. * Gibt timeline-Event Name zum abhören von Änderungen der Zeitschritte zurück.
  240. * @private
  241. * @returns {string}
  242. */
  243. _getTimelineChangeTimesEvent() {
  244. return (this.options.enabledStepsOnly || this.options.allEnabledStepsOnly)
  245. ? 'change:enabledTimes' : 'change:times';
  246. }
  247. /**
  248. * Gibt timeline-Methode aller Zeitschritte zurück.
  249. * @private
  250. * @returns {string}
  251. */
  252. _getTimelineTimesMethod() {
  253. return this.options.allEnabledStepsOnly ? 'getAllEnabledTimes' :
  254. this.options.enabledStepsOnly ? 'getEnabledTimes' : 'getTimes';
  255. }
  256. /**
  257. * Gibt Anzahl Animationsschritte zurück
  258. * @private
  259. * @returns {number}
  260. */
  261. _getCount() {
  262. return this.options.timeline[this._getTimelineTimesMethod()]().length;
  263. }
  264. /**
  265. * Handelt die Animation
  266. * @private
  267. * @fires module:meteoJS/timeline/animation#end:animation
  268. * @fires module:meteoJS/timeline/animation#restart:animation
  269. */
  270. _updateAnimation() {
  271. this._clearAnimation();
  272. if (this.animationStep < this._getCount()-1)
  273. this._initAnimation();
  274. else
  275. this._initRestartPause();
  276. }
  277. /**
  278. * Startet Animation
  279. * @private
  280. */
  281. _initAnimation() {
  282. if (this.animationIntervalID === undefined)
  283. this.animationIntervalID = window.setInterval(() => {
  284. this.animationStep++;
  285. if (this.animationStep < this.times.length)
  286. this.options.timeline.setSelectedTime(this.times[this.animationStep]);
  287. if (this.animationStep >= this._getCount()-1) {
  288. this.trigger('end:animation');
  289. this._clearAnimation();
  290. this._initRestartPause();
  291. }
  292. }, this.options.imagePeriod * 1000);
  293. }
  294. /**
  295. * Startet den Timer für die Restart-Pause
  296. * Verwende als Zeitspanne imagePeriod+restartPause. Sonst wird bei restartPause
  297. * 0s der letzte Zeitschritt gar nie angezeigt.
  298. * @private
  299. */
  300. _initRestartPause() {
  301. if (this.animationTimeoutID === undefined)
  302. this.animationTimeoutID = window.setTimeout(() => {
  303. this.animationStep = 0;
  304. this.trigger('restart:animation');
  305. if (this.animationStep < this.times.length)
  306. this.options.timeline.setSelectedTime(this.times[this.animationStep]);
  307. this._clearAnimation();
  308. this._initAnimation();
  309. }, (this.options.imagePeriod + this.options.restartPause) * 1000);
  310. }
  311. /**
  312. * Löscht window.interval, falls vorhanden
  313. * @private
  314. */
  315. _clearAnimation() {
  316. if (this.animationIntervalID !== undefined) {
  317. window.clearInterval(this.animationIntervalID);
  318. this.animationIntervalID = undefined;
  319. }
  320. if (this.animationTimeoutID !== undefined) {
  321. window.clearTimeout(this.animationTimeoutID);
  322. this.animationTimeoutID = undefined;
  323. }
  324. }
  325. }
  326. addEventFunctions(Animation.prototype);
  327. export default Animation;
  328. /**
  329. * Insert an input-group to change frequency.
  330. *
  331. * <pre><code>import { insertFrequencyInput } from 'meteojs/timeline/Animation';</code></pre>
  332. *
  333. * @param {external:jQuery} node - Node to insert input-group.
  334. * @param {Object} options - Options for input-group.
  335. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  336. * Animation object.
  337. * @param {string} [options.suffix='fps'] - Suffix text for input-group.
  338. * @returns {external:jQuery} Input-group node.
  339. */
  340. export function insertFrequencyInput(node, { animation, suffix = 'fps' }) {
  341. const number = $('<input>')
  342. .addClass('form-control')
  343. .attr('type', 'number')
  344. .attr('min', 1)
  345. .attr('step', 1);
  346. const inputGroupNumber = $('<div>')
  347. .addClass('input-group')
  348. .append(number)
  349. .append($('<span>').addClass('input-group-text').text(suffix));
  350. number.on('change', () => animation.setImageFrequency(number.val()));
  351. const onChangeImageFrequency = () => number.val(animation.getImageFrequency());
  352. animation.on('change:imageFrequency', onChangeImageFrequency);
  353. onChangeImageFrequency();
  354. node.append(inputGroupNumber);
  355. return inputGroupNumber;
  356. }
  357. /**
  358. * Insert an input-range to change frequency.
  359. *
  360. * <pre><code>import { insertFrequencyRange } from 'meteojs/timeline/Animation';</code></pre>
  361. *
  362. * @param {external:jQuery} node - Node to insert input-range.
  363. * @param {Object} options - Options for input-range.
  364. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  365. * Animation object.
  366. * @param {number[]} options.frequencies - Frequencies to select.
  367. * @returns {external:jQuery} Input-range node.
  368. */
  369. export function insertFrequencyRange(node, { animation, frequencies }) {
  370. frequencies = frequencies ? frequencies : [1];
  371. let range = $('<input>')
  372. .addClass('form-range')
  373. .attr('type', 'range')
  374. .attr('min', 0)
  375. .attr('max', frequencies.length-1);
  376. range.on('change input', () => {
  377. let i = range.val();
  378. if (i < frequencies.length)
  379. animation.setImageFrequency(frequencies[i]);
  380. });
  381. let onChangeImageFrequency = () => {
  382. let i = frequencies.indexOf(animation.getImageFrequency());
  383. if (i > -1)
  384. range.val(i);
  385. };
  386. animation.on('change:imageFrequency', onChangeImageFrequency);
  387. onChangeImageFrequency();
  388. node.append(range);
  389. return range;
  390. }
  391. /**
  392. * Insert an button-group to change frequency.
  393. *
  394. * <pre><code>import { insertFrequencyButtonGroup } from 'meteojs/timeline/Animation';</code></pre>
  395. *
  396. * @param {external:jQuery} node - Node to insert the button-group.
  397. * @param {Object} options - Options for the button-group.
  398. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  399. * Animation object.
  400. * @param {number[]} options.frequencies - Frequencies to select.
  401. * @param {string|undefined} [options.btnGroupClass='btn-group']
  402. * Class added to the button-group node.
  403. * @param {string|undefined} [options.btnClass='btn btn-primary']
  404. * Class added to each button.
  405. * @param {string} [options.suffix='fps']
  406. * Suffix text for each button after frequency.
  407. * @returns {external:jQuery} Button-group node.
  408. */
  409. export function insertFrequencyButtonGroup(node, {
  410. animation,
  411. frequencies,
  412. btnGroupClass = 'btn-group',
  413. btnClass = 'btn btn-primary',
  414. suffix = 'fps'
  415. }) {
  416. let btnGroup = $('<div>').addClass(btnGroupClass);
  417. frequencies = frequencies ? frequencies : [];
  418. frequencies.forEach(freq => {
  419. btnGroup.append($('<button>')
  420. .addClass(btnClass)
  421. .data('frequency', freq)
  422. .text(freq + ' ' + suffix)
  423. .click(() => animation.setImageFrequency(freq)));
  424. });
  425. let onChange = () => {
  426. btnGroup.children('button').removeClass('active').each(function () {
  427. if ($(this).data('frequency') == animation.getImageFrequency())
  428. $(this).addClass('active');
  429. });
  430. };
  431. animation.on('change:imageFrequency', onChange);
  432. onChange();
  433. node.append(btnGroup);
  434. return btnGroup;
  435. }
  436. /**
  437. * Insert an input-group to change restart pause.
  438. *
  439. * <pre><code>import { insertRestartPauseInput } from 'meteojs/timeline/Animation';</code></pre>
  440. *
  441. * @param {external:jQuery} node - Node to insert input-group.
  442. * @param {Object} options - Options for input-group.
  443. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  444. * Animation object.
  445. * @param {string} [options.suffix='s'] - Suffix text for input-group.
  446. * @returns {external:jQuery} Input-group node.
  447. */
  448. export function insertRestartPauseInput(node, { animation, suffix = 's' }) {
  449. const input = $('<input>')
  450. .addClass('form-control')
  451. .attr('type', 'number')
  452. .attr('min', 0)
  453. .attr('step', 0.1);
  454. const inputGroupNumber = $('<div>')
  455. .addClass('input-group')
  456. .append(input)
  457. .append($('<span>').addClass('input-group-text').text(suffix));
  458. input.on('change', () => animation.setRestartPause(input.val()));
  459. const onChange = () => input.val(animation.getRestartPause());
  460. animation.on('change:restartPause', onChange);
  461. onChange();
  462. node.append(inputGroupNumber);
  463. return inputGroupNumber;
  464. }
  465. /**
  466. * Insert an input-range to change restart pause.
  467. *
  468. * <pre><code>import { insertRestartPauseRange } from 'meteojs/timeline/Animation';</code></pre>
  469. *
  470. * @param {external:jQuery} node - Node to insert input-range.
  471. * @param {Object} options - Options for input-range.
  472. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  473. * Animation object.
  474. * @param {number[]} options.pauses - Restart pauses to select.
  475. * @returns {external:jQuery} Input-range node.
  476. */
  477. export function insertRestartPauseRange(node, { animation, pauses }) {
  478. pauses = pauses ? pauses : [1];
  479. pauses = pauses.map(p => Math.round(p * 1000));
  480. let range = $('<input>')
  481. .addClass('form-range')
  482. .attr('type', 'range')
  483. .attr('min', 0)
  484. .attr('max', pauses.length-1);
  485. range.on('change input', () => {
  486. let i = range.val();
  487. if (i < pauses.length)
  488. animation.setRestartPause(pauses[i] / 1000);
  489. });
  490. let onChangeImageFrequency = () => {
  491. let i =
  492. pauses.indexOf(Math.round(animation.getRestartPause() * 1000));
  493. if (i > -1)
  494. range.val(i);
  495. };
  496. animation.on('change:imageFrequency', onChangeImageFrequency);
  497. onChangeImageFrequency();
  498. node.append(range);
  499. return range;
  500. }
  501. /**
  502. * Insert an button-group to change restart pause.
  503. *
  504. * <pre><code>import { insertRestartPauseButtonGroup } from 'meteojs/timeline/Animation';</code></pre>
  505. *
  506. * @param {external:jQuery} node - Node to insert the button-group.
  507. * @param {Object} options - Options for the button-group.
  508. * @param {module:meteoJS/timeline/animation.Animation} options.animation
  509. * Animation object.
  510. * @param {number[]} options.pauses - Restart pauses to select.
  511. * @param {string|undefined} [options.btnGroupClass='btn-group']
  512. * Class added to the button-group node.
  513. * @param {string|undefined} [options.btnClass='btn btn-primary']
  514. * Class added to each button.
  515. * @param {string} [options.suffix='s']
  516. * Suffix in each button after duration text.
  517. * @returns {external:jQuery} Button-group node.
  518. */
  519. export function insertRestartPauseButtonGroup(node, {
  520. animation,
  521. pauses,
  522. btnGroupClass = 'btn-group',
  523. btnClass = 'btn btn-primary',
  524. suffix = 's'
  525. }) {
  526. let btnGroup = $('<div>').addClass(btnGroupClass);
  527. pauses = pauses ? pauses : [];
  528. pauses.forEach(pause => {
  529. btnGroup.append($('<button>')
  530. .addClass(btnClass)
  531. .data('pause', pause)
  532. .text(pause + ' ' + suffix)
  533. .click(() => animation.setRestartPause(pause)));
  534. });
  535. let onChange = () => {
  536. btnGroup.children('button').removeClass('active').each(function () {
  537. if ($(this).data('pause') == animation.getRestartPause())
  538. $(this).addClass('active');
  539. });
  540. };
  541. animation.on('change:restartPause', onChange);
  542. onChange();
  543. node.append(btnGroup);
  544. return btnGroup;
  545. }