201 lines
25 KiB
JavaScript
201 lines
25 KiB
JavaScript
import { Directive, ElementRef, EventEmitter, forwardRef, HostBinding, HostListener, inject, Input, NgZone, Output, Renderer2, } from '@angular/core';
|
|
import { dndState, endDrag, startDrag } from './dnd-state';
|
|
import { calculateDragImageOffset, setDragData, setDragImage, } from './dnd-utils';
|
|
import * as i0 from "@angular/core";
|
|
class DndDragImageRefDirective {
|
|
dndDraggableDirective = inject(forwardRef(() => DndDraggableDirective));
|
|
elementRef = inject(ElementRef);
|
|
ngOnInit() {
|
|
this.dndDraggableDirective.registerDragImage(this.elementRef);
|
|
}
|
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: DndDragImageRefDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.1", type: DndDragImageRefDirective, isStandalone: true, selector: "[dndDragImageRef]", ngImport: i0 });
|
|
}
|
|
export { DndDragImageRefDirective };
|
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: DndDragImageRefDirective, decorators: [{
|
|
type: Directive,
|
|
args: [{ selector: '[dndDragImageRef]', standalone: true }]
|
|
}] });
|
|
class DndDraggableDirective {
|
|
dndDraggable;
|
|
dndEffectAllowed = 'copy';
|
|
dndType;
|
|
dndDraggingClass = 'dndDragging';
|
|
dndDraggingSourceClass = 'dndDraggingSource';
|
|
dndDraggableDisabledClass = 'dndDraggableDisabled';
|
|
dndDragImageOffsetFunction = calculateDragImageOffset;
|
|
dndStart = new EventEmitter();
|
|
dndDrag = new EventEmitter();
|
|
dndEnd = new EventEmitter();
|
|
dndMoved = new EventEmitter();
|
|
dndCopied = new EventEmitter();
|
|
dndLinked = new EventEmitter();
|
|
dndCanceled = new EventEmitter();
|
|
draggable = true;
|
|
dndHandle;
|
|
dndDragImageElementRef;
|
|
dragImage;
|
|
isDragStarted = false;
|
|
elementRef = inject(ElementRef);
|
|
renderer = inject(Renderer2);
|
|
ngZone = inject(NgZone);
|
|
set dndDisableIf(value) {
|
|
this.draggable = !value;
|
|
if (this.draggable) {
|
|
this.renderer.removeClass(this.elementRef.nativeElement, this.dndDraggableDisabledClass);
|
|
}
|
|
else {
|
|
this.renderer.addClass(this.elementRef.nativeElement, this.dndDraggableDisabledClass);
|
|
}
|
|
}
|
|
set dndDisableDragIf(value) {
|
|
this.dndDisableIf = value;
|
|
}
|
|
ngAfterViewInit() {
|
|
this.ngZone.runOutsideAngular(() => {
|
|
this.elementRef.nativeElement.addEventListener('drag', this.dragEventHandler);
|
|
});
|
|
}
|
|
ngOnDestroy() {
|
|
this.elementRef.nativeElement.removeEventListener('drag', this.dragEventHandler);
|
|
if (this.isDragStarted) {
|
|
endDrag();
|
|
}
|
|
}
|
|
onDragStart(event) {
|
|
if (!this.draggable) {
|
|
return false;
|
|
}
|
|
// check if there is dnd handle and if the dnd handle was used to start the drag
|
|
if (this.dndHandle != null && event._dndUsingHandle == null) {
|
|
event.stopPropagation();
|
|
return false;
|
|
}
|
|
// initialize global state
|
|
startDrag(event, this.dndEffectAllowed, this.dndType);
|
|
this.isDragStarted = true;
|
|
setDragData(event, { data: this.dndDraggable, type: this.dndType }, dndState.effectAllowed);
|
|
this.dragImage = this.determineDragImage();
|
|
// set dragging css class prior to setDragImage so styles are applied before
|
|
// TODO breaking change: add class to elementRef rather than drag image which could be another element
|
|
this.renderer.addClass(this.dragImage, this.dndDraggingClass);
|
|
// set custom dragimage if present
|
|
// set dragimage if drag is started from dndHandle
|
|
if (this.dndDragImageElementRef != null || event._dndUsingHandle != null) {
|
|
setDragImage(event, this.dragImage, this.dndDragImageOffsetFunction);
|
|
}
|
|
// add dragging source css class on first drag event
|
|
const unregister = this.renderer.listen(this.elementRef.nativeElement, 'drag', () => {
|
|
this.renderer.addClass(this.elementRef.nativeElement, this.dndDraggingSourceClass);
|
|
unregister();
|
|
});
|
|
this.dndStart.emit(event);
|
|
event.stopPropagation();
|
|
setTimeout(() => {
|
|
this.renderer.setStyle(this.dragImage, 'pointer-events', 'none');
|
|
}, 100);
|
|
return true;
|
|
}
|
|
onDrag(event) {
|
|
this.dndDrag.emit(event);
|
|
}
|
|
onDragEnd(event) {
|
|
if (!this.draggable || !this.isDragStarted) {
|
|
return;
|
|
}
|
|
// get drop effect from custom stored state as its not reliable across browsers
|
|
const dropEffect = dndState.dropEffect;
|
|
this.renderer.setStyle(this.dragImage, 'pointer-events', 'unset');
|
|
let dropEffectEmitter;
|
|
switch (dropEffect) {
|
|
case 'copy':
|
|
dropEffectEmitter = this.dndCopied;
|
|
break;
|
|
case 'link':
|
|
dropEffectEmitter = this.dndLinked;
|
|
break;
|
|
case 'move':
|
|
dropEffectEmitter = this.dndMoved;
|
|
break;
|
|
default:
|
|
dropEffectEmitter = this.dndCanceled;
|
|
break;
|
|
}
|
|
dropEffectEmitter.emit(event);
|
|
this.dndEnd.emit(event);
|
|
// reset global state
|
|
endDrag();
|
|
this.isDragStarted = false;
|
|
this.renderer.removeClass(this.dragImage, this.dndDraggingClass);
|
|
// IE9 special hammering
|
|
window.setTimeout(() => {
|
|
this.renderer.removeClass(this.elementRef.nativeElement, this.dndDraggingSourceClass);
|
|
}, 0);
|
|
event.stopPropagation();
|
|
}
|
|
registerDragHandle(handle) {
|
|
this.dndHandle = handle;
|
|
}
|
|
registerDragImage(elementRef) {
|
|
this.dndDragImageElementRef = elementRef;
|
|
}
|
|
dragEventHandler = (event) => this.onDrag(event);
|
|
determineDragImage() {
|
|
// evaluate custom drag image existence
|
|
if (typeof this.dndDragImageElementRef !== 'undefined') {
|
|
return this.dndDragImageElementRef.nativeElement;
|
|
}
|
|
else {
|
|
return this.elementRef.nativeElement;
|
|
}
|
|
}
|
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: DndDraggableDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.1", type: DndDraggableDirective, isStandalone: true, selector: "[dndDraggable]", inputs: { dndDraggable: "dndDraggable", dndEffectAllowed: "dndEffectAllowed", dndType: "dndType", dndDraggingClass: "dndDraggingClass", dndDraggingSourceClass: "dndDraggingSourceClass", dndDraggableDisabledClass: "dndDraggableDisabledClass", dndDragImageOffsetFunction: "dndDragImageOffsetFunction", dndDisableIf: "dndDisableIf", dndDisableDragIf: "dndDisableDragIf" }, outputs: { dndStart: "dndStart", dndDrag: "dndDrag", dndEnd: "dndEnd", dndMoved: "dndMoved", dndCopied: "dndCopied", dndLinked: "dndLinked", dndCanceled: "dndCanceled" }, host: { listeners: { "dragstart": "onDragStart($event)", "dragend": "onDragEnd($event)" }, properties: { "attr.draggable": "this.draggable" } }, ngImport: i0 });
|
|
}
|
|
export { DndDraggableDirective };
|
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: DndDraggableDirective, decorators: [{
|
|
type: Directive,
|
|
args: [{ selector: '[dndDraggable]', standalone: true }]
|
|
}], propDecorators: { dndDraggable: [{
|
|
type: Input
|
|
}], dndEffectAllowed: [{
|
|
type: Input
|
|
}], dndType: [{
|
|
type: Input
|
|
}], dndDraggingClass: [{
|
|
type: Input
|
|
}], dndDraggingSourceClass: [{
|
|
type: Input
|
|
}], dndDraggableDisabledClass: [{
|
|
type: Input
|
|
}], dndDragImageOffsetFunction: [{
|
|
type: Input
|
|
}], dndStart: [{
|
|
type: Output
|
|
}], dndDrag: [{
|
|
type: Output
|
|
}], dndEnd: [{
|
|
type: Output
|
|
}], dndMoved: [{
|
|
type: Output
|
|
}], dndCopied: [{
|
|
type: Output
|
|
}], dndLinked: [{
|
|
type: Output
|
|
}], dndCanceled: [{
|
|
type: Output
|
|
}], draggable: [{
|
|
type: HostBinding,
|
|
args: ['attr.draggable']
|
|
}], dndDisableIf: [{
|
|
type: Input
|
|
}], dndDisableDragIf: [{
|
|
type: Input
|
|
}], onDragStart: [{
|
|
type: HostListener,
|
|
args: ['dragstart', ['$event']]
|
|
}], onDragEnd: [{
|
|
type: HostListener,
|
|
args: ['dragend', ['$event']]
|
|
}] } });
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dnd-draggable.directive.js","sourceRoot":"","sources":["../../../../projects/dnd/src/lib/dnd-draggable.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  HostListener,\n  inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2,\n} from '@angular/core';\nimport { DndHandleDirective } from './dnd-handle.directive';\nimport { dndState, endDrag, startDrag } from './dnd-state';\nimport { EffectAllowed } from './dnd-types';\nimport {\n  calculateDragImageOffset,\n  DndDragImageOffsetFunction,\n  DndEvent,\n  setDragData,\n  setDragImage,\n} from './dnd-utils';\n\n@Directive({ selector: '[dndDragImageRef]', standalone: true })\nexport class DndDragImageRefDirective implements OnInit {\n  dndDraggableDirective = inject(forwardRef(() => DndDraggableDirective));\n  elementRef: ElementRef<HTMLElement> = inject(ElementRef);\n\n  ngOnInit() {\n    this.dndDraggableDirective.registerDragImage(this.elementRef);\n  }\n}\n\n@Directive({ selector: '[dndDraggable]', standalone: true })\nexport class DndDraggableDirective implements AfterViewInit, OnDestroy {\n  @Input() dndDraggable: any;\n  @Input() dndEffectAllowed: EffectAllowed = 'copy';\n  @Input() dndType?: string;\n  @Input() dndDraggingClass = 'dndDragging';\n  @Input() dndDraggingSourceClass = 'dndDraggingSource';\n  @Input() dndDraggableDisabledClass = 'dndDraggableDisabled';\n  @Input() dndDragImageOffsetFunction: DndDragImageOffsetFunction =\n    calculateDragImageOffset;\n\n  @Output() readonly dndStart: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndDrag: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndEnd: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndMoved: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndCopied: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndLinked: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n  @Output() readonly dndCanceled: EventEmitter<DragEvent> =\n    new EventEmitter<DragEvent>();\n\n  @HostBinding('attr.draggable') draggable = true;\n\n  private dndHandle?: DndHandleDirective;\n  private dndDragImageElementRef?: ElementRef;\n  private dragImage: Element | undefined;\n  private isDragStarted: boolean = false;\n\n  private elementRef: ElementRef<HTMLElement> = inject(ElementRef);\n  private renderer = inject(Renderer2);\n  private ngZone = inject(NgZone);\n\n  @Input() set dndDisableIf(value: boolean) {\n    this.draggable = !value;\n\n    if (this.draggable) {\n      this.renderer.removeClass(\n        this.elementRef.nativeElement,\n        this.dndDraggableDisabledClass\n      );\n    } else {\n      this.renderer.addClass(\n        this.elementRef.nativeElement,\n        this.dndDraggableDisabledClass\n      );\n    }\n  }\n\n  @Input() set dndDisableDragIf(value: boolean) {\n    this.dndDisableIf = value;\n  }\n\n  ngAfterViewInit(): void {\n    this.ngZone.runOutsideAngular(() => {\n      this.elementRef.nativeElement.addEventListener(\n        'drag',\n        this.dragEventHandler\n      );\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.elementRef.nativeElement.removeEventListener(\n      'drag',\n      this.dragEventHandler\n    );\n    if (this.isDragStarted) {\n      endDrag();\n    }\n  }\n\n  @HostListener('dragstart', ['$event']) onDragStart(event: DndEvent): boolean {\n    if (!this.draggable) {\n      return false;\n    }\n\n    // check if there is dnd handle and if the dnd handle was used to start the drag\n    if (this.dndHandle != null && event._dndUsingHandle == null) {\n      event.stopPropagation();\n      return false;\n    }\n\n    // initialize global state\n    startDrag(event, this.dndEffectAllowed, this.dndType);\n\n    this.isDragStarted = true;\n\n    setDragData(\n      event,\n      { data: this.dndDraggable, type: this.dndType },\n      dndState.effectAllowed!\n    );\n\n    this.dragImage = this.determineDragImage();\n\n    // set dragging css class prior to setDragImage so styles are applied before\n    // TODO breaking change: add class to elementRef rather than drag image which could be another element\n    this.renderer.addClass(this.dragImage, this.dndDraggingClass);\n\n    // set custom dragimage if present\n    // set dragimage if drag is started from dndHandle\n    if (this.dndDragImageElementRef != null || event._dndUsingHandle != null) {\n      setDragImage(event, this.dragImage, this.dndDragImageOffsetFunction);\n    }\n\n    // add dragging source css class on first drag event\n    const unregister = this.renderer.listen(\n      this.elementRef.nativeElement,\n      'drag',\n      () => {\n        this.renderer.addClass(\n          this.elementRef.nativeElement,\n          this.dndDraggingSourceClass\n        );\n        unregister();\n      }\n    );\n\n    this.dndStart.emit(event);\n\n    event.stopPropagation();\n\n    setTimeout(() => {\n      this.renderer.setStyle(this.dragImage, 'pointer-events', 'none');\n    }, 100);\n\n    return true;\n  }\n\n  onDrag(event: DragEvent) {\n    this.dndDrag.emit(event);\n  }\n\n  @HostListener('dragend', ['$event']) onDragEnd(event: DragEvent) {\n    if (!this.draggable || !this.isDragStarted) {\n      return;\n    }\n    // get drop effect from custom stored state as its not reliable across browsers\n    const dropEffect = dndState.dropEffect;\n\n    this.renderer.setStyle(this.dragImage, 'pointer-events', 'unset');\n\n    let dropEffectEmitter: EventEmitter<DragEvent>;\n\n    switch (dropEffect) {\n      case 'copy':\n        dropEffectEmitter = this.dndCopied;\n        break;\n\n      case 'link':\n        dropEffectEmitter = this.dndLinked;\n        break;\n\n      case 'move':\n        dropEffectEmitter = this.dndMoved;\n        break;\n\n      default:\n        dropEffectEmitter = this.dndCanceled;\n        break;\n    }\n\n    dropEffectEmitter.emit(event);\n    this.dndEnd.emit(event);\n\n    // reset global state\n    endDrag();\n\n    this.isDragStarted = false;\n\n    this.renderer.removeClass(this.dragImage, this.dndDraggingClass);\n\n    // IE9 special hammering\n    window.setTimeout(() => {\n      this.renderer.removeClass(\n        this.elementRef.nativeElement,\n        this.dndDraggingSourceClass\n      );\n    }, 0);\n\n    event.stopPropagation();\n  }\n\n  registerDragHandle(handle: DndHandleDirective | undefined) {\n    this.dndHandle = handle;\n  }\n\n  registerDragImage(elementRef: ElementRef | undefined) {\n    this.dndDragImageElementRef = elementRef;\n  }\n\n  private readonly dragEventHandler: (event: DragEvent) => void = (\n    event: DragEvent\n  ) => this.onDrag(event);\n\n  private determineDragImage(): Element {\n    // evaluate custom drag image existence\n    if (typeof this.dndDragImageElementRef !== 'undefined') {\n      return this.dndDragImageElementRef.nativeElement as Element;\n    } else {\n      return this.elementRef.nativeElement;\n    }\n  }\n}\n"]}
|