mirror of
https://github.com/siyuan-note/siyuan.git
synced 2025-12-20 16:40:13 +01:00
❤️ 完整开源界面和内核 https://github.com/siyuan-note/siyuan/issues/5013
This commit is contained in:
parent
e650b8100c
commit
f40ed985e1
1214 changed files with 345766 additions and 9 deletions
807
app/src/protyle/util/editorCommonEvent.ts
Normal file
807
app/src/protyle/util/editorCommonEvent.ts
Normal file
|
|
@ -0,0 +1,807 @@
|
|||
import {focusBlock, focusByWbr, focusByRange} from "./selection";
|
||||
import {hasClosestBlock, hasClosestByAttribute, hasClosestByClassName} from "./hasClosest";
|
||||
import {Constants} from "../../constants";
|
||||
import {paste} from "./paste";
|
||||
import {genEmptyElement, genSBElement} from "../../block/util";
|
||||
import {transaction} from "../wysiwyg/transaction";
|
||||
import {getTopAloneElement} from "../wysiwyg/getBlock";
|
||||
import {updateListOrder} from "../wysiwyg/list";
|
||||
import {hideElements} from "../ui/hideElements";
|
||||
import {mathRender} from "../markdown/mathRender";
|
||||
import {fetchPost, fetchSyncPost} from "../../util/fetch";
|
||||
import {onGet} from "./onGet";
|
||||
import {getInstanceById} from "../../layout/util";
|
||||
import {Editor} from "../../editor";
|
||||
import {Tab} from "../../layout/Tab";
|
||||
import {blockRender} from "../markdown/blockRender";
|
||||
import {getAllModels} from "../../layout/getAll";
|
||||
import {processRender} from "./processCode";
|
||||
import {highlightRender} from "../markdown/highlightRender";
|
||||
import {updatePanelByEditor} from "../../editor/util";
|
||||
|
||||
const dragSb = (protyle: IProtyle, sourceElements: Element[], targetElement: Element, isBottom: boolean, direct: "col" | "row") => {
|
||||
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
|
||||
}];
|
||||
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"),
|
||||
previousID: sbElement.previousElementSibling?.getAttribute("data-node-id"),
|
||||
parentID: sbElement.parentElement.getAttribute("data-node-id") || protyle.block.parentID
|
||||
}];
|
||||
if (newSourceElement) {
|
||||
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: newSourceElement.getAttribute("data-node-id"),
|
||||
previousID: targetElement.getAttribute("data-node-id"),
|
||||
});
|
||||
} else {
|
||||
targetElement.insertAdjacentElement("beforebegin", newSourceElement);
|
||||
doOperations.push({
|
||||
action: "insert",
|
||||
data: newSourceElement.outerHTML,
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
previousID: newSourceElement.previousElementSibling?.getAttribute("data-node-id"),
|
||||
parentID: newSourceElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID
|
||||
});
|
||||
}
|
||||
sourceElements.reverse().forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
newSourceElement.insertAdjacentElement("afterbegin", item);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: item.getAttribute("data-node-id"),
|
||||
parentID: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
undoOperations.push({
|
||||
action: "delete",
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
} else {
|
||||
if (!isBottom) {
|
||||
sbElement.insertAdjacentElement("afterbegin", targetElement);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: targetElement.getAttribute("data-node-id"),
|
||||
parentID: sbElement.getAttribute("data-node-id")
|
||||
});
|
||||
}
|
||||
sourceElements.reverse().forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
sbElement.insertAdjacentElement("afterbegin", item);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: item.getAttribute("data-node-id"),
|
||||
parentID: sbElement.getAttribute("data-node-id"),
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
if (isBottom) {
|
||||
sbElement.insertAdjacentElement("afterbegin", targetElement);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: targetElement.getAttribute("data-node-id"),
|
||||
parentID: sbElement.getAttribute("data-node-id")
|
||||
});
|
||||
}
|
||||
}
|
||||
undoOperations.push({
|
||||
action: "delete",
|
||||
id: sbElement.getAttribute("data-node-id"),
|
||||
});
|
||||
// https://github.com/siyuan-note/insider/issues/536
|
||||
if (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;
|
||||
}
|
||||
undoOperations.push({
|
||||
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 (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
|
||||
});
|
||||
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 (oldSourceParentElement && (oldSourceParentElement.classList.contains("bq") || oldSourceParentElement.classList.contains("sb")) && oldSourceParentElement.childElementCount === 1) {
|
||||
// 拖拽后,bq 为空
|
||||
const protyleElement = hasClosestByClassName(oldSourceParentElement, "protyle", true);
|
||||
if (protyleElement) {
|
||||
const editor = getInstanceById(protyleElement.getAttribute("data-id")) as Tab;
|
||||
if (editor && editor.model instanceof Editor) {
|
||||
const newId = Lute.NewNodeID();
|
||||
const newElement = genEmptyElement(false, false, newId);
|
||||
doOperations.splice(0, 0, {
|
||||
action: "insert",
|
||||
id: newId,
|
||||
data: newElement.outerHTML,
|
||||
parentID: oldSourceParentElement.getAttribute("data-node-id")
|
||||
});
|
||||
undoOperations.splice(0, 0, {
|
||||
action: "delete",
|
||||
id: newId,
|
||||
});
|
||||
if (isSameDoc) {
|
||||
oldSourceParentElement.insertAdjacentElement("afterbegin", newElement);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (oldSourceParentElement && oldSourceParentElement.classList.contains("protyle-wysiwyg") && oldSourceParentElement.innerHTML === "") {
|
||||
// 拖拽后,根文档原内容为空,且不为悬浮窗
|
||||
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,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
if (isSameDoc) {
|
||||
transaction(protyle, doOperations, undoOperations);
|
||||
} else {
|
||||
// 跨文档不支持撤销
|
||||
transaction(protyle, doOperations);
|
||||
}
|
||||
focusBlock(sourceElements[0]);
|
||||
};
|
||||
|
||||
const dragSame = (protyle: IProtyle, sourceElements: Element[], targetElement: Element, isBottom: 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) {
|
||||
targetElement.insertAdjacentElement("afterend", newSourceElement);
|
||||
doOperations.push({
|
||||
action: "insert",
|
||||
data: newSourceElement.outerHTML,
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
previousID: targetElement.getAttribute("data-node-id"),
|
||||
});
|
||||
sourceElements.reverse().forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
newSourceElement.insertAdjacentElement("afterbegin", item);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: item.getAttribute("data-node-id"),
|
||||
parentID: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
undoOperations.push({
|
||||
action: "delete",
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
} else {
|
||||
sourceElements.reverse().forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
targetElement.insertAdjacentElement("afterend", item);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: item.getAttribute("data-node-id"),
|
||||
previousID: targetElement.getAttribute("data-node-id"),
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
}
|
||||
} else {
|
||||
if (newSourceElement) {
|
||||
targetElement.insertAdjacentElement("beforebegin", newSourceElement);
|
||||
doOperations.push({
|
||||
action: "insert",
|
||||
data: newSourceElement.outerHTML,
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
previousID: newSourceElement.previousElementSibling?.getAttribute("data-node-id"),
|
||||
parentID: newSourceElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID
|
||||
});
|
||||
sourceElements.reverse().forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
newSourceElement.insertAdjacentElement("afterbegin", item);
|
||||
doOperations.push({
|
||||
action: "move",
|
||||
id: item.getAttribute("data-node-id"),
|
||||
parentID: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
undoOperations.push({
|
||||
action: "delete",
|
||||
id: newSourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
} else {
|
||||
sourceElements.forEach((item, index) => {
|
||||
if (index === sourceElements.length - 1) {
|
||||
topSourceElement = getTopAloneElement(item);
|
||||
if (topSourceElement.isSameNode(item)) {
|
||||
topSourceElement = undefined;
|
||||
}
|
||||
}
|
||||
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) {
|
||||
// 打开两个相同的文档
|
||||
const sameElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${item.getAttribute("data-node-id")}"]`);
|
||||
if (sameElement) {
|
||||
sameElement.remove();
|
||||
}
|
||||
}
|
||||
targetElement.insertAdjacentElement("beforebegin", item);
|
||||
doOperations.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.parentID
|
||||
});
|
||||
});
|
||||
undoOperations.reverse();
|
||||
}
|
||||
}
|
||||
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 (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 (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
|
||||
});
|
||||
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 (oldSourceParentElement && (oldSourceParentElement.classList.contains("bq") || oldSourceParentElement.classList.contains("sb")) && oldSourceParentElement.childElementCount === 1) {
|
||||
// 拖拽后,bq 为空
|
||||
const protyleElement = hasClosestByClassName(oldSourceParentElement, "protyle", true);
|
||||
if (protyleElement) {
|
||||
const editor = getInstanceById(protyleElement.getAttribute("data-id")) as Tab;
|
||||
if (editor && editor.model instanceof Editor) {
|
||||
const newId = Lute.NewNodeID();
|
||||
const newElement = genEmptyElement(false, false, newId);
|
||||
doOperations.splice(0, 0, {
|
||||
action: "insert",
|
||||
id: newId,
|
||||
data: newElement.outerHTML,
|
||||
parentID: oldSourceParentElement.getAttribute("data-node-id")
|
||||
});
|
||||
undoOperations.splice(0, 0, {
|
||||
action: "delete",
|
||||
id: newId,
|
||||
});
|
||||
if (isSameDoc) {
|
||||
oldSourceParentElement.insertAdjacentElement("afterbegin", newElement);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (oldSourceParentElement && oldSourceParentElement.classList.contains("protyle-wysiwyg") && oldSourceParentElement.childElementCount === 0) {
|
||||
// 拖拽后,根文档原内容为空,且不为悬浮窗
|
||||
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,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
if (isSameDoc) {
|
||||
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")) {
|
||||
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 (event.dataTransfer.getData(Constants.SIYUAN_DROP_EDITOR)) {
|
||||
// 编辑器内选中文字拖拽
|
||||
if (getSelection().rangeCount === 0) {
|
||||
return;
|
||||
}
|
||||
const targetElement = hasClosestBlock(event.target);
|
||||
const sourceElement = hasClosestBlock(getSelection().getRangeAt(0).startContainer);
|
||||
if (!targetElement || !sourceElement) {
|
||||
return;
|
||||
}
|
||||
hideElements(["toolbar"], protyle);
|
||||
const isSameNode = targetElement.isSameNode(sourceElement);
|
||||
const undoOperations: IOperation[] = [{
|
||||
action: "update",
|
||||
data: targetElement.outerHTML,
|
||||
id: targetElement.getAttribute("data-node-id"),
|
||||
}];
|
||||
if (!isSameNode) {
|
||||
undoOperations.push({
|
||||
action: "update",
|
||||
data: sourceElement.outerHTML,
|
||||
id: sourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
}
|
||||
setTimeout(() => {
|
||||
const range = getSelection().getRangeAt(0);
|
||||
range.insertNode(document.createElement("wbr"));
|
||||
const targetHTML = protyle.lute.SpinBlockDOM(targetElement.outerHTML);
|
||||
targetElement.outerHTML = targetHTML;
|
||||
const doOperations: IOperation[] = [{
|
||||
action: "update",
|
||||
data: targetHTML,
|
||||
id: targetElement.getAttribute("data-node-id"),
|
||||
}];
|
||||
if (!isSameNode) {
|
||||
const sourceHTML = protyle.lute.SpinBlockDOM(sourceElement.outerHTML);
|
||||
sourceElement.outerHTML = sourceHTML;
|
||||
doOperations.push({
|
||||
action: "update",
|
||||
data: sourceHTML,
|
||||
id: sourceElement.getAttribute("data-node-id"),
|
||||
});
|
||||
}
|
||||
transaction(protyle, doOperations, undoOperations);
|
||||
mathRender(protyle.wysiwyg.element);
|
||||
focusByWbr(protyle.wysiwyg.element, range);
|
||||
});
|
||||
return;
|
||||
}
|
||||
const targetElement = editorElement.querySelector(".dragover__bottom") || editorElement.querySelector(".dragover__top") || editorElement.querySelector(".dragover__left") || editorElement.querySelector(".dragover__right");
|
||||
const isBacklink = window.siyuan.dragElement?.getAttribute("data-treetype") === "backlink";
|
||||
if (window.siyuan.dragElement && targetElement && (
|
||||
window.siyuan.dragElement.parentElement.classList.contains("protyle-gutters") ||
|
||||
isBacklink ||
|
||||
window.siyuan.dragElement.getAttribute("data-type") === "NodeListItem")) {
|
||||
// gutter 拖拽
|
||||
const sourceId = window.siyuan.dragElement.getAttribute("data-node-id");
|
||||
|
||||
let sourceElements: Element[] = [];
|
||||
if (isBacklink) {
|
||||
let sourceElement = document.createElement("div");
|
||||
const domHTML = await fetchSyncPost("/api/block/getBlockDOM", {id: sourceId});
|
||||
sourceElement.innerHTML = domHTML.data.dom;
|
||||
sourceElement = sourceElement.firstElementChild as HTMLDivElement;
|
||||
processRender(sourceElement);
|
||||
highlightRender(sourceElement);
|
||||
blockRender(protyle, sourceElement);
|
||||
sourceElements = [sourceElement];
|
||||
} else {
|
||||
const selectedIdsData = window.siyuan.dragElement.getAttribute("data-selected-ids");
|
||||
const selectedIds = selectedIdsData ? selectedIdsData.split(",") : [window.siyuan.dragElement.getAttribute("data-node-id")];
|
||||
selectedIds.forEach(item => {
|
||||
const selectElement = window.siyuan.dragElement.parentElement.parentElement.querySelector(`.protyle-wysiwyg [data-node-id="${item}"]`);
|
||||
if (selectElement) {
|
||||
sourceElements.push(selectElement);
|
||||
}
|
||||
});
|
||||
}
|
||||
sourceElements.forEach(item => {
|
||||
item.classList.remove("protyle-wysiwyg--select", "protyle-wysiwyg--hl");
|
||||
});
|
||||
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"));
|
||||
} else {
|
||||
dragSb(protyle, sourceElements, targetElement, targetClass.includes("dragover__bottom"), "row");
|
||||
}
|
||||
} else {
|
||||
if (targetClass.includes("dragover__left") || targetClass.includes("dragover__right")) {
|
||||
dragSb(protyle, sourceElements, targetElement, targetClass.includes("dragover__right"), "col");
|
||||
} else {
|
||||
dragSame(protyle, sourceElements, targetElement, targetClass.includes("dragover__bottom"));
|
||||
}
|
||||
}
|
||||
if (isBacklink) {
|
||||
setTimeout(() => {
|
||||
// 等待 drag transaction
|
||||
getAllModels().backlinks.forEach(item => {
|
||||
item.refresh();
|
||||
});
|
||||
}, 200);
|
||||
}
|
||||
} else if (window.siyuan.dragElement && window.siyuan.dragElement.getAttribute("data-type") === "navigation-file" && targetElement) {
|
||||
// 文件树拖拽
|
||||
fetchPost("/api/filetree/doc2Heading", {
|
||||
srcID: window.siyuan.dragElement.getAttribute("data-node-id"),
|
||||
after: targetElement.classList.contains("dragover__bottom"),
|
||||
targetID: targetElement.getAttribute("data-node-id"),
|
||||
}, response => {
|
||||
const scrollTop = protyle.contentElement.scrollTop;
|
||||
fetchPost("/api/filetree/removeDoc", {
|
||||
notebook: response.data.srcTreeBox,
|
||||
path: response.data.srcTreePath,
|
||||
});
|
||||
fetchPost("/api/filetree/getDoc", {
|
||||
id: protyle.block.id,
|
||||
size: Constants.SIZE_GET,
|
||||
}, getResponse => {
|
||||
onGet(getResponse, protyle);
|
||||
// 文档标题互转后,需更新大纲
|
||||
updatePanelByEditor(protyle, false, false, true);
|
||||
// 文档标题互转后,编辑区会跳转到开头 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));
|
||||
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 }) => {
|
||||
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) {
|
||||
return;
|
||||
}
|
||||
// 编辑器内文字拖拽或资源文件拖拽进入编辑器时不能运行 event.preventDefault(), 否则吴光标; 需放在 !window.siyuan.dragElement 之后
|
||||
event.preventDefault();
|
||||
const targetElement = hasClosestBlock(event.target) as Element;
|
||||
if (!targetElement) {
|
||||
return;
|
||||
}
|
||||
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" || window.siyuan.dragElement.getAttribute("data-type") === "navigation-file") {
|
||||
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 (window.siyuan.dragElement.getAttribute("data-type") === "navigation-file" && window.siyuan.dragElement.getAttribute("data-node-id") !== protyle.block.rootID) {
|
||||
dragoverElement = targetElement;
|
||||
return;
|
||||
}
|
||||
|
||||
if (window.siyuan.dragElement.parentElement.classList.contains("protyle-gutters") ||
|
||||
// 列表项之前的点
|
||||
window.siyuan.dragElement.getAttribute("data-type") === "NodeListItem" ||
|
||||
// 反向链接
|
||||
window.siyuan.dragElement.getAttribute("data-treetype") === "backlink") {
|
||||
// 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.classList.remove("dragover__top", "dragover__bottom", "dragover__left", "dragover__right");
|
||||
}
|
||||
});
|
||||
};
|
||||
Loading…
Add table
Add a link
Reference in a new issue