draggable.directive.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { Directive, ElementRef, Input, Output, EventEmitter } from '@angular/core';
  6. import { fromEvent } from 'rxjs';
  7. import { takeUntil } from 'rxjs/operators';
  8. /**
  9. * Draggable Directive for Angular2
  10. *
  11. * Inspiration:
  12. * https://github.com/AngularClass/angular2-examples/blob/master/rx-draggable/directives/draggable.ts
  13. * http://stackoverflow.com/questions/35662530/how-to-implement-drag-and-drop-in-angular2
  14. *
  15. */
  16. export class DraggableDirective {
  17. /**
  18. * @param {?} element
  19. */
  20. constructor(element) {
  21. this.dragX = true;
  22. this.dragY = true;
  23. this.dragStart = new EventEmitter();
  24. this.dragging = new EventEmitter();
  25. this.dragEnd = new EventEmitter();
  26. this.isDragging = false;
  27. this.element = element.nativeElement;
  28. }
  29. /**
  30. * @param {?} changes
  31. * @return {?}
  32. */
  33. ngOnChanges(changes) {
  34. if (changes['dragEventTarget'] && changes['dragEventTarget'].currentValue && this.dragModel.dragging) {
  35. this.onMousedown(changes['dragEventTarget'].currentValue);
  36. }
  37. }
  38. /**
  39. * @return {?}
  40. */
  41. ngOnDestroy() {
  42. this._destroySubscription();
  43. }
  44. /**
  45. * @param {?} event
  46. * @return {?}
  47. */
  48. onMouseup(event) {
  49. if (!this.isDragging)
  50. return;
  51. this.isDragging = false;
  52. this.element.classList.remove('dragging');
  53. if (this.subscription) {
  54. this._destroySubscription();
  55. this.dragEnd.emit({
  56. event,
  57. element: this.element,
  58. model: this.dragModel
  59. });
  60. }
  61. }
  62. /**
  63. * @param {?} event
  64. * @return {?}
  65. */
  66. onMousedown(event) {
  67. // we only want to drag the inner header text
  68. /** @type {?} */
  69. const isDragElm = ((/** @type {?} */ (event.target))).classList.contains('draggable');
  70. if (isDragElm && (this.dragX || this.dragY)) {
  71. event.preventDefault();
  72. this.isDragging = true;
  73. /** @type {?} */
  74. const mouseDownPos = { x: event.clientX, y: event.clientY };
  75. /** @type {?} */
  76. const mouseup = fromEvent(document, 'mouseup');
  77. this.subscription = mouseup.subscribe((/**
  78. * @param {?} ev
  79. * @return {?}
  80. */
  81. (ev) => this.onMouseup(ev)));
  82. /** @type {?} */
  83. const mouseMoveSub = fromEvent(document, 'mousemove')
  84. .pipe(takeUntil(mouseup))
  85. .subscribe((/**
  86. * @param {?} ev
  87. * @return {?}
  88. */
  89. (ev) => this.move(ev, mouseDownPos)));
  90. this.subscription.add(mouseMoveSub);
  91. this.dragStart.emit({
  92. event,
  93. element: this.element,
  94. model: this.dragModel
  95. });
  96. }
  97. }
  98. /**
  99. * @param {?} event
  100. * @param {?} mouseDownPos
  101. * @return {?}
  102. */
  103. move(event, mouseDownPos) {
  104. if (!this.isDragging)
  105. return;
  106. /** @type {?} */
  107. const x = event.clientX - mouseDownPos.x;
  108. /** @type {?} */
  109. const y = event.clientY - mouseDownPos.y;
  110. if (this.dragX)
  111. this.element.style.left = `${x}px`;
  112. if (this.dragY)
  113. this.element.style.top = `${y}px`;
  114. this.element.classList.add('dragging');
  115. this.dragging.emit({
  116. event,
  117. element: this.element,
  118. model: this.dragModel
  119. });
  120. }
  121. /**
  122. * @private
  123. * @return {?}
  124. */
  125. _destroySubscription() {
  126. if (this.subscription) {
  127. this.subscription.unsubscribe();
  128. this.subscription = undefined;
  129. }
  130. }
  131. }
  132. DraggableDirective.decorators = [
  133. { type: Directive, args: [{ selector: '[draggable]' },] }
  134. ];
  135. /** @nocollapse */
  136. DraggableDirective.ctorParameters = () => [
  137. { type: ElementRef }
  138. ];
  139. DraggableDirective.propDecorators = {
  140. dragEventTarget: [{ type: Input }],
  141. dragModel: [{ type: Input }],
  142. dragX: [{ type: Input }],
  143. dragY: [{ type: Input }],
  144. dragStart: [{ type: Output }],
  145. dragging: [{ type: Output }],
  146. dragEnd: [{ type: Output }]
  147. };
  148. if (false) {
  149. /** @type {?} */
  150. DraggableDirective.prototype.dragEventTarget;
  151. /** @type {?} */
  152. DraggableDirective.prototype.dragModel;
  153. /** @type {?} */
  154. DraggableDirective.prototype.dragX;
  155. /** @type {?} */
  156. DraggableDirective.prototype.dragY;
  157. /** @type {?} */
  158. DraggableDirective.prototype.dragStart;
  159. /** @type {?} */
  160. DraggableDirective.prototype.dragging;
  161. /** @type {?} */
  162. DraggableDirective.prototype.dragEnd;
  163. /** @type {?} */
  164. DraggableDirective.prototype.element;
  165. /** @type {?} */
  166. DraggableDirective.prototype.isDragging;
  167. /** @type {?} */
  168. DraggableDirective.prototype.subscription;
  169. }
  170. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"draggable.directive.js","sourceRoot":"ng://@swimlane/ngx-datatable/","sources":["lib/directives/draggable.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, Input, Output, EventEmitter, OnDestroy, OnChanges, SimpleChanges } from '@angular/core';\nimport { Observable, Subscription, fromEvent } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { MouseEvent } from '../events';\n\n/**\n * Draggable Directive for Angular2\n *\n * Inspiration:\n *   https://github.com/AngularClass/angular2-examples/blob/master/rx-draggable/directives/draggable.ts\n *   http://stackoverflow.com/questions/35662530/how-to-implement-drag-and-drop-in-angular2\n *\n */\n@Directive({ selector: '[draggable]' })\nexport class DraggableDirective implements OnDestroy, OnChanges {\n  @Input() dragEventTarget: any;\n  @Input() dragModel: any;\n  @Input() dragX: boolean = true;\n  @Input() dragY: boolean = true;\n\n  @Output() dragStart: EventEmitter<any> = new EventEmitter();\n  @Output() dragging: EventEmitter<any> = new EventEmitter();\n  @Output() dragEnd: EventEmitter<any> = new EventEmitter();\n\n  element: HTMLElement;\n  isDragging: boolean = false;\n  subscription: Subscription;\n\n  constructor(element: ElementRef) {\n    this.element = element.nativeElement;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['dragEventTarget'] && changes['dragEventTarget'].currentValue && this.dragModel.dragging) {\n      this.onMousedown(changes['dragEventTarget'].currentValue);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this._destroySubscription();\n  }\n\n  onMouseup(event: MouseEvent): void {\n    if (!this.isDragging) return;\n\n    this.isDragging = false;\n    this.element.classList.remove('dragging');\n\n    if (this.subscription) {\n      this._destroySubscription();\n      this.dragEnd.emit({\n        event,\n        element: this.element,\n        model: this.dragModel\n      });\n    }\n  }\n\n  onMousedown(event: MouseEvent): void {\n    // we only want to drag the inner header text\n    const isDragElm = (<HTMLElement>event.target).classList.contains('draggable');\n\n    if (isDragElm && (this.dragX || this.dragY)) {\n      event.preventDefault();\n      this.isDragging = true;\n\n      const mouseDownPos = { x: event.clientX, y: event.clientY };\n\n      const mouseup = fromEvent(document, 'mouseup');\n      this.subscription = mouseup.subscribe((ev: MouseEvent) => this.onMouseup(ev));\n\n      const mouseMoveSub = fromEvent(document, 'mousemove')\n        .pipe(takeUntil(mouseup))\n        .subscribe((ev: MouseEvent) => this.move(ev, mouseDownPos));\n\n      this.subscription.add(mouseMoveSub);\n\n      this.dragStart.emit({\n        event,\n        element: this.element,\n        model: this.dragModel\n      });\n    }\n  }\n\n  move(event: MouseEvent, mouseDownPos: { x: number; y: number }): void {\n    if (!this.isDragging) return;\n\n    const x = event.clientX - mouseDownPos.x;\n    const y = event.clientY - mouseDownPos.y;\n\n    if (this.dragX) this.element.style.left = `${x}px`;\n    if (this.dragY) this.element.style.top = `${y}px`;\n\n    this.element.classList.add('dragging');\n\n    this.dragging.emit({\n      event,\n      element: this.element,\n      model: this.dragModel\n    });\n  }\n\n  private _destroySubscription(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n      this.subscription = undefined;\n    }\n  }\n}\n"]}