projects/angular2-draggable/src/lib/angular-draggable.directive.ts
        
                            OnInit
                            OnDestroy
                            OnChanges
                            AfterViewInit
            
| Selector | [ngDraggable] | 
| Properties | 
| 
 | 
| Methods | 
| 
 | 
| Inputs | 
| Outputs | 
| HostListeners | 
| Accessors | 
| constructor(el: ElementRef, renderer: Renderer2) | |||||||||
| 
                                    Parameters :
                                     
 | 
| bounds | |
| Type : HTMLElement | |
| Set the bounds HTMLElement | |
| gridSize | |
| Type : number | |
| Default value : 1 | |
| Round the position to nearest grid | |
| handle | |
| Type : HTMLElement | |
| Make the handle HTMLElement draggable | |
| inBounds | |
| Type : boolean | |
| Default value : false | |
| Whether to limit the element stay in the bounds | |
| lockAxis | |
| Type : string | |
| Default value : null | |
| Lock axis: 'x' or 'y' | |
| ngDraggable | |
| Type : any | |
| outOfBounds | |
| Type : { top: boolean; right: boolean; bottom: boolean; left: boolean; } | |
| Default value : {
    top: false,
    right: false,
    bottom: false,
    left: false,
  } | |
| List of allowed out of bounds edges | |
| position | |
| Type : IPosition | |
| Default value : { x: 0, y: 0 } | |
| Set initial position by offsets | |
| preventDefaultEvent | |
| Type : boolean | |
| Default value : false | |
| Whether to prevent default event | |
| scale | |
| Type : number | |
| Default value : 1 | |
| Input css scale transform of element so translations are correct | |
| trackPosition | |
| Type : boolean | |
| Default value : true | |
| Whether the element should use it's previous drag position on a new drag event. | |
| zIndex | |
| Type : string | |
| Set z-index when not dragging | |
| zIndexMoving | |
| Type : string | |
| Set z-index when dragging | |
| edge | |
| Type : EventEmitter | |
| endOffset | |
| Type : EventEmitter | |
| Emit position offsets when put back | |
| movingOffset | |
| Type : EventEmitter | |
| Emit position offsets when moving | |
| started | |
| Type : EventEmitter | |
| stopped | |
| Type : EventEmitter | |
| mousedown | 
| Arguments : '$event'  | 
| touchstart | 
| Arguments : '$event'  | 
| boundsCheck | 
| boundsCheck() | 
| 
                        Returns :      { top: boolean; right: boolean; bottom: boolean; left: boolean; } | 
| checkHandleTarget | |||||||||
| checkHandleTarget(target: EventTarget, element: Element) | |||||||||
| 
                        Parameters :
                        
                         
 
                        Returns :          boolean | 
| getCurrentOffset | 
| getCurrentOffset() | 
| Get current offset 
                        Returns :          any | 
| Private getDragEl | 
| getDragEl() | 
| 
                        Returns :          any | 
| Private moveTo | ||||||
| moveTo(p: Position) | ||||||
| 
                        Parameters :
                        
                         
 
                        Returns :          void | 
| ngAfterViewInit | 
| ngAfterViewInit() | 
| 
                        Returns :          void | 
| ngOnChanges | ||||||
| ngOnChanges(changes: SimpleChanges) | ||||||
| 
                        Parameters :
                        
                         
 
                        Returns :          void | 
| ngOnDestroy | 
| ngOnDestroy() | 
| 
                        Returns :          void | 
| ngOnInit | 
| ngOnInit() | 
| 
                        Returns :          void | 
| onMouseDown | ||||||
| onMouseDown(event: MouseEvent | TouchEvent) | ||||||
| Decorators : @HostListener('mousedown', ['$event']) | ||||||
| 
                        Parameters :
                        
                         
 
                        Returns :          void | 
| onMouseMove | ||||||
| onMouseMove(event: MouseEvent | TouchEvent) | ||||||
| 
                        Parameters :
                        
                         
 
                        Returns :          void | 
| Private pickUp | 
| pickUp() | 
| 
                        Returns :          void | 
| Private putBack | 
| putBack() | 
| 
                        Returns :          void | 
| resetPosition | 
| resetPosition() | 
| 
                        Returns :          void | 
| Private subscribeEvents | 
| subscribeEvents() | 
| 
                        Returns :          void | 
| Private transform | 
| transform() | 
| 
                        Returns :          void | 
| Private unsubscribeEvents | 
| unsubscribeEvents() | 
| 
                        Returns :          void | 
