2022-04-03 02:13:15 +02:00
|
|
|
import { clearDragSource, initDraggables } from '../draggable';
|
2022-01-31 20:28:38 +01:00
|
|
|
import { fireEvent } from '@testing-library/dom';
|
|
|
|
import { getRandomArrayItem } from '../../../test/randomness';
|
2024-05-01 00:15:06 +02:00
|
|
|
import { MockInstance } from 'vitest';
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
describe('Draggable Utilities', () => {
|
|
|
|
// jsdom lacks proper support for window.DragEvent so this is an attempt at a minimal recreation
|
|
|
|
const createDragEvent = (name: string, init?: DragEventInit): DragEvent => {
|
|
|
|
const mockEvent = new Event(name, { bubbles: true, cancelable: true }) as unknown as DragEvent;
|
|
|
|
let dataTransfer = init?.dataTransfer;
|
|
|
|
if (!dataTransfer) {
|
|
|
|
const items: Pick<DataTransferItem, 'type' | 'getAsString'>[] = [];
|
|
|
|
dataTransfer = {
|
|
|
|
items: items as unknown as DataTransferItemList,
|
|
|
|
setData(format: string, data: string) {
|
|
|
|
items.push({ type: format, getAsString: (callback: FunctionStringCallback) => callback(data) });
|
2024-07-03 22:54:14 +02:00
|
|
|
},
|
2022-01-31 20:28:38 +01:00
|
|
|
} as unknown as DataTransfer;
|
|
|
|
}
|
|
|
|
Object.assign(mockEvent, { dataTransfer });
|
|
|
|
return mockEvent;
|
|
|
|
};
|
|
|
|
|
|
|
|
const createDraggableElement = (): HTMLDivElement => {
|
|
|
|
const el = document.createElement('div');
|
|
|
|
el.setAttribute('draggable', 'true');
|
|
|
|
return el;
|
|
|
|
};
|
|
|
|
|
|
|
|
describe('initDraggables', () => {
|
|
|
|
const draggingClass = 'dragging';
|
|
|
|
const dragContainerClass = 'drag-container';
|
|
|
|
const dragOverClass = 'over';
|
2024-04-30 20:44:26 +02:00
|
|
|
let documentEventListenerSpy: MockInstance;
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
let mockDragContainer: HTMLDivElement;
|
|
|
|
let mockDraggable: HTMLDivElement;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
mockDragContainer = document.createElement('div');
|
|
|
|
mockDragContainer.classList.add(dragContainerClass);
|
|
|
|
document.body.appendChild(mockDragContainer);
|
|
|
|
|
|
|
|
mockDraggable = createDraggableElement();
|
|
|
|
mockDragContainer.appendChild(mockDraggable);
|
|
|
|
|
|
|
|
// Redirect all document event listeners to this element for easier cleanup
|
2024-04-30 20:44:26 +02:00
|
|
|
documentEventListenerSpy = vi.spyOn(document, 'addEventListener').mockImplementation((...params) => {
|
2022-01-31 20:28:38 +01:00
|
|
|
mockDragContainer.addEventListener(...params);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
document.body.removeChild(mockDragContainer);
|
|
|
|
documentEventListenerSpy.mockRestore();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('dragStart', () => {
|
|
|
|
it('should add the dragging class to the element that starts moving', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
|
2024-07-03 22:54:14 +02:00
|
|
|
it("should add dummy data to the dragstart event if it's empty", () => {
|
2022-01-31 20:28:38 +01:00
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
|
|
|
expect(mockEvent.dataTransfer?.items).toHaveLength(0);
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.dataTransfer?.items).toHaveLength(1);
|
|
|
|
|
|
|
|
const dataTransferItem = (mockEvent.dataTransfer as DataTransfer).items[0];
|
|
|
|
expect(dataTransferItem.type).toEqual('text/plain');
|
|
|
|
|
|
|
|
let stringValue: string | undefined;
|
2024-07-03 22:54:14 +02:00
|
|
|
dataTransferItem.getAsString((value) => {
|
2022-01-31 20:28:38 +01:00
|
|
|
stringValue = value;
|
|
|
|
});
|
|
|
|
expect(stringValue).toEqual('');
|
|
|
|
});
|
|
|
|
|
2024-07-03 22:54:14 +02:00
|
|
|
it("should keep data in the dragstart event if it's present", () => {
|
2022-01-31 20:28:38 +01:00
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockTransferItemType = getRandomArrayItem(['text/javascript', 'image/jpg', 'application/json']);
|
|
|
|
const mockDataTransferItem: DataTransferItem = {
|
|
|
|
type: mockTransferItemType,
|
|
|
|
} as unknown as DataTransferItem;
|
|
|
|
|
2024-07-03 22:54:14 +02:00
|
|
|
const mockEvent = createDragEvent('dragstart', {
|
|
|
|
dataTransfer: { items: [mockDataTransferItem] as unknown as DataTransferItemList },
|
|
|
|
} as DragEventInit);
|
2022-01-31 20:28:38 +01:00
|
|
|
expect(mockEvent.dataTransfer?.items).toHaveLength(1);
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.dataTransfer?.items).toHaveLength(1);
|
|
|
|
|
|
|
|
const dataTransferItem = (mockEvent.dataTransfer as DataTransfer).items[0];
|
|
|
|
expect(dataTransferItem.type).toEqual(mockTransferItemType);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set the allowed effect to move on the data transfer', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
|
|
|
expect(mockEvent.dataTransfer?.effectAllowed).toBeFalsy();
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.dataTransfer?.effectAllowed).toEqual('move');
|
|
|
|
});
|
2022-04-03 02:13:15 +02:00
|
|
|
|
|
|
|
it('should not throw if the event has no dataTransfer property', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
|
|
|
delete (mockEvent as Record<keyof typeof mockEvent, unknown>).dataTransfer;
|
|
|
|
expect(() => fireEvent(mockDraggable, mockEvent)).not.toThrow();
|
|
|
|
});
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('dragOver', () => {
|
|
|
|
it('should cancel event and set the drop effect to move on the data transfer', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragover');
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.defaultPrevented).toBe(true);
|
|
|
|
expect(mockEvent.dataTransfer?.dropEffect).toEqual('move');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('dragEnter', () => {
|
|
|
|
it('should add the over class to the target', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragenter');
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).toHaveClass(dragOverClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('dragLeave', () => {
|
|
|
|
it('should remove the over class from the target', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
mockDraggable.classList.add(dragOverClass);
|
|
|
|
const mockEvent = createDragEvent('dragleave');
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).not.toHaveClass(dragOverClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('drop', () => {
|
|
|
|
it('should cancel the event and remove dragging class if dropped on same element', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockStartEvent = createDragEvent('dragstart');
|
|
|
|
fireEvent(mockDraggable, mockStartEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
const mockDropEvent = createDragEvent('drop');
|
|
|
|
fireEvent(mockDraggable, mockDropEvent);
|
|
|
|
|
|
|
|
expect(mockDropEvent.defaultPrevented).toBe(true);
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).not.toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should cancel the event and insert source before target if dropped on left side', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockSecondDraggable = createDraggableElement();
|
|
|
|
mockDragContainer.appendChild(mockSecondDraggable);
|
|
|
|
|
|
|
|
const mockStartEvent = createDragEvent('dragstart');
|
|
|
|
fireEvent(mockSecondDraggable, mockStartEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockSecondDraggable).toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
const mockDropEvent = createDragEvent('drop');
|
|
|
|
Object.assign(mockDropEvent, { clientX: 124 });
|
2024-04-30 20:44:26 +02:00
|
|
|
const boundingBoxSpy = vi.spyOn(mockDraggable, 'getBoundingClientRect').mockReturnValue({
|
2022-01-31 20:28:38 +01:00
|
|
|
left: 100,
|
|
|
|
width: 50,
|
|
|
|
} as unknown as DOMRect);
|
|
|
|
fireEvent(mockDraggable, mockDropEvent);
|
|
|
|
|
|
|
|
try {
|
|
|
|
expect(mockDropEvent.defaultPrevented).toBe(true);
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockSecondDraggable).not.toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
expect(mockSecondDraggable.nextElementSibling).toBe(mockDraggable);
|
2024-07-03 22:54:14 +02:00
|
|
|
} finally {
|
2022-01-31 20:28:38 +01:00
|
|
|
boundingBoxSpy.mockRestore();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should cancel the event and insert source after target if dropped on right side', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockSecondDraggable = createDraggableElement();
|
|
|
|
mockDragContainer.appendChild(mockSecondDraggable);
|
|
|
|
|
|
|
|
const mockStartEvent = createDragEvent('dragstart');
|
|
|
|
fireEvent(mockSecondDraggable, mockStartEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockSecondDraggable).toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
const mockDropEvent = createDragEvent('drop');
|
|
|
|
Object.assign(mockDropEvent, { clientX: 125 });
|
2024-04-30 20:44:26 +02:00
|
|
|
const boundingBoxSpy = vi.spyOn(mockDraggable, 'getBoundingClientRect').mockReturnValue({
|
2022-01-31 20:28:38 +01:00
|
|
|
left: 100,
|
|
|
|
width: 50,
|
|
|
|
} as unknown as DOMRect);
|
|
|
|
fireEvent(mockDraggable, mockDropEvent);
|
|
|
|
|
|
|
|
try {
|
|
|
|
expect(mockDropEvent.defaultPrevented).toBe(true);
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockSecondDraggable).not.toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
expect(mockDraggable.nextElementSibling).toBe(mockSecondDraggable);
|
2024-07-03 22:54:14 +02:00
|
|
|
} finally {
|
2022-01-31 20:28:38 +01:00
|
|
|
boundingBoxSpy.mockRestore();
|
|
|
|
}
|
|
|
|
});
|
2022-04-03 02:13:15 +02:00
|
|
|
|
|
|
|
it('should not throw if drag source element is not set', () => {
|
|
|
|
clearDragSource();
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockSecondDraggable = createDraggableElement();
|
|
|
|
mockDragContainer.appendChild(mockSecondDraggable);
|
|
|
|
|
|
|
|
const mockDropEvent = createDragEvent('drop');
|
|
|
|
fireEvent(mockDraggable, mockDropEvent);
|
|
|
|
|
|
|
|
expect(() => fireEvent(mockDraggable, mockDropEvent)).not.toThrow();
|
|
|
|
expect(mockDropEvent.defaultPrevented).toBe(true);
|
|
|
|
});
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('dragEnd', () => {
|
2024-07-03 22:54:14 +02:00
|
|
|
it("should remove dragging class from source and over class from target's descendants", () => {
|
2022-01-31 20:28:38 +01:00
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockStartEvent = createDragEvent('dragstart');
|
|
|
|
fireEvent(mockDraggable, mockStartEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).toHaveClass(draggingClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
const mockOverElement = createDraggableElement();
|
|
|
|
mockOverElement.classList.add(dragOverClass);
|
|
|
|
mockDraggable.parentNode?.appendChild(mockOverElement);
|
|
|
|
const mockOverEvent = createDragEvent('dragend');
|
|
|
|
fireEvent(mockOverElement, mockOverEvent);
|
|
|
|
|
|
|
|
const mockDropEvent = createDragEvent('dragend');
|
|
|
|
fireEvent(mockDraggable, mockDropEvent);
|
|
|
|
|
2024-05-01 00:20:14 +02:00
|
|
|
expect(mockDraggable).not.toHaveClass(draggingClass);
|
|
|
|
expect(mockOverElement).not.toHaveClass(dragOverClass);
|
2022-01-31 20:28:38 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('wrapper', () => {
|
|
|
|
it('should do nothing when event target has no closest method', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
|
|
|
Object.assign(mockDraggable, { closest: undefined });
|
|
|
|
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.dataTransfer?.effectAllowed).toBeFalsy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should do nothing when event target does not have a parent matching the predefined selector', () => {
|
|
|
|
initDraggables();
|
|
|
|
|
|
|
|
const mockEvent = createDragEvent('dragstart');
|
2024-04-30 20:44:26 +02:00
|
|
|
const draggableClosestSpy = vi.spyOn(mockDraggable, 'closest').mockReturnValue(null);
|
2022-01-31 20:28:38 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
fireEvent(mockDraggable, mockEvent);
|
|
|
|
|
|
|
|
expect(mockEvent.dataTransfer?.effectAllowed).toBeFalsy();
|
2024-07-03 22:54:14 +02:00
|
|
|
} finally {
|
2022-03-25 22:34:08 +01:00
|
|
|
draggableClosestSpy.mockRestore();
|
2022-01-31 20:28:38 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|