2023-04-28 21:20:56 +02:00
|
|
|
class DragDropManager {
|
2024-02-16 22:49:06 +01:00
|
|
|
dragContainer;
|
|
|
|
wrapper;
|
|
|
|
pageDirection;
|
|
|
|
movePageTo;
|
|
|
|
pageDragging;
|
|
|
|
draggelEl;
|
|
|
|
draggedImageEl;
|
|
|
|
hoveredEl;
|
|
|
|
endInsertionElement;
|
|
|
|
|
|
|
|
constructor(id, wrapperId) {
|
|
|
|
this.dragContainer = document.getElementById(id);
|
2024-03-21 21:58:01 +01:00
|
|
|
this.pageDirection = document.documentElement.getAttribute("dir");
|
2024-02-16 22:49:06 +01:00
|
|
|
this.wrapper = document.getElementById(wrapperId);
|
|
|
|
this.pageDragging = false;
|
|
|
|
this.hoveredEl = undefined;
|
|
|
|
this.draggelEl = undefined;
|
|
|
|
this.draggedImageEl = undefined;
|
|
|
|
|
|
|
|
var styleElement = document.createElement("link");
|
|
|
|
styleElement.rel = "stylesheet";
|
|
|
|
styleElement.href = "css/dragdrop.css";
|
|
|
|
|
|
|
|
document.head.appendChild(styleElement);
|
|
|
|
|
|
|
|
const div = document.createElement("div");
|
|
|
|
div.classList.add("drag-manager_endpoint");
|
|
|
|
div.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="bi bi-file-earmark-arrow-down" viewBox="0 0 16 16">
|
2023-04-30 13:38:30 +02:00
|
|
|
<path d="M8.5 6.5a.5.5 0 0 0-1 0v3.793L6.354 9.146a.5.5 0 1 0-.708.708l2 2a.5.5 0 0 0 .708 0l2-2a.5.5 0 0 0-.708-.708L8.5 10.293V6.5z"/>
|
|
|
|
<path d="M14 14V4.5L9.5 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2zM9.5 3A1.5 1.5 0 0 0 11 4.5h2V14a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1h5.5v2z"/>
|
2024-02-16 22:49:06 +01:00
|
|
|
</svg>`;
|
|
|
|
this.endInsertionElement = div;
|
|
|
|
|
|
|
|
this.startDraggingPage = this.startDraggingPage.bind(this);
|
|
|
|
this.onDragEl = this.onDragEl.bind(this);
|
|
|
|
this.stopDraggingPage = this.stopDraggingPage.bind(this);
|
|
|
|
|
|
|
|
this.adapt(div);
|
|
|
|
}
|
|
|
|
|
|
|
|
startDraggingPage(div) {
|
|
|
|
this.pageDragging = true;
|
|
|
|
this.draggedEl = div;
|
|
|
|
const img = div.querySelector("img");
|
|
|
|
div.classList.add("drag-manager_dragging");
|
|
|
|
const imageSrc = img.src;
|
|
|
|
|
|
|
|
const imgEl = document.createElement("img");
|
|
|
|
imgEl.classList.add("dragged-img");
|
|
|
|
imgEl.src = imageSrc;
|
|
|
|
this.draggedImageEl = imgEl;
|
|
|
|
imgEl.style.visibility = "hidden";
|
|
|
|
imgEl.style.transform = `rotate(${img.style.rotate === "" ? "0deg" : img.style.rotate}) translate(-50%, -50%)`;
|
|
|
|
this.dragContainer.appendChild(imgEl);
|
|
|
|
|
|
|
|
window.addEventListener("mouseup", this.stopDraggingPage);
|
|
|
|
window.addEventListener("mousemove", this.onDragEl);
|
|
|
|
this.wrapper.classList.add("drag-manager_dragging-container");
|
|
|
|
this.wrapper.appendChild(this.endInsertionElement);
|
|
|
|
}
|
|
|
|
|
|
|
|
onDragEl(mouseEvent) {
|
|
|
|
const { clientX, clientY } = mouseEvent;
|
|
|
|
if (this.draggedImageEl) {
|
|
|
|
this.draggedImageEl.style.visibility = "visible";
|
|
|
|
this.draggedImageEl.style.left = `${clientX}px`;
|
|
|
|
this.draggedImageEl.style.top = `${clientY}px`;
|
2023-04-28 21:20:56 +02:00
|
|
|
}
|
2024-02-16 22:49:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
stopDraggingPage() {
|
|
|
|
window.removeEventListener("mousemove", this.onDragEl);
|
|
|
|
this.wrapper.classList.remove("drag-manager_dragging-container");
|
|
|
|
this.wrapper.removeChild(this.endInsertionElement);
|
|
|
|
window.removeEventListener("mouseup", this.stopDraggingPage);
|
|
|
|
this.draggedImageEl = undefined;
|
|
|
|
this.pageDragging = false;
|
|
|
|
this.draggedEl.classList.remove("drag-manager_dragging");
|
|
|
|
this.hoveredEl?.classList.remove("drag-manager_draghover");
|
|
|
|
this.dragContainer.childNodes.forEach((dragChild) => {
|
|
|
|
this.dragContainer.removeChild(dragChild);
|
|
|
|
});
|
|
|
|
if (!this.hoveredEl) {
|
|
|
|
return;
|
2023-04-29 12:43:12 +02:00
|
|
|
}
|
2024-02-16 22:49:06 +01:00
|
|
|
if (this.hoveredEl === this.endInsertionElement) {
|
|
|
|
this.movePageTo(this.draggedEl);
|
|
|
|
return;
|
2023-04-28 21:20:56 +02:00
|
|
|
}
|
2024-02-16 22:49:06 +01:00
|
|
|
this.movePageTo(this.draggedEl, this.hoveredEl);
|
|
|
|
}
|
|
|
|
|
|
|
|
setActions({ movePageTo }) {
|
|
|
|
this.movePageTo = movePageTo;
|
|
|
|
}
|
|
|
|
|
|
|
|
adapt(div) {
|
2024-09-08 00:34:36 +03:00
|
|
|
const onDragStart = (e) => {
|
|
|
|
e.preventDefault();
|
2024-02-16 22:49:06 +01:00
|
|
|
this.startDraggingPage(div);
|
|
|
|
};
|
|
|
|
|
|
|
|
const onMouseEnter = () => {
|
|
|
|
if (this.pageDragging) {
|
|
|
|
this.hoveredEl = div;
|
|
|
|
div.classList.add("drag-manager_draghover");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const onMouseLeave = () => {
|
|
|
|
this.hoveredEl = undefined;
|
|
|
|
div.classList.remove("drag-manager_draghover");
|
|
|
|
};
|
|
|
|
|
|
|
|
div.addEventListener("dragstart", onDragStart);
|
|
|
|
div.addEventListener("mouseenter", onMouseEnter);
|
|
|
|
div.addEventListener("mouseleave", onMouseLeave);
|
|
|
|
|
|
|
|
return div;
|
|
|
|
}
|
2023-04-28 21:20:56 +02:00
|
|
|
}
|
|
|
|
|
2023-12-25 15:34:16 -05:00
|
|
|
export default DragDropManager;
|