| Private _helperBlock | 
| Type : HelperBlock | 
| Default value : null | 
| Bugfix: iFrames, and context unrelated elements block all events, and are unusable https://github.com/xieziyu/angular2-draggable/issues/84 | 
| Private _zIndex | 
| Type : string | 
| Default value : '' | 
| Private allowDrag | 
| Default value : true | 
| Private currTrans | 
| Default value : new Position(0, 0) | 
| Private draggingSub | 
| Type : Subscription | 
| Default value : null | 
| Private moving | 
| Default value : false | 
| Private needTransform | 
| Default value : false | 
| Private oldTrans | 
| Default value : new Position(0, 0) | 
| Private oldZIndex | 
| Type : string | 
| Default value : '' | 
| Private orignal | 
| Type : Position | 
| Default value : null | 
| Private tempTrans | 
| Default value : new Position(0, 0) | 
| zIndex | ||||||
| set zIndex(setting: string) | ||||||
| Set z-index when not dragging 
                                        Parameters :
                                         
 
                                    Returns :          void | 
| ngDraggable | ||||||
| set ngDraggable(setting: any) | ||||||
| 
                                        Parameters :
                                         
 
                                    Returns :          void | 
import {
  Directive,
  ElementRef,
  Renderer2,
  Input,
  Output,
  OnInit,
  HostListener,
  EventEmitter,
  OnChanges,
  SimpleChanges,
  OnDestroy,
  AfterViewInit,
} from '@angular/core';
import { Subscription, fromEvent } from 'rxjs';
import { IPosition, Position } from './models/position';
import { HelperBlock } from './widgets/helper-block';
@Directive({
  selector: '[ngDraggable]',
  exportAs: 'ngDraggable',
})
export class AngularDraggableDirective implements OnInit, OnDestroy, OnChanges, AfterViewInit {
  private allowDrag = true;
  private moving = false;
  private orignal: Position = null;
  private oldTrans = new Position(0, 0);
  private tempTrans = new Position(0, 0);
  private currTrans = new Position(0, 0);
  private oldZIndex = '';
  private _zIndex = '';
  private needTransform = false;
  private draggingSub: Subscription = null;
  /**
   * Bugfix: iFrames, and context unrelated elements block all events, and are unusable
   * https://github.com/xieziyu/angular2-draggable/issues/84
   */
  private _helperBlock: HelperBlock = null;
  @Output() started = new EventEmitter<any>();
  @Output() stopped = new EventEmitter<any>();
  @Output() edge = new EventEmitter<any>();
  /** Make the handle HTMLElement draggable */
  @Input() handle: HTMLElement;
  /** Set the bounds HTMLElement */
  @Input() bounds: HTMLElement;
  /** List of allowed out of bounds edges **/
  @Input() outOfBounds = {
    top: false,
    right: false,
    bottom: false,
    left: false,
  };
  /** Round the position to nearest grid */
  @Input() gridSize = 1;
  /** Set z-index when dragging */
  @Input() zIndexMoving: string;
  /** Set z-index when not dragging */
  @Input() set zIndex(setting: string) {
    this.renderer.setStyle(this.el.nativeElement, 'z-index', setting);
    this._zIndex = setting;
  }
  /** Whether to limit the element stay in the bounds */
  @Input() inBounds = false;
  /** Whether the element should use it's previous drag position on a new drag event. */
  @Input() trackPosition = true;
  /** Input css scale transform of element so translations are correct */
  @Input() scale = 1;
  /** Whether to prevent default event */
  @Input() preventDefaultEvent = false;
  /** Set initial position by offsets */
  @Input() position: IPosition = { x: 0, y: 0 };
  /** Lock axis: 'x' or 'y' */
  @Input() lockAxis: string = null;
  /** Emit position offsets when moving */
  @Output() movingOffset = new EventEmitter<IPosition>();
  /** Emit position offsets when put back */
  @Output() endOffset = new EventEmitter<IPosition>();
  @Input()
  set ngDraggable(setting: any) {
    if (setting !== undefined && setting !== null && setting !== '') {
      this.allowDrag = !!setting;
      let element = this.getDragEl();
      if (this.allowDrag) {
        this.renderer.addClass(element, 'ng-draggable');
      } else {
        this.putBack();
        this.renderer.removeClass(element, 'ng-draggable');
      }
    }
  }
  constructor(private el: ElementRef, private renderer: Renderer2) {
    this._helperBlock = new HelperBlock(el.nativeElement, renderer);
  }
  ngOnInit() {
    if (this.allowDrag) {
      let element = this.getDragEl();
      this.renderer.addClass(element, 'ng-draggable');
    }
    this.resetPosition();
  }
  ngOnDestroy() {
    this.bounds = null;
    this.handle = null;
    this.orignal = null;
    this.oldTrans = null;
    this.tempTrans = null;
    this.currTrans = null;
    this._helperBlock.dispose();
    this._helperBlock = null;
    if (this.draggingSub) {
      this.draggingSub.unsubscribe();
    }
  }
  ngOnChanges(changes: SimpleChanges) {
    if (changes['position'] && !changes['position'].isFirstChange()) {
      let p = changes['position'].currentValue;
      if (!this.moving) {
        if (Position.isIPosition(p)) {
          this.oldTrans.set(p);
        } else {
          this.oldTrans.reset();
        }
        this.transform();
      } else {
        this.needTransform = true;
      }
    }
  }
  ngAfterViewInit() {
    if (this.inBounds) {
      this.boundsCheck();
      this.oldTrans.add(this.tempTrans);
      this.tempTrans.reset();
    }
  }
  private getDragEl() {
    return this.handle ? this.handle : this.el.nativeElement;
  }
  resetPosition() {
    if (Position.isIPosition(this.position)) {
      this.oldTrans.set(this.position);
    } else {
      this.oldTrans.reset();
    }
    this.tempTrans.reset();
    this.transform();
  }
  private moveTo(p: Position) {
    if (this.orignal) {
      p.subtract(this.orignal);
      this.tempTrans.set(p);
      this.tempTrans.divide(this.scale);
      this.transform();
      if (this.bounds) {
        let edgeEv = this.boundsCheck();
        if (edgeEv) {
          this.edge.emit(edgeEv);
        }
      }
      this.movingOffset.emit(this.currTrans.value);
    }
  }
  private transform() {
    let translateX = this.tempTrans.x + this.oldTrans.x;
    let translateY = this.tempTrans.y + this.oldTrans.y;
    if (this.lockAxis === 'x') {
      translateX = this.oldTrans.x;
      this.tempTrans.x = 0;
    } else if (this.lockAxis === 'y') {
      translateY = this.oldTrans.y;
      this.tempTrans.y = 0;
    }
    // Snap to grid: by grid size
    if (this.gridSize > 1) {
      translateX = Math.round(translateX / this.gridSize) * this.gridSize;
      translateY = Math.round(translateY / this.gridSize) * this.gridSize;
    }
    let value = `translate(${Math.round(translateX)}px, ${Math.round(translateY)}px)`;
    this.renderer.setStyle(this.el.nativeElement, 'transform', value);
    this.renderer.setStyle(this.el.nativeElement, '-webkit-transform', value);
    this.renderer.setStyle(this.el.nativeElement, '-ms-transform', value);
    this.renderer.setStyle(this.el.nativeElement, '-moz-transform', value);
    this.renderer.setStyle(this.el.nativeElement, '-o-transform', value);
    // save current position
    this.currTrans.x = translateX;
    this.currTrans.y = translateY;
  }
  private pickUp() {
    // get old z-index:
    this.oldZIndex = this.el.nativeElement.style.zIndex ? this.el.nativeElement.style.zIndex : '';
    if (window) {
      this.oldZIndex = window
        .getComputedStyle(this.el.nativeElement, null)
        .getPropertyValue('z-index');
    }
    if (this.zIndexMoving) {
      this.renderer.setStyle(this.el.nativeElement, 'z-index', this.zIndexMoving);
    }
    if (!this.moving) {
      this.started.emit(this.el.nativeElement);
      this.moving = true;
      const element = this.getDragEl();
      this.renderer.addClass(element, 'ng-dragging');
      /**
       * Fix performance issue:
       * https://github.com/xieziyu/angular2-draggable/issues/112
       */
      this.subscribeEvents();
    }
  }
  private subscribeEvents() {
    this.draggingSub = fromEvent(document, 'mousemove', { passive: false }).subscribe(event =>
      this.onMouseMove(event as MouseEvent)
    );
    this.draggingSub.add(
      fromEvent(document, 'touchmove', { passive: false }).subscribe(event =>
        this.onMouseMove(event as TouchEvent)
      )
    );
    this.draggingSub.add(
      fromEvent(document, 'mouseup', { passive: false }).subscribe(() => this.putBack())
    );
    // checking if browser is IE or Edge - https://github.com/xieziyu/angular2-draggable/issues/153
    let isIEOrEdge = /msie\s|trident\//i.test(window.navigator.userAgent);
    if (!isIEOrEdge) {
      this.draggingSub.add(
        fromEvent(document, 'mouseleave', { passive: false }).subscribe(() => this.putBack())
      );
    }
    this.draggingSub.add(
      fromEvent(document, 'touchend', { passive: false }).subscribe(() => this.putBack())
    );
    this.draggingSub.add(
      fromEvent(document, 'touchcancel', { passive: false }).subscribe(() => this.putBack())
    );
  }
  private unsubscribeEvents() {
    this.draggingSub.unsubscribe();
    this.draggingSub = null;
  }
  boundsCheck() {
    if (this.bounds) {
      let boundary = this.bounds.getBoundingClientRect();
      let elem = this.el.nativeElement.getBoundingClientRect();
      let result = {
        top: this.outOfBounds.top ? true : boundary.top < elem.top,
        right: this.outOfBounds.right ? true : boundary.right > elem.right,
        bottom: this.outOfBounds.bottom ? true : boundary.bottom > elem.bottom,
        left: this.outOfBounds.left ? true : boundary.left < elem.left,
      };
      if (this.inBounds) {
        if (!result.top) {
          this.tempTrans.y -= (elem.top - boundary.top) / this.scale;
        }
        if (!result.bottom) {
          this.tempTrans.y -= (elem.bottom - boundary.bottom) / this.scale;
        }
        if (!result.right) {
          this.tempTrans.x -= (elem.right - boundary.right) / this.scale;
        }
        if (!result.left) {
          this.tempTrans.x -= (elem.left - boundary.left) / this.scale;
        }
        this.transform();
      }
      return result;
    }
    return null;
  }
  /** Get current offset */
  getCurrentOffset() {
    return this.currTrans.value;
  }
  private putBack() {
    if (this._zIndex) {
      this.renderer.setStyle(this.el.nativeElement, 'z-index', this._zIndex);
    } else if (this.zIndexMoving) {
      if (this.oldZIndex) {
        this.renderer.setStyle(this.el.nativeElement, 'z-index', this.oldZIndex);
      } else {
        this.el.nativeElement.style.removeProperty('z-index');
      }
    }
    if (this.moving) {
      this.stopped.emit(this.el.nativeElement);
      // Remove the helper div:
      this._helperBlock.remove();
      if (this.needTransform) {
        if (Position.isIPosition(this.position)) {
          this.oldTrans.set(this.position);
        } else {
          this.oldTrans.reset();
        }
        this.transform();
        this.needTransform = false;
      }
      if (this.bounds) {
        let edgeEv = this.boundsCheck();
        if (edgeEv) {
          this.edge.emit(edgeEv);
        }
      }
      this.moving = false;
      this.endOffset.emit(this.currTrans.value);
      if (this.trackPosition) {
        this.oldTrans.add(this.tempTrans);
      }
      this.tempTrans.reset();
      if (!this.trackPosition) {
        this.transform();
      }
      const element = this.getDragEl();
      this.renderer.removeClass(element, 'ng-dragging');
      /**
       * Fix performance issue:
       * https://github.com/xieziyu/angular2-draggable/issues/112
       */
      this.unsubscribeEvents();
    }
  }
  checkHandleTarget(target: EventTarget, element: Element) {
    // Checks if the target is the element clicked, then checks each child element of element as well
    // Ignores button clicks
    // Ignore elements of type button
    if (element.tagName === 'BUTTON') {
      return false;
    }
    // If the target was found, return true (handle was found)
    if (element === target) {
      return true;
    }
    // Recursively iterate this elements children
    for (let child in element.children) {
      if (element.children.hasOwnProperty(child)) {
        if (this.checkHandleTarget(target, element.children[child])) {
          return true;
        }
      }
    }
    // Handle was not found in this lineage
    // Note: return false is ignore unless it is the parent element
    return false;
  }
  @HostListener('mousedown', ['$event'])
  @HostListener('touchstart', ['$event'])
  onMouseDown(event: MouseEvent | TouchEvent) {
    // 1. skip right click;
    if (event instanceof MouseEvent && event.button === 2) {
      return;
    }
    // 2. if handle is set, the element can only be moved by handle
    let target = event.target || event.srcElement;
    if (this.handle !== undefined && !this.checkHandleTarget(target, this.handle)) {
      return;
    }
    // 3. if allow drag is set to false, ignore the mousedown
    if (this.allowDrag === false) {
      return;
    }
    if (this.preventDefaultEvent) {
      event.stopPropagation();
      event.preventDefault();
    }
    this.orignal = Position.fromEvent(event, this.getDragEl());
    this.pickUp();
  }
  onMouseMove(event: MouseEvent | TouchEvent) {
    if (this.moving && this.allowDrag) {
      if (this.preventDefaultEvent) {
        event.stopPropagation();
        event.preventDefault();
      }
      // Add a transparent helper div:
      this._helperBlock.add();
      this.moveTo(Position.fromEvent(event, this.getDragEl()));
    }
  }
}