mirror of https://github.com/Squidex/squidex.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
209 lines
5.1 KiB
209 lines
5.1 KiB
/*
|
|
* Squidex Headless CMS
|
|
*
|
|
* @license
|
|
* Copyright (c) Squidex UG (haftungsbeschränkt). All rights reserved.
|
|
*/
|
|
|
|
// tslint:disable:prefer-for-of
|
|
|
|
import { Directive, ElementRef, EventEmitter, HostListener, Input, Output, Renderer2 } from '@angular/core';
|
|
|
|
import { Types } from '@app/framework/internal';
|
|
|
|
const ImageTypes: ReadonlyArray<string> = [
|
|
'image/jpeg',
|
|
'image/png',
|
|
'image/jpg',
|
|
'image/gif'
|
|
];
|
|
|
|
@Directive({
|
|
selector: '[sqxDropFile]'
|
|
})
|
|
export class FileDropDirective {
|
|
private dragCounter = 0;
|
|
|
|
@Input()
|
|
public allowedFiles: ReadonlyArray<string>;
|
|
|
|
@Input()
|
|
public onlyImages: boolean;
|
|
|
|
@Input()
|
|
public noDrop: boolean;
|
|
|
|
@Input('sqxDropDisabled')
|
|
public disabled = false;
|
|
|
|
@Output('sqxDropFile')
|
|
public drop = new EventEmitter<ReadonlyArray<File>>();
|
|
|
|
constructor(
|
|
private readonly element: ElementRef,
|
|
private readonly renderer: Renderer2
|
|
) {
|
|
}
|
|
|
|
@HostListener('paste', ['$event'])
|
|
public onPaste(event: ClipboardEvent) {
|
|
if (this.noDrop) {
|
|
return;
|
|
}
|
|
|
|
const files = this.getAllowedFiles(event.clipboardData);
|
|
|
|
if (files && !this.disabled) {
|
|
this.drop.emit(files);
|
|
}
|
|
|
|
this.stopEvent(event);
|
|
}
|
|
|
|
@HostListener('dragend', ['$event'])
|
|
@HostListener('dragleave', ['$event'])
|
|
public onDragEnd(event: DragDropEvent) {
|
|
const hasFile = this.hasAllowedFile(event.dataTransfer);
|
|
|
|
if (hasFile) {
|
|
this.dragEnd();
|
|
}
|
|
}
|
|
|
|
@HostListener('dragenter', ['$event'])
|
|
public onDragEnter(event: DragDropEvent) {
|
|
const hasFile = this.hasAllowedFile(event.dataTransfer);
|
|
|
|
if (hasFile) {
|
|
this.dragStart();
|
|
}
|
|
}
|
|
|
|
@HostListener('dragover', ['$event'])
|
|
public onDragOver(event: DragDropEvent) {
|
|
const isFiles = hasFiles(event.dataTransfer);
|
|
|
|
if (isFiles) {
|
|
this.stopEvent(event);
|
|
}
|
|
}
|
|
|
|
@HostListener('drop', ['$event'])
|
|
public onDrop(event: DragDropEvent) {
|
|
if (hasFiles(event.dataTransfer)) {
|
|
const files = this.getAllowedFiles(event.dataTransfer);
|
|
|
|
if (files && !this.disabled) {
|
|
this.drop.emit(files);
|
|
}
|
|
|
|
this.dragEnd(0);
|
|
this.stopEvent(event);
|
|
}
|
|
}
|
|
|
|
private stopEvent(event: Event) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
}
|
|
|
|
private dragStart() {
|
|
this.dragCounter++;
|
|
|
|
if (this.dragCounter === 1 && !this.disabled) {
|
|
this.renderer.addClass(this.element.nativeElement, 'drag');
|
|
}
|
|
}
|
|
|
|
private dragEnd(number?: number ) {
|
|
this.dragCounter = number || this.dragCounter - 1;
|
|
|
|
if (this.dragCounter === 0 && !this.disabled) {
|
|
this.renderer.removeClass(this.element.nativeElement, 'drag');
|
|
}
|
|
}
|
|
|
|
private getAllowedFiles(dataTransfer: DataTransfer | null) {
|
|
if (!dataTransfer || !hasFiles(dataTransfer)) {
|
|
return null;
|
|
}
|
|
|
|
const files: File[] = [];
|
|
|
|
for (let i = 0; i < dataTransfer.files.length; i++) {
|
|
const file = dataTransfer.files.item(i);
|
|
|
|
if (file && this.isAllowedFile(file)) {
|
|
files.push(file);
|
|
}
|
|
}
|
|
|
|
if (files.length === 0) {
|
|
for (let i = 0; i < dataTransfer.items.length; i++) {
|
|
const file = dataTransfer.items[i].getAsFile();
|
|
|
|
if (file && this.isAllowedFile(file)) {
|
|
files.push(file);
|
|
}
|
|
}
|
|
}
|
|
|
|
return files.length > 0 ? files : null;
|
|
}
|
|
|
|
private hasAllowedFile(dataTransfer: DataTransfer | null) {
|
|
if (!dataTransfer || !hasFiles(dataTransfer)) {
|
|
return null;
|
|
}
|
|
|
|
for (let i = 0; i < dataTransfer.files.length; i++) {
|
|
const file = dataTransfer.files.item(i);
|
|
|
|
if (file && this.isAllowedFile(file)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
for (let i = 0; i < dataTransfer.items.length; i++) {
|
|
const file = dataTransfer.items[i];
|
|
|
|
if (file && this.isAllowedFile(file)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private isAllowedFile(file: { type: string }) {
|
|
return this.isAllowed(file) && this.isImage(file);
|
|
}
|
|
|
|
private isAllowed(file: { type: string }) {
|
|
return !this.allowedFiles || this.allowedFiles.indexOf(file.type) >= 0;
|
|
}
|
|
|
|
private isImage(file: { type: string }) {
|
|
return !this.onlyImages || ImageTypes.indexOf(file.type) >= 0;
|
|
}
|
|
}
|
|
|
|
function hasFiles(dataTransfer: DataTransfer): boolean {
|
|
if (!dataTransfer || !dataTransfer.types) {
|
|
return false;
|
|
}
|
|
|
|
const types: any = dataTransfer.types;
|
|
|
|
if (Types.isFunction(types.indexOf)) {
|
|
return types.indexOf('Files') !== -1;
|
|
} else if (Types.isFunction(types.contains)) {
|
|
return types.contains('Files');
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
interface DragDropEvent extends MouseEvent {
|
|
readonly dataTransfer: DataTransfer;
|
|
}
|