Home Reference Source

src/Plugins/SortAnimation/SortAnimation.js

  1. import AbstractPlugin from 'shared/AbstractPlugin';
  2.  
  3. const onSortableSorted = Symbol('onSortableSorted');
  4. const onSortableSort = Symbol('onSortableSort');
  5.  
  6. /**
  7. * SortAnimation default options
  8. * @property {Object} defaultOptions
  9. * @property {Number} defaultOptions.duration
  10. * @property {String} defaultOptions.easingFunction
  11. * @type {Object}
  12. */
  13. export const defaultOptions = {
  14. duration: 150,
  15. easingFunction: 'ease-in-out',
  16. };
  17.  
  18. /**
  19. * SortAnimation plugin adds sort animation for sortable
  20. * @class SortAnimation
  21. * @module SortAnimation
  22. * @extends AbstractPlugin
  23. */
  24. export default class SortAnimation extends AbstractPlugin {
  25. /**
  26. * SortAnimation constructor.
  27. * @constructs SortAnimation
  28. * @param {Draggable} draggable - Draggable instance
  29. */
  30. constructor(draggable) {
  31. super(draggable);
  32.  
  33. /**
  34. * SortAnimation options
  35. * @property {Object} options
  36. * @property {Number} defaultOptions.duration
  37. * @property {String} defaultOptions.easingFunction
  38. * @type {Object}
  39. */
  40. this.options = {
  41. ...defaultOptions,
  42. ...this.getOptions(),
  43. };
  44.  
  45. /**
  46. * Last animation frame
  47. * @property {Number} lastAnimationFrame
  48. * @type {Number}
  49. */
  50. this.lastAnimationFrame = null;
  51. this.lastElements = [];
  52.  
  53. this[onSortableSorted] = this[onSortableSorted].bind(this);
  54. this[onSortableSort] = this[onSortableSort].bind(this);
  55. }
  56.  
  57. /**
  58. * Attaches plugins event listeners
  59. */
  60. attach() {
  61. this.draggable.on('sortable:sort', this[onSortableSort]);
  62. this.draggable.on('sortable:sorted', this[onSortableSorted]);
  63. }
  64.  
  65. /**
  66. * Detaches plugins event listeners
  67. */
  68. detach() {
  69. this.draggable.off('sortable:sort', this[onSortableSort]);
  70. this.draggable.off('sortable:sorted', this[onSortableSorted]);
  71. }
  72.  
  73. /**
  74. * Returns options passed through draggable
  75. * @return {Object}
  76. */
  77. getOptions() {
  78. return this.draggable.options.sortAnimation || {};
  79. }
  80.  
  81. /**
  82. * Sortable sort handler
  83. * @param {SortableSortEvent} sortableEvent
  84. * @private
  85. */
  86. [onSortableSort]({dragEvent}) {
  87. const {sourceContainer} = dragEvent;
  88. const elements = this.draggable.getDraggableElementsForContainer(sourceContainer);
  89. this.lastElements = Array.from(elements).map((el) => {
  90. return {
  91. domEl: el,
  92. offsetTop: el.offsetTop,
  93. offsetLeft: el.offsetLeft,
  94. };
  95. });
  96. }
  97.  
  98. /**
  99. * Sortable sorted handler
  100. * @param {SortableSortedEvent} sortableEvent
  101. * @private
  102. */
  103. [onSortableSorted]({oldIndex, newIndex}) {
  104. if (oldIndex === newIndex) {
  105. return;
  106. }
  107.  
  108. const effectedElements = [];
  109. let start;
  110. let end;
  111. let num;
  112. if (oldIndex > newIndex) {
  113. start = newIndex;
  114. end = oldIndex - 1;
  115. num = 1;
  116. } else {
  117. start = oldIndex + 1;
  118. end = newIndex;
  119. num = -1;
  120. }
  121.  
  122. for (let i = start; i <= end; i++) {
  123. const from = this.lastElements[i];
  124. const to = this.lastElements[i + num];
  125. effectedElements.push({from, to});
  126. }
  127. cancelAnimationFrame(this.lastAnimationFrame);
  128.  
  129. // Can be done in a separate frame
  130. this.lastAnimationFrame = requestAnimationFrame(() => {
  131. effectedElements.forEach((element) => animate(element, this.options));
  132. });
  133. }
  134. }
  135.  
  136. /**
  137. * Animates two elements
  138. * @param {Object} element
  139. * @param {Object} element.from
  140. * @param {Object} element.to
  141. * @param {Object} options
  142. * @param {Number} options.duration
  143. * @param {String} options.easingFunction
  144. * @private
  145. */
  146. function animate({from, to}, {duration, easingFunction}) {
  147. const domEl = from.domEl;
  148. const x = from.offsetLeft - to.offsetLeft;
  149. const y = from.offsetTop - to.offsetTop;
  150.  
  151. domEl.style.pointerEvents = 'none';
  152. domEl.style.transform = `translate3d(${x}px, ${y}px, 0)`;
  153.  
  154. requestAnimationFrame(() => {
  155. domEl.addEventListener('transitionend', resetElementOnTransitionEnd);
  156. domEl.style.transition = `transform ${duration}ms ${easingFunction}`;
  157. domEl.style.transform = '';
  158. });
  159. }
  160.  
  161. /**
  162. * Resets animation style properties after animation has completed
  163. * @param {Event} event
  164. * @private
  165. */
  166. function resetElementOnTransitionEnd(event) {
  167. event.target.style.transition = '';
  168. event.target.style.pointerEvents = '';
  169. event.target.removeEventListener('transitionend', resetElementOnTransitionEnd);
  170. }