mirror of
https://github.com/siyuan-note/siyuan.git
synced 2025-12-20 08:30:12 +01:00
951 lines
44 KiB
TypeScript
951 lines
44 KiB
TypeScript
import {focusBlock, focusByRange} from "./selection";
|
||
import {hasClosestBlock, hasClosestByAttribute, hasClosestByClassName} from "./hasClosest";
|
||
import {Constants} from "../../constants";
|
||
import {paste} from "./paste";
|
||
import {cancelSB, genEmptyElement, genSBElement} from "../../block/util";
|
||
import {transaction} from "../wysiwyg/transaction";
|
||
import {getTopAloneElement} from "../wysiwyg/getBlock";
|
||
import {updateListOrder} from "../wysiwyg/list";
|
||
import {fetchPost, fetchSyncPost} from "../../util/fetch";
|
||
import {onGet} from "./onGet";
|
||
/// #if !MOBILE
|
||
import {getInstanceById} from "../../layout/util";
|
||
import {Tab} from "../../layout/Tab";
|
||
import {updatePanelByEditor} from "../../editor/util";
|
||
/// #endif
|
||
import {Editor} from "../../editor";
|
||
import {blockRender} from "../markdown/blockRender";
|
||
import {uploadLocalFiles} from "../upload";
|
||
import {insertHTML} from "./insertHTML";
|
||
import {isBrowser} from "../../util/functions";
|
||
import {hideElements} from "../ui/hideElements";
|
||
|
||
const moveToNew = (protyle: IProtyle, sourceElements: Element[], targetElement: Element, newSourceElement: Element,
|
||
isSameDoc: boolean, isBottom: boolean, isCopy: boolean) => {
|
||
let topSourceElement;
|
||
const targetId = targetElement.getAttribute("data-node-id");
|
||
const newSourceId = newSourceElement.getAttribute("data-node-id");
|
||
const doOperations: IOperation[] = [];
|
||
const undoOperations: IOperation[] = [];
|
||
targetElement.insertAdjacentElement(isBottom ? "afterend" : "beforebegin", newSourceElement);
|
||
if (isBottom) {
|
||
doOperations.push({
|
||
action: "insert",
|
||
data: newSourceElement.outerHTML,
|
||
id: newSourceId,
|
||
previousID: targetId,
|
||
});
|
||
} else {
|
||
doOperations.push({
|
||
action: "insert",
|
||
data: newSourceElement.outerHTML,
|
||
id: newSourceId,
|
||
nextID: targetId,
|
||
});
|
||
}
|
||
sourceElements.reverse().forEach((item, index) => {
|
||
const itemId = item.getAttribute("data-node-id");
|
||
if (index === sourceElements.length - 1) {
|
||
topSourceElement = getTopAloneElement(item);
|
||
if (topSourceElement.isSameNode(item)) {
|
||
topSourceElement = undefined;
|
||
}
|
||
}
|
||
const copyId = Lute.NewNodeID();
|
||
if (isCopy) {
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: copyId,
|
||
});
|
||
} else {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id: itemId,
|
||
previousID: item.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: item.parentElement.getAttribute("data-node-id") || protyle.block.rootID,
|
||
});
|
||
}
|
||
if (!isSameDoc && !isCopy) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${itemId}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
if (isCopy) {
|
||
const copyElement = item.cloneNode(true) as HTMLElement;
|
||
copyElement.setAttribute("data-node-id", copyId);
|
||
copyElement.querySelectorAll("[data-node-id]").forEach((e) => {
|
||
const newId = Lute.NewNodeID();
|
||
e.setAttribute("data-node-id", newId);
|
||
e.setAttribute("updated", newId.split("-")[0]);
|
||
});
|
||
newSourceElement.insertAdjacentElement("afterbegin", copyElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
id: copyId,
|
||
data: copyElement.outerHTML,
|
||
parentID: newSourceId,
|
||
});
|
||
} else {
|
||
newSourceElement.insertAdjacentElement("afterbegin", item);
|
||
doOperations.push({
|
||
action: "move",
|
||
id: itemId,
|
||
parentID: newSourceId,
|
||
});
|
||
}
|
||
});
|
||
undoOperations.reverse();
|
||
if (newSourceElement.getAttribute("data-subtype") === "o") {
|
||
undoOperations.splice(0, 0, {
|
||
action: "update",
|
||
id: newSourceId,
|
||
data: newSourceElement.outerHTML
|
||
});
|
||
updateListOrder(newSourceElement, 1);
|
||
doOperations.push({
|
||
action: "update",
|
||
id: newSourceId,
|
||
data: newSourceElement.outerHTML
|
||
});
|
||
}
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: newSourceId,
|
||
});
|
||
return {
|
||
doOperations,
|
||
undoOperations,
|
||
topSourceElement,
|
||
};
|
||
};
|
||
|
||
const moveTo = async (protyle: IProtyle, sourceElements: Element[], targetElement: Element,
|
||
isSameDoc: boolean, position: InsertPosition, isCopy: boolean) => {
|
||
let topSourceElement;
|
||
const doOperations: IOperation[] = [];
|
||
const undoOperations: IOperation[] = [];
|
||
const foldHeadingIds: { id: string, parentID: string }[] = [];
|
||
const targetId = targetElement.getAttribute("data-node-id");
|
||
let tempTargetElement = targetElement;
|
||
sourceElements.reverse().forEach((item, index) => {
|
||
const id = item.getAttribute("data-node-id");
|
||
const parentID = item.parentElement.getAttribute("data-node-id") || protyle.block.rootID;
|
||
if (index === sourceElements.length - 1) {
|
||
topSourceElement = getTopAloneElement(item);
|
||
if (topSourceElement.isSameNode(item)) {
|
||
topSourceElement = undefined;
|
||
} else if (topSourceElement.contains(item) && topSourceElement.contains(targetElement)) {
|
||
// * * 1 列表项拖拽到父级列表项下 https://ld246.com/article/1665448570858
|
||
topSourceElement = targetElement;
|
||
}
|
||
}
|
||
if (item.getAttribute("data-type") === "NodeHeading" && item.getAttribute("fold") === "1") {
|
||
item.removeAttribute("fold");
|
||
if (isCopy) {
|
||
foldHeadingIds.push({id, parentID});
|
||
}
|
||
}
|
||
let copyId;
|
||
let copyElement;
|
||
if (isCopy) {
|
||
copyId = Lute.NewNodeID();
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: copyId,
|
||
});
|
||
} else {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id,
|
||
previousID: item.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID,
|
||
});
|
||
}
|
||
if (!isSameDoc && !isCopy) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${id}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
|
||
if (isCopy) {
|
||
copyElement = item.cloneNode(true) as HTMLElement;
|
||
copyElement.setAttribute("data-node-id", copyId);
|
||
copyElement.querySelectorAll("[data-node-id]").forEach((e) => {
|
||
const newId = Lute.NewNodeID();
|
||
e.setAttribute("data-node-id", newId);
|
||
e.setAttribute("updated", newId.split("-")[0]);
|
||
});
|
||
tempTargetElement.insertAdjacentElement(position, copyElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
id: copyId,
|
||
data: copyElement.outerHTML,
|
||
previousID: position === "afterend" ? targetId : item.previousElementSibling?.getAttribute("data-node-id"), // 不能使用常量,移动后会被修改
|
||
parentID: item.parentElement?.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID,
|
||
});
|
||
} else {
|
||
tempTargetElement.insertAdjacentElement(position, item);
|
||
doOperations.push({
|
||
action: "move",
|
||
id,
|
||
previousID: position === "afterend" ? targetId : item.previousElementSibling?.getAttribute("data-node-id"), // 不能使用常量,移动后会被修改
|
||
parentID: item.parentElement?.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID,
|
||
});
|
||
}
|
||
if (position !== "afterend") {
|
||
tempTargetElement = isCopy ? copyElement : item;
|
||
}
|
||
});
|
||
undoOperations.reverse();
|
||
for (let j = 0; j < foldHeadingIds.length; j++) {
|
||
const childrenItem = foldHeadingIds[j];
|
||
const headingIds = await fetchSyncPost("/api/block/getHeadingChildrenIDs", {id: childrenItem.id});
|
||
headingIds.data.reverse().forEach((headingId: string) => {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id: headingId,
|
||
previousID: childrenItem.id,
|
||
parentID: childrenItem.parentID,
|
||
});
|
||
});
|
||
undoOperations.push({
|
||
action: "foldHeading",
|
||
id: childrenItem.id,
|
||
data: "remove"
|
||
});
|
||
doOperations.push({
|
||
action: "unfoldHeading",
|
||
id: childrenItem.id,
|
||
});
|
||
}
|
||
return {
|
||
doOperations,
|
||
undoOperations,
|
||
topSourceElement,
|
||
};
|
||
};
|
||
|
||
const dragSb = async (protyle: IProtyle, sourceElements: Element[], targetElement: Element, isBottom: boolean,
|
||
direct: "col" | "row", isCopy: boolean) => {
|
||
const isSameDoc = protyle.element.contains(sourceElements[0]);
|
||
|
||
let newSourceElement: HTMLElement;
|
||
if (sourceElements[0].getAttribute("data-type") === "NodeListItem" && targetElement.getAttribute("data-type") !== "NodeListItem") {
|
||
newSourceElement = document.createElement("div");
|
||
newSourceElement.setAttribute("data-node-id", Lute.NewNodeID());
|
||
newSourceElement.setAttribute("data-type", "NodeList");
|
||
newSourceElement.setAttribute("data-subtype", sourceElements[0].getAttribute("data-subtype"));
|
||
newSourceElement.className = "list";
|
||
newSourceElement.insertAdjacentHTML("beforeend", `<div class="protyle-attr" contenteditable="false">${Constants.ZWSP}</div>`);
|
||
}
|
||
|
||
const undoOperations: IOperation[] = [{
|
||
action: "move",
|
||
id: targetElement.getAttribute("data-node-id"),
|
||
previousID: targetElement.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: targetElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID
|
||
}];
|
||
let topSourceElement: Element;
|
||
let oldSourceParentElement = sourceElements[0].parentElement;
|
||
const sbElement = genSBElement(direct);
|
||
targetElement.parentElement.replaceChild(sbElement, targetElement);
|
||
const doOperations: IOperation[] = [{
|
||
action: "insert",
|
||
data: sbElement.outerHTML,
|
||
id: sbElement.getAttribute("data-node-id"),
|
||
nextID: sbElement.nextElementSibling?.getAttribute("data-node-id"),
|
||
previousID: sbElement.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: sbElement.parentElement.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID
|
||
}];
|
||
if (newSourceElement) {
|
||
const newSourceId = newSourceElement.getAttribute("data-node-id");
|
||
sbElement.insertAdjacentElement("afterbegin", targetElement);
|
||
doOperations.push({
|
||
action: "move",
|
||
id: targetElement.getAttribute("data-node-id"),
|
||
parentID: sbElement.getAttribute("data-node-id")
|
||
});
|
||
if (isBottom) {
|
||
targetElement.insertAdjacentElement("afterend", newSourceElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
data: newSourceElement.outerHTML,
|
||
id: newSourceId,
|
||
previousID: targetElement.getAttribute("data-node-id"),
|
||
});
|
||
} else {
|
||
targetElement.insertAdjacentElement("beforebegin", newSourceElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
data: newSourceElement.outerHTML,
|
||
id: newSourceId,
|
||
nextID: targetElement.getAttribute("data-node-id"),
|
||
});
|
||
}
|
||
sourceElements.reverse().forEach((item, index) => {
|
||
if (index === sourceElements.length - 1) {
|
||
topSourceElement = getTopAloneElement(item);
|
||
if (topSourceElement.isSameNode(item)) {
|
||
topSourceElement = undefined;
|
||
}
|
||
}
|
||
const copyId = Lute.NewNodeID();
|
||
if (isCopy) {
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: copyId
|
||
});
|
||
} else {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id: item.getAttribute("data-node-id"),
|
||
previousID: item.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: item.parentElement.getAttribute("data-node-id") || protyle.block.rootID,
|
||
});
|
||
}
|
||
if (!isSameDoc && !isCopy) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
if (isCopy) {
|
||
const copyElement = item.cloneNode(true) as HTMLElement;
|
||
copyElement.setAttribute("data-node-id", copyId);
|
||
copyElement.querySelectorAll("[data-node-id]").forEach((e) => {
|
||
const newId = Lute.NewNodeID();
|
||
e.setAttribute("data-node-id", newId);
|
||
e.setAttribute("updated", newId.split("-")[0]);
|
||
});
|
||
newSourceElement.insertAdjacentElement("afterbegin", copyElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
id: copyId,
|
||
data: copyElement.outerHTML,
|
||
parentID: newSourceId,
|
||
});
|
||
} else {
|
||
newSourceElement.insertAdjacentElement("afterbegin", item);
|
||
doOperations.push({
|
||
action: "move",
|
||
id: item.getAttribute("data-node-id"),
|
||
parentID: newSourceId,
|
||
});
|
||
}
|
||
});
|
||
undoOperations.reverse();
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: newSourceId,
|
||
});
|
||
} else {
|
||
const foldHeadingIds: { id: string, parentID: string }[] = [];
|
||
let afterPreviousID;
|
||
sourceElements.reverse().forEach((item, index) => {
|
||
const id = item.getAttribute("data-node-id");
|
||
const parentID = item.parentElement.getAttribute("data-node-id") || protyle.block.rootID;
|
||
if (index === sourceElements.length - 1) {
|
||
topSourceElement = getTopAloneElement(item);
|
||
if (topSourceElement.isSameNode(item)) {
|
||
topSourceElement = undefined;
|
||
}
|
||
}
|
||
const copyId = Lute.NewNodeID();
|
||
if (index === 0) {
|
||
afterPreviousID = isCopy ? copyId : id;
|
||
}
|
||
if (item.getAttribute("data-type") === "NodeHeading" && item.getAttribute("fold") === "1") {
|
||
item.removeAttribute("fold");
|
||
if (isCopy) {
|
||
foldHeadingIds.push({id, parentID});
|
||
}
|
||
}
|
||
if (isCopy) {
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: copyId,
|
||
});
|
||
} else {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id,
|
||
previousID: item.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID
|
||
});
|
||
}
|
||
if (!isSameDoc && !isCopy) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${id}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
if (isCopy) {
|
||
const copyElement = item.cloneNode(true) as HTMLElement;
|
||
copyElement.setAttribute("data-node-id", copyId);
|
||
copyElement.querySelectorAll("[data-node-id]").forEach((e) => {
|
||
const newId = Lute.NewNodeID();
|
||
e.setAttribute("data-node-id", newId);
|
||
e.setAttribute("updated", newId.split("-")[0]);
|
||
});
|
||
sbElement.insertAdjacentElement("afterbegin", copyElement);
|
||
doOperations.push({
|
||
action: "insert",
|
||
id: copyId,
|
||
data: copyElement.outerHTML,
|
||
parentID: sbElement.getAttribute("data-node-id"),
|
||
});
|
||
} else {
|
||
sbElement.insertAdjacentElement("afterbegin", item);
|
||
doOperations.push({
|
||
action: "move",
|
||
id,
|
||
parentID: sbElement.getAttribute("data-node-id"),
|
||
});
|
||
}
|
||
});
|
||
undoOperations.reverse();
|
||
for (let j = 0; j < foldHeadingIds.length; j++) {
|
||
const childrenItem = foldHeadingIds[j];
|
||
const headingIds = await fetchSyncPost("/api/block/getHeadingChildrenIDs", {id: childrenItem.id});
|
||
headingIds.data.reverse().forEach((headingId: string) => {
|
||
undoOperations.push({
|
||
action: "move",
|
||
id: headingId,
|
||
previousID: childrenItem.id,
|
||
parentID: childrenItem.parentID,
|
||
});
|
||
});
|
||
if (j === 0) {
|
||
afterPreviousID = headingIds.data[0];
|
||
}
|
||
undoOperations.push({
|
||
action: "foldHeading",
|
||
id: childrenItem.id,
|
||
data: "remove"
|
||
});
|
||
doOperations.push({
|
||
action: "unfoldHeading",
|
||
id: childrenItem.id,
|
||
});
|
||
}
|
||
if (isBottom) {
|
||
sbElement.insertAdjacentElement("afterbegin", targetElement);
|
||
doOperations.push({
|
||
action: "move",
|
||
id: targetElement.getAttribute("data-node-id"),
|
||
parentID: sbElement.getAttribute("data-node-id")
|
||
});
|
||
} else {
|
||
sbElement.lastElementChild.insertAdjacentElement("beforebegin", targetElement);
|
||
doOperations.push({
|
||
action: "move",
|
||
id: targetElement.getAttribute("data-node-id"),
|
||
previousID: afterPreviousID
|
||
});
|
||
}
|
||
}
|
||
undoOperations.push({
|
||
action: "delete",
|
||
id: sbElement.getAttribute("data-node-id"),
|
||
});
|
||
// https://github.com/siyuan-note/insider/issues/536
|
||
if (!isCopy && oldSourceParentElement && oldSourceParentElement.classList.contains("list") &&
|
||
oldSourceParentElement.getAttribute("data-subtype") === "o" &&
|
||
!oldSourceParentElement.isSameNode(sourceElements[0].parentElement) && oldSourceParentElement.childElementCount > 1) {
|
||
Array.from(oldSourceParentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
// 撤销更新不能位于最后,否则又更新为最新结果 https://github.com/siyuan-note/siyuan/issues/5725
|
||
undoOperations.splice(0, 0, {
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
});
|
||
updateListOrder(oldSourceParentElement, 1);
|
||
Array.from(oldSourceParentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
doOperations.push({
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
});
|
||
}
|
||
// 删除空元素
|
||
if (!isCopy && topSourceElement) {
|
||
doOperations.push({
|
||
action: "delete",
|
||
id: topSourceElement.getAttribute("data-node-id"),
|
||
});
|
||
undoOperations.splice(0, 0, {
|
||
action: "insert",
|
||
data: topSourceElement.outerHTML,
|
||
id: topSourceElement.getAttribute("data-node-id"),
|
||
previousID: topSourceElement.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: topSourceElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID
|
||
});
|
||
if (!isSameDoc) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${topSourceElement.getAttribute("data-node-id")}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
oldSourceParentElement = topSourceElement.parentElement;
|
||
topSourceElement.remove();
|
||
}
|
||
if (!isCopy && oldSourceParentElement && oldSourceParentElement.classList.contains("sb") && oldSourceParentElement.childElementCount === 2) {
|
||
// 拖拽后,sb 只剩下一个元素
|
||
const sbData = cancelSB(protyle, oldSourceParentElement);
|
||
doOperations.push(sbData.doOperations[0], sbData.doOperations[1]);
|
||
undoOperations.splice(0, 0, sbData.undoOperations[0], sbData.undoOperations[1]);
|
||
} else if (!isCopy && oldSourceParentElement && oldSourceParentElement.classList.contains("protyle-wysiwyg") && oldSourceParentElement.innerHTML === "") {
|
||
/// #if !MOBILE
|
||
// 拖拽后,根文档原内容为空,且不为悬浮窗
|
||
const protyleElement = hasClosestByClassName(oldSourceParentElement, "protyle", true);
|
||
if (protyleElement && !protyleElement.classList.contains("block__edit")) {
|
||
const editor = getInstanceById(protyleElement.getAttribute("data-id")) as Tab;
|
||
if (editor && editor.model instanceof Editor && editor.model.editor.protyle.block.id === editor.model.editor.protyle.block.rootID) {
|
||
const newId = Lute.NewNodeID();
|
||
doOperations.splice(0, 0, {
|
||
action: "insert",
|
||
id: newId,
|
||
data: genEmptyElement(false, false, newId).outerHTML,
|
||
parentID: editor.model.editor.protyle.block.parentID
|
||
});
|
||
undoOperations.splice(0, 0, {
|
||
action: "delete",
|
||
id: newId,
|
||
});
|
||
}
|
||
}
|
||
/// #endif
|
||
}
|
||
if (isSameDoc || isCopy) {
|
||
transaction(protyle, doOperations, undoOperations);
|
||
} else {
|
||
// 跨文档不支持撤销
|
||
transaction(protyle, doOperations);
|
||
}
|
||
focusBlock(sourceElements[0]);
|
||
};
|
||
|
||
const dragSame = async (protyle: IProtyle, sourceElements: Element[], targetElement: Element, isBottom: boolean, isCopy: boolean) => {
|
||
const isSameDoc = protyle.element.contains(sourceElements[0]);
|
||
const doOperations: IOperation[] = [];
|
||
const undoOperations: IOperation[] = [];
|
||
|
||
let newSourceElement: HTMLElement;
|
||
if (sourceElements[0].getAttribute("data-type") === "NodeListItem" && targetElement.getAttribute("data-type") !== "NodeListItem") {
|
||
newSourceElement = document.createElement("div");
|
||
newSourceElement.setAttribute("data-node-id", Lute.NewNodeID());
|
||
newSourceElement.setAttribute("data-type", "NodeList");
|
||
newSourceElement.setAttribute("data-subtype", sourceElements[0].getAttribute("data-subtype"));
|
||
newSourceElement.className = "list";
|
||
newSourceElement.insertAdjacentHTML("beforeend", `<div class="protyle-attr" contenteditable="false">${Constants.ZWSP}</div>`);
|
||
}
|
||
let topSourceElement: Element;
|
||
let oldSourceParentElement = sourceElements[0].parentElement;
|
||
if (isBottom) {
|
||
if (newSourceElement) {
|
||
const moveToResult = moveToNew(protyle, sourceElements, targetElement, newSourceElement, isSameDoc, isBottom, isCopy);
|
||
doOperations.push(...moveToResult.doOperations);
|
||
undoOperations.push(...moveToResult.undoOperations);
|
||
topSourceElement = moveToResult.topSourceElement;
|
||
} else {
|
||
const moveToResult = await moveTo(protyle, sourceElements, targetElement, isSameDoc, "afterend", isCopy);
|
||
doOperations.push(...moveToResult.doOperations);
|
||
undoOperations.push(...moveToResult.undoOperations);
|
||
topSourceElement = moveToResult.topSourceElement;
|
||
}
|
||
} else {
|
||
if (newSourceElement) {
|
||
const moveToResult = moveToNew(protyle, sourceElements, targetElement, newSourceElement, isSameDoc, isBottom, isCopy);
|
||
doOperations.push(...moveToResult.doOperations);
|
||
undoOperations.push(...moveToResult.undoOperations);
|
||
topSourceElement = moveToResult.topSourceElement;
|
||
} else {
|
||
const moveToResult = await moveTo(protyle, sourceElements, targetElement, isSameDoc, "beforebegin", isCopy);
|
||
doOperations.push(...moveToResult.doOperations);
|
||
undoOperations.push(...moveToResult.undoOperations);
|
||
topSourceElement = moveToResult.topSourceElement;
|
||
}
|
||
}
|
||
if (targetElement.getAttribute("data-type") === "NodeListItem" && targetElement.getAttribute("data-subtype") === "o") {
|
||
// https://github.com/siyuan-note/insider/issues/536
|
||
Array.from(targetElement.parentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
undoOperations.splice(0, 0, {
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
});
|
||
updateListOrder(targetElement.parentElement, 1);
|
||
Array.from(targetElement.parentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
doOperations.push({
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
});
|
||
}
|
||
if (!isCopy &&
|
||
oldSourceParentElement && oldSourceParentElement.classList.contains("list") &&
|
||
oldSourceParentElement.getAttribute("data-subtype") === "o" &&
|
||
!oldSourceParentElement.isSameNode(sourceElements[0].parentElement) && oldSourceParentElement.childElementCount > 1) {
|
||
Array.from(oldSourceParentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
if (oldSourceParentElement.contains(targetElement)) {
|
||
undoOperations.splice(0, 0, {
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
} else {
|
||
undoOperations.splice(targetElement.parentElement.childElementCount - 1, 0, {
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
}
|
||
});
|
||
updateListOrder(oldSourceParentElement, 1);
|
||
Array.from(oldSourceParentElement.children).forEach((item) => {
|
||
if (item.classList.contains("protyle-attr")) {
|
||
return;
|
||
}
|
||
doOperations.push({
|
||
action: "update",
|
||
id: item.getAttribute("data-node-id"),
|
||
data: item.outerHTML
|
||
});
|
||
});
|
||
}
|
||
|
||
// 删除空元素
|
||
if (!isCopy && topSourceElement) {
|
||
doOperations.push({
|
||
action: "delete",
|
||
id: topSourceElement.getAttribute("data-node-id"),
|
||
});
|
||
undoOperations.splice(0, 0, {
|
||
action: "insert",
|
||
data: topSourceElement.outerHTML,
|
||
id: topSourceElement.getAttribute("data-node-id"),
|
||
previousID: topSourceElement.previousElementSibling?.getAttribute("data-node-id"),
|
||
parentID: topSourceElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID || protyle.block.rootID
|
||
});
|
||
oldSourceParentElement = topSourceElement.parentElement;
|
||
topSourceElement.remove();
|
||
if (!isSameDoc) {
|
||
// 打开两个相同的文档
|
||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${topSourceElement.getAttribute("data-node-id")}"]`);
|
||
if (sameElement) {
|
||
sameElement.remove();
|
||
}
|
||
}
|
||
}
|
||
if (!isCopy && oldSourceParentElement && oldSourceParentElement.classList.contains("sb") && oldSourceParentElement.childElementCount === 2) {
|
||
// 拖拽后,sb 只剩下一个元素
|
||
const sbData = cancelSB(protyle, oldSourceParentElement);
|
||
doOperations.push(sbData.doOperations[0], sbData.doOperations[1]);
|
||
undoOperations.splice(0, 0, sbData.undoOperations[0], sbData.undoOperations[1]);
|
||
} else if (!isCopy && oldSourceParentElement && oldSourceParentElement.classList.contains("protyle-wysiwyg") && oldSourceParentElement.childElementCount === 0) {
|
||
/// #if !MOBILE
|
||
// 拖拽后,根文档原内容为空,且不为悬浮窗
|
||
const protyleElement = hasClosestByClassName(oldSourceParentElement, "protyle", true);
|
||
if (protyleElement && !protyleElement.classList.contains("block__edit")) {
|
||
const editor = getInstanceById(protyleElement.getAttribute("data-id")) as Tab;
|
||
if (editor && editor.model instanceof Editor && editor.model.editor.protyle.block.id === editor.model.editor.protyle.block.rootID) {
|
||
const newId = Lute.NewNodeID();
|
||
doOperations.splice(0, 0, {
|
||
action: "insert",
|
||
id: newId,
|
||
data: genEmptyElement(false, false, newId).outerHTML,
|
||
parentID: editor.model.editor.protyle.block.parentID
|
||
});
|
||
undoOperations.splice(0, 0, {
|
||
action: "delete",
|
||
id: newId,
|
||
});
|
||
}
|
||
}
|
||
/// #endif
|
||
}
|
||
if (isSameDoc || isCopy) {
|
||
transaction(protyle, doOperations, undoOperations);
|
||
} else {
|
||
// 跨文档不支持撤销
|
||
transaction(protyle, doOperations);
|
||
}
|
||
focusBlock(sourceElements[0]);
|
||
};
|
||
|
||
export const dropEvent = (protyle: IProtyle, editorElement: HTMLElement) => {
|
||
editorElement.addEventListener("dragstart", (event) => {
|
||
const target = event.target as HTMLElement;
|
||
if (target.tagName === "IMG") {
|
||
window.siyuan.dragElement = undefined;
|
||
event.preventDefault();
|
||
return;
|
||
}
|
||
if (target.classList && target.classList.contains("protyle-action")) {
|
||
if (hasClosestByClassName(target, "protyle-wysiwyg__embed")) {
|
||
window.siyuan.dragElement = undefined;
|
||
event.preventDefault();
|
||
} else {
|
||
window.siyuan.dragElement = target.parentElement;
|
||
}
|
||
return;
|
||
}
|
||
// 选中编辑器中的文字进行拖拽
|
||
event.dataTransfer.setData(Constants.SIYUAN_DROP_EDITOR, Constants.SIYUAN_DROP_EDITOR);
|
||
protyle.element.style.userSelect = "auto";
|
||
document.onmousemove = null;
|
||
document.onmouseup = null;
|
||
});
|
||
editorElement.addEventListener("drop", async (event: DragEvent & { target: HTMLElement }) => {
|
||
if (protyle.disabled || event.dataTransfer.getData(Constants.SIYUAN_DROP_EDITOR)) {
|
||
// 只读模式/编辑器内选中文字拖拽
|
||
event.preventDefault();
|
||
event.stopPropagation();
|
||
return;
|
||
}
|
||
const targetElement = editorElement.querySelector(".dragover__bottom") || editorElement.querySelector(".dragover__top") || editorElement.querySelector(".dragover__left") || editorElement.querySelector(".dragover__right");
|
||
if (window.siyuan.dragElement && (
|
||
window.siyuan.dragElement.parentElement?.classList.contains("protyle-gutters") ||
|
||
window.siyuan.dragElement.getAttribute("data-type") === "NodeListItem")) {
|
||
// gutter 或反链面板拖拽
|
||
const sourceElements: Element[] = [];
|
||
const selectedIdsData = window.siyuan.dragElement.getAttribute("data-selected-ids");
|
||
const selectedIds = selectedIdsData ? selectedIdsData.split(",") : [window.siyuan.dragElement.getAttribute("data-node-id")];
|
||
selectedIds.forEach(item => {
|
||
window.siyuan.dragElement.parentElement.parentElement.querySelectorAll(`.protyle-wysiwyg [data-node-id="${item}"]`).forEach(elementItem => {
|
||
if (elementItem.getAttribute("data-type") === "NodeBlockQueryEmbed" ||
|
||
!hasClosestByAttribute(elementItem, "data-type", "NodeBlockQueryEmbed")) {
|
||
sourceElements.push(elementItem);
|
||
}
|
||
});
|
||
});
|
||
sourceElements.forEach(item => {
|
||
item.classList.remove("protyle-wysiwyg--select", "protyle-wysiwyg--hl");
|
||
item.removeAttribute("select-start");
|
||
item.removeAttribute("select-end");
|
||
// 反链提及有高亮,如果拖拽到正文的话,应移除
|
||
item.querySelectorAll('[data-type="search-mark"]').forEach(markItem => {
|
||
markItem.outerHTML = markItem.innerHTML;
|
||
});
|
||
});
|
||
if (event.altKey) {
|
||
focusByRange(document.caretRangeFromPoint(event.clientX, event.clientY));
|
||
let html = "";
|
||
for (let i = 0; i < selectedIds.length; i++) {
|
||
const response = await fetchSyncPost("/api/block/getRefText", {id: selectedIds[i]});
|
||
html += `((${selectedIds[i]} '${response.data}')) `;
|
||
}
|
||
insertHTML(html, protyle);
|
||
} else if (event.shiftKey) {
|
||
focusByRange(document.caretRangeFromPoint(event.clientX, event.clientY));
|
||
let html = "";
|
||
selectedIds.forEach(item => {
|
||
html += `{{select * from blocks where id='${item}'}}\n`;
|
||
});
|
||
insertHTML(protyle.lute.SpinBlockDOM(html), protyle, true);
|
||
blockRender(protyle, protyle.wysiwyg.element);
|
||
} else if (targetElement) {
|
||
hideElements(["gutter"], protyle);
|
||
const targetClass = targetElement.className.split(" ");
|
||
targetElement.classList.remove("dragover__bottom", "dragover__top", "dragover__left", "dragover__right", "protyle-wysiwyg--select");
|
||
if (targetElement.parentElement.getAttribute("data-type") === "NodeSuperBlock" &&
|
||
targetElement.parentElement.getAttribute("data-sb-layout") === "col") {
|
||
if (targetClass.includes("dragover__left") || targetClass.includes("dragover__right")) {
|
||
dragSame(protyle, sourceElements, targetElement, targetClass.includes("dragover__right"), event.ctrlKey);
|
||
} else {
|
||
dragSb(protyle, sourceElements, targetElement, targetClass.includes("dragover__bottom"), "row", event.ctrlKey);
|
||
}
|
||
} else {
|
||
if (targetClass.includes("dragover__left") || targetClass.includes("dragover__right")) {
|
||
dragSb(protyle, sourceElements, targetElement, targetClass.includes("dragover__right"), "col", event.ctrlKey);
|
||
} else {
|
||
dragSame(protyle, sourceElements, targetElement, targetClass.includes("dragover__bottom"), event.ctrlKey);
|
||
}
|
||
}
|
||
}
|
||
} else if (event.dataTransfer.getData(Constants.SIYUAN_DROP_FILE)?.split("-").length > 1
|
||
&& targetElement && !protyle.options.backlinkData) {
|
||
// 文件树拖拽
|
||
const scrollTop = protyle.contentElement.scrollTop;
|
||
const ids = event.dataTransfer.getData(Constants.SIYUAN_DROP_FILE).split(",");
|
||
for (let i = 0; i < ids.length; i++) {
|
||
if (ids[i]) {
|
||
const response = await fetchSyncPost("/api/filetree/doc2Heading", {
|
||
srcID: ids[i],
|
||
after: targetElement.classList.contains("dragover__bottom"),
|
||
targetID: targetElement.getAttribute("data-node-id"),
|
||
});
|
||
fetchPost("/api/filetree/removeDoc", {
|
||
notebook: response.data.srcTreeBox,
|
||
path: response.data.srcTreePath,
|
||
});
|
||
}
|
||
}
|
||
fetchPost("/api/filetree/getDoc", {
|
||
id: protyle.block.id,
|
||
size: window.siyuan.config.editor.dynamicLoadBlocks,
|
||
}, getResponse => {
|
||
onGet(getResponse, protyle);
|
||
/// #if !MOBILE
|
||
// 文档标题互转后,需更新大纲
|
||
updatePanelByEditor(protyle, false, false, true);
|
||
/// #endif
|
||
// 文档标题互转后,编辑区会跳转到开头 https://github.com/siyuan-note/siyuan/issues/2939
|
||
setTimeout(() => {
|
||
protyle.contentElement.scrollTop = scrollTop;
|
||
protyle.scroll.lastScrollTop = scrollTop - 1;
|
||
}, Constants.TIMEOUT_BLOCKLOAD);
|
||
});
|
||
targetElement.classList.remove("dragover__bottom", "dragover__top");
|
||
} else if (!window.siyuan.dragElement && (event.dataTransfer.types[0] === "Files" || event.dataTransfer.types.includes("text/html"))) {
|
||
// 外部文件拖入编辑器中或者编辑器内选中文字拖拽
|
||
focusByRange(document.caretRangeFromPoint(event.clientX, event.clientY));
|
||
if (event.dataTransfer.types[0] === "Files" && !isBrowser()) {
|
||
const files: string[] = [];
|
||
for (let i = 0; i < event.dataTransfer.files.length; i++) {
|
||
files.push(event.dataTransfer.files[i].path);
|
||
}
|
||
uploadLocalFiles(files, protyle, !event.altKey);
|
||
} else {
|
||
paste(protyle, event);
|
||
}
|
||
}
|
||
if (window.siyuan.dragElement) {
|
||
window.siyuan.dragElement.style.opacity = "";
|
||
window.siyuan.dragElement = undefined;
|
||
}
|
||
});
|
||
let dragoverElement: Element;
|
||
editorElement.addEventListener("dragover", (event: DragEvent & { target: HTMLElement }) => {
|
||
// 设置了的话 drop 就无法监听 shift/control event.dataTransfer.dropEffect = "move";
|
||
if (event.dataTransfer.types.includes("Files") && event.target.classList.contains("protyle-wysiwyg")) {
|
||
// 文档底部拖拽文件需 preventDefault,否则无法触发 drop 事件 https://github.com/siyuan-note/siyuan/issues/2665
|
||
event.preventDefault();
|
||
return;
|
||
}
|
||
if (!window.siyuan.dragElement) {
|
||
// https://github.com/siyuan-note/siyuan/issues/6436
|
||
event.preventDefault();
|
||
return;
|
||
}
|
||
if (event.shiftKey || event.altKey) {
|
||
const targetElement = hasClosestBlock(event.target);
|
||
if (targetElement) {
|
||
targetElement.classList.remove("dragover__top", "protyle-wysiwyg--select", "dragover__bottom", "dragover__left", "dragover__right");
|
||
}
|
||
event.preventDefault();
|
||
return;
|
||
}
|
||
// 编辑器内文字拖拽或资源文件拖拽或按住 alt/shift 拖拽反链图标进入编辑器时不能运行 event.preventDefault(), 否则无光标; 需放在 !window.siyuan.dragElement 之后
|
||
event.preventDefault();
|
||
const targetElement = hasClosestBlock(event.target) as Element;
|
||
if (!targetElement) {
|
||
return;
|
||
}
|
||
const fileTreeIds = window.siyuan.dragElement.innerText;
|
||
if (targetElement && dragoverElement && targetElement.isSameNode(dragoverElement)) {
|
||
// 性能优化,目标为同一个元素不再进行校验
|
||
const nodeRect = targetElement.getBoundingClientRect();
|
||
targetElement.classList.remove("dragover__top", "dragover__bottom", "dragover__left", "dragover__right");
|
||
|
||
if (targetElement.getAttribute("data-type") === "NodeListItem" || fileTreeIds.indexOf("-") > -1) {
|
||
if (event.clientY > nodeRect.top + nodeRect.height / 2) {
|
||
targetElement.classList.add("dragover__bottom", "protyle-wysiwyg--select");
|
||
} else {
|
||
targetElement.classList.add("dragover__top", "protyle-wysiwyg--select");
|
||
}
|
||
return;
|
||
}
|
||
|
||
if (event.clientX < nodeRect.left + 32 && event.clientX > nodeRect.left) {
|
||
targetElement.classList.add("dragover__left", "protyle-wysiwyg--select");
|
||
} else if (event.clientX > nodeRect.right - 32 && event.clientX < nodeRect.right) {
|
||
targetElement.classList.add("dragover__right", "protyle-wysiwyg--select");
|
||
} else {
|
||
if (event.clientY > nodeRect.top + nodeRect.height / 2) {
|
||
targetElement.classList.add("dragover__bottom", "protyle-wysiwyg--select");
|
||
} else {
|
||
targetElement.classList.add("dragover__top", "protyle-wysiwyg--select");
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
if (fileTreeIds.indexOf("-") > -1) {
|
||
if (fileTreeIds.split(",").includes(protyle.block.rootID)) {
|
||
dragoverElement = undefined;
|
||
} else {
|
||
dragoverElement = targetElement;
|
||
}
|
||
return;
|
||
}
|
||
|
||
if (window.siyuan.dragElement.parentElement?.classList.contains("protyle-gutters") ||
|
||
// 列表项之前的点
|
||
window.siyuan.dragElement.getAttribute("data-type") === "NodeListItem") {
|
||
// gutter 文档内拖拽限制
|
||
// 排除自己及子孙
|
||
const selectedIdsData = window.siyuan.dragElement.getAttribute("data-selected-ids");
|
||
const selectedIds = selectedIdsData ? selectedIdsData.split(",") : [window.siyuan.dragElement.getAttribute("data-node-id")];
|
||
const isSelf = selectedIds.find((item: string) => {
|
||
if (item && hasClosestByAttribute(targetElement, "data-node-id", item)) {
|
||
return true;
|
||
}
|
||
});
|
||
if (isSelf) {
|
||
return;
|
||
}
|
||
if (hasClosestByAttribute(targetElement.parentElement, "data-type", "NodeBlockQueryEmbed")) {
|
||
// 不允许托入嵌入块
|
||
return;
|
||
}
|
||
if (window.siyuan.dragElement.getAttribute("data-type") === "NodeListItem" &&
|
||
window.siyuan.dragElement.getAttribute("data-subtype") !== targetElement.getAttribute("data-subtype") &&
|
||
window.siyuan.dragElement.getAttribute("data-type") === targetElement.getAttribute("data-type")) {
|
||
// 排除类型不同的列表项
|
||
return;
|
||
}
|
||
if (window.siyuan.dragElement.getAttribute("data-type") !== "NodeListItem" && targetElement.getAttribute("data-type") === "NodeListItem") {
|
||
// 非列表项不能拖入列表项周围
|
||
return;
|
||
}
|
||
dragoverElement = targetElement;
|
||
}
|
||
});
|
||
editorElement.addEventListener("dragleave", (event: DragEvent & { target: HTMLElement }) => {
|
||
const nodeElement = hasClosestBlock(event.target);
|
||
if (nodeElement) {
|
||
if ((window.siyuan.dragElement?.getAttribute("data-selected-ids") || "").indexOf(nodeElement.getAttribute("data-node-id")) === -1) {
|
||
nodeElement.classList.remove("protyle-wysiwyg--select");
|
||
nodeElement.removeAttribute("select-start");
|
||
nodeElement.removeAttribute("select-end");
|
||
}
|
||
nodeElement.classList.remove("dragover__top", "dragover__bottom", "dragover__left", "dragover__right");
|
||
}
|
||
});
|
||
};
|