mutable-div
Version:
Directives to enable moving or resizing a div element.
143 lines • 15.3 kB
JavaScript
import { Directive, EventEmitter, HostListener, Input, Output } from '@angular/core';
import * as i0 from "@angular/core";
export class DraggableElementDirective {
constructor(el) {
this.el = el;
this.pointerDown = false;
this.rotate = 0;
this.coordinates = new EventEmitter();
this.stopped = new EventEmitter();
el.nativeElement.style.position = 'absolute';
}
onPointerDown(event) {
event.preventDefault();
this.pointerDown = true;
this.setOriginalPosition(event);
}
onPointerUp() {
this.stopped.emit();
}
onPointerMove(event) {
if (this.pointerDown) {
const element = this.el.nativeElement;
const newPosition = this.setNewPosition(event);
this.setOriginalPosition(event);
element.style.left = (element.offsetLeft + newPosition.x) + 'px';
element.style.top = (element.offsetTop + newPosition.y) + 'px';
this.coordinates.emit({ x: newPosition.x, y: newPosition.y });
}
}
onWindowPointerUp() {
this.pointerDown = false;
}
onPanStart(event) {
this.onPointerDown(event.srcEvent);
}
onPanEnd() {
this.onPointerUp();
}
onPanMove(event) {
this.onPointerMove(event.srcEvent);
}
onWindowPanEnd() {
this.onWindowPointerUp();
}
setOriginalPosition(event) {
switch (this.rotate) {
case 90:
this.originalPosition = {
x: event.clientY,
y: screen.width - event.clientX
};
break;
case 180:
this.originalPosition = {
x: screen.width - event.clientX,
y: screen.height - event.clientY
};
break;
case 270:
this.originalPosition = {
x: screen.height - event.clientY,
y: event.clientX
};
break;
default:
this.originalPosition = {
x: event.clientX,
y: event.clientY
};
}
}
setNewPosition(event) {
let newPosition = {
x: 0,
y: 0
};
switch (this.rotate) {
case 90:
newPosition = {
x: event.clientY - this.originalPosition.x,
y: (screen.width - event.clientX) - this.originalPosition.y
};
break;
case 180:
newPosition = {
x: (screen.width - event.clientX) - this.originalPosition.x,
y: (screen.height - event.clientY) - this.originalPosition.y
};
break;
case 270:
newPosition = {
x: (screen.height - event.clientY) - this.originalPosition.x,
y: event.clientX - this.originalPosition.y
};
break;
default:
newPosition = {
x: event.clientX - this.originalPosition.x,
y: event.clientY - this.originalPosition.y
};
}
return newPosition;
}
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableElementDirective, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive }); }
/** @nocollapse */ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: DraggableElementDirective, selector: "[draggable]", inputs: { rotate: "rotate" }, outputs: { coordinates: "coordinates", stopped: "stopped" }, host: { listeners: { "pointerdown": "onPointerDown($event)", "pointerup": "onPointerUp()", "window:pointermove": "onPointerMove($event)", "window:pointerup": "onWindowPointerUp()", "panstart": "onPanStart($event)", "panend": "onPanEnd()", "window:panmove": "onPanMove($event)", "window:panend": "onWindowPanEnd()" } }, ngImport: i0 }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableElementDirective, decorators: [{
type: Directive,
args: [{
selector: '[draggable]'
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { rotate: [{
type: Input
}], coordinates: [{
type: Output
}], stopped: [{
type: Output
}], onPointerDown: [{
type: HostListener,
args: ['pointerdown', ['$event']]
}], onPointerUp: [{
type: HostListener,
args: ['pointerup']
}], onPointerMove: [{
type: HostListener,
args: ['window:pointermove', ['$event']]
}], onWindowPointerUp: [{
type: HostListener,
args: ['window:pointerup']
}], onPanStart: [{
type: HostListener,
args: ['panstart', ['$event']]
}], onPanEnd: [{
type: HostListener,
args: ['panend']
}], onPanMove: [{
type: HostListener,
args: ['window:panmove', ['$event']]
}], onWindowPanEnd: [{
type: HostListener,
args: ['window:panend']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"draggable-element.directive.js","sourceRoot":"","sources":["../../../../projects/mutable-div/src/lib/draggable-element.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, EventEmitter, HostListener, Input, Output } from '@angular/core';\nimport { Position } from './model/position.model';\n\n@Directive({\n  selector: '[draggable]'\n})\nexport class DraggableElementDirective {\n\n  originalPosition: Position;\n  private pointerDown: boolean = false;\n\n  @Input() rotate = 0;\n  @Output() coordinates = new EventEmitter<Position>();\n  @Output() stopped = new EventEmitter();\n\n  constructor(private el: ElementRef) {\n    el.nativeElement.style.position = 'absolute';\n  }\n\n  @HostListener('pointerdown', ['$event']) onPointerDown(event: PointerEvent) {\n    event.preventDefault();\n    this.pointerDown = true;\n    this.setOriginalPosition(event);\n  }\n\n  @HostListener('pointerup') onPointerUp() {\n    this.stopped.emit();\n  }\n\n  @HostListener('window:pointermove', ['$event']) onPointerMove(event: PointerEvent) {\n    if (this.pointerDown) {\n      const element = this.el.nativeElement;\n      const newPosition = this.setNewPosition(event);\n      this.setOriginalPosition(event);\n      element.style.left = (element.offsetLeft + newPosition.x) + 'px';\n      element.style.top = (element.offsetTop + newPosition.y) + 'px';\n      this.coordinates.emit({ x: newPosition.x, y: newPosition.y });\n    }\n  }\n\n  @HostListener('window:pointerup') onWindowPointerUp() {\n    this.pointerDown = false;\n  }\n\n  @HostListener('panstart', ['$event']) onPanStart(event: any) {\n    this.onPointerDown(event.srcEvent);\n  }\n\n  @HostListener('panend') onPanEnd() {\n      this.onPointerUp();\n  }\n\n  @HostListener('window:panmove', ['$event']) onPanMove(event: any) {\n    this.onPointerMove(event.srcEvent);\n  }\n\n  @HostListener('window:panend') onWindowPanEnd() {\n    this.onWindowPointerUp();\n  }\n\n  private setOriginalPosition(event: PointerEvent): void {\n    switch (this.rotate) {\n      case 90:\n        this.originalPosition = {\n          x: event.clientY,\n          y: screen.width - event.clientX\n        };\n        break;\n      case 180:\n        this.originalPosition = {\n          x: screen.width - event.clientX,\n          y: screen.height - event.clientY\n        };\n        break;\n      case 270:\n        this.originalPosition = {\n          x: screen.height - event.clientY,\n          y: event.clientX\n        };\n        break;\n      default:\n        this.originalPosition = {\n          x: event.clientX,\n          y: event.clientY\n        };\n    }\n  }\n\n  private setNewPosition(event: PointerEvent): Position {\n    let newPosition: Position = {\n      x: 0,\n      y: 0\n    };\n    switch (this.rotate) {\n      case 90:\n        newPosition = {\n          x: event.clientY - this.originalPosition.x,\n          y: (screen.width - event.clientX) - this.originalPosition.y\n        };\n        break;\n      case 180:\n        newPosition = {\n          x: (screen.width - event.clientX) - this.originalPosition.x,\n          y: (screen.height - event.clientY) - this.originalPosition.y\n        };\n        break;\n      case 270:\n        newPosition = {\n          x: (screen.height - event.clientY) - this.originalPosition.x,\n          y: event.clientX - this.originalPosition.y\n        };\n        break;\n      default:\n        newPosition = {\n          x: event.clientX - this.originalPosition.x,\n          y: event.clientY - this.originalPosition.y\n        };\n    }\n    return newPosition;\n  }\n}\n"]}