2022-05-26 15:18:53 +08:00
|
|
|
|
import {paste} from "../util/paste";
|
|
|
|
|
|
import {
|
|
|
|
|
|
hasClosestBlock,
|
|
|
|
|
|
hasClosestByAttribute,
|
|
|
|
|
|
hasClosestByClassName,
|
|
|
|
|
|
hasClosestByMatchTag,
|
|
|
|
|
|
hasTopClosestByClassName,
|
|
|
|
|
|
} from "../util/hasClosest";
|
|
|
|
|
|
import {
|
|
|
|
|
|
focusBlock,
|
|
|
|
|
|
focusByRange,
|
|
|
|
|
|
focusByWbr,
|
|
|
|
|
|
focusSideBlock,
|
|
|
|
|
|
getEditorRange,
|
|
|
|
|
|
getSelectionOffset,
|
|
|
|
|
|
} from "../util/selection";
|
|
|
|
|
|
import {Constants} from "../../constants";
|
|
|
|
|
|
import {getSearch, isMobile} from "../../util/functions";
|
|
|
|
|
|
import {isLocalPath, pathPosix} from "../../util/pathName";
|
|
|
|
|
|
import {genEmptyElement} from "../../block/util";
|
|
|
|
|
|
import {previewImage} from "../preview/image";
|
|
|
|
|
|
import {openGlobalSearch} from "../../search/util";
|
2022-06-01 23:52:22 +08:00
|
|
|
|
import {contentMenu, imgMenu, linkMenu, refMenu, setFold, zoomOut} from "../../menus/protyle";
|
2022-05-26 15:18:53 +08:00
|
|
|
|
import * as dayjs from "dayjs";
|
|
|
|
|
|
import {dropEvent} from "../util/editorCommonEvent";
|
|
|
|
|
|
import {input} from "./input";
|
|
|
|
|
|
import {
|
|
|
|
|
|
getContenteditableElement,
|
|
|
|
|
|
getLastBlock,
|
|
|
|
|
|
getPreviousHeading,
|
|
|
|
|
|
getTopAloneElement,
|
|
|
|
|
|
hasNextSibling,
|
|
|
|
|
|
isNotEditBlock
|
|
|
|
|
|
} from "./getBlock";
|
|
|
|
|
|
import {transaction, updateTransaction} from "./transaction";
|
|
|
|
|
|
import {hideElements} from "../ui/hideElements";
|
|
|
|
|
|
/// #if !BROWSER
|
|
|
|
|
|
import {shell} from "electron";
|
|
|
|
|
|
/// #endif
|
|
|
|
|
|
import {removeEmbed} from "./removeEmbed";
|
|
|
|
|
|
import {keydown} from "./keydown";
|
|
|
|
|
|
import {openMobileFileById} from "../../mobile/editor";
|
|
|
|
|
|
import {removeBlock} from "./remove";
|
|
|
|
|
|
import {highlightRender} from "../markdown/highlightRender";
|
|
|
|
|
|
import {getAllModels} from "../../layout/getAll";
|
|
|
|
|
|
import {setPosition} from "../../util/setPosition";
|
|
|
|
|
|
import {openAttr} from "../../menus/commonMenuItem";
|
|
|
|
|
|
import {blockRender} from "../markdown/blockRender";
|
|
|
|
|
|
import {pushBack} from "../../util/backForward";
|
|
|
|
|
|
import {openAsset, openBy, openFileById} from "../../editor/util";
|
|
|
|
|
|
import {BlockPanel} from "../../block/Panel";
|
|
|
|
|
|
import {isCtrl} from "../util/compatibility";
|
|
|
|
|
|
import {MenuItem} from "../../menus/Menu";
|
|
|
|
|
|
import {fetchPost} from "../../util/fetch";
|
|
|
|
|
|
import {onGet} from "../util/onGet";
|
2022-06-12 22:06:19 +08:00
|
|
|
|
import {setTableAlign} from "../util/table";
|
2022-05-26 15:18:53 +08:00
|
|
|
|
|
|
|
|
|
|
export class WYSIWYG {
|
|
|
|
|
|
public lastHTMLs: { [key: string]: string } = {};
|
|
|
|
|
|
|
|
|
|
|
|
public element: HTMLDivElement;
|
|
|
|
|
|
public preventKeyup: boolean;
|
|
|
|
|
|
|
|
|
|
|
|
constructor(protyle: IProtyle) {
|
|
|
|
|
|
this.element = document.createElement("div");
|
|
|
|
|
|
this.element.className = "protyle-wysiwyg";
|
|
|
|
|
|
this.element.setAttribute("spellcheck", "false");
|
|
|
|
|
|
if (navigator && navigator.maxTouchPoints > 1 && navigator.platform === "MacIntel") {
|
|
|
|
|
|
// iPhone,iPad 端输入 contenteditable 为 true 时会在块中间插入 span
|
|
|
|
|
|
this.element.setAttribute("contenteditable", "false");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.element.setAttribute("contenteditable", "true");
|
|
|
|
|
|
}
|
|
|
|
|
|
if (window.siyuan.config.editor.displayBookmarkIcon) {
|
|
|
|
|
|
this.element.classList.add("protyle-wysiwyg--attr");
|
|
|
|
|
|
}
|
|
|
|
|
|
this.bindEvent(protyle);
|
|
|
|
|
|
keydown(protyle, this.element);
|
|
|
|
|
|
dropEvent(protyle, this.element);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public renderCustom(ial: IObject) {
|
|
|
|
|
|
const ialKeys = Object.keys(ial);
|
|
|
|
|
|
for (let i = 0; i < this.element.attributes.length; i++) {
|
|
|
|
|
|
const oldKey = this.element.attributes[i].nodeName;
|
|
|
|
|
|
if (!["type", "class", "spellcheck", "contenteditable", "data-doc-type", "style"].includes(oldKey) &&
|
|
|
|
|
|
!ialKeys.includes(oldKey)) {
|
|
|
|
|
|
this.element.removeAttribute(oldKey);
|
|
|
|
|
|
i--;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
ialKeys.forEach((key: string) => {
|
|
|
|
|
|
if (!["title-img", "title", "updated", "icon", "id", "type", "class", "spellcheck", "contenteditable", "data-doc-type", "style"].includes(key)) {
|
|
|
|
|
|
this.element.setAttribute(key, ial[key]);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private escapeInline(protyle: IProtyle, range: Range, event: InputEvent) {
|
|
|
|
|
|
if (!event.data) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const inputData = event.data;
|
|
|
|
|
|
protyle.toolbar.range = range;
|
|
|
|
|
|
const inlineElement = range.startContainer.parentElement;
|
|
|
|
|
|
let outElement = inlineElement;
|
|
|
|
|
|
if (!["DIV", "TD", "TH"].includes(inlineElement.tagName) && !["DIV", "TD", "TH"].includes(inlineElement.parentElement.tagName) && !hasNextSibling(inlineElement)) {
|
|
|
|
|
|
// 表格行内公式之前无法插入文字 https://github.com/siyuan-note/siyuan/issues/3908
|
|
|
|
|
|
// 引用外面加粗体且引用后无文字 https://github.com/siyuan-note/siyuan/issues/3103
|
|
|
|
|
|
outElement = inlineElement.parentElement;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (// 表格行内公式之前无法插入文字 https://github.com/siyuan-note/siyuan/issues/3908
|
|
|
|
|
|
!["DIV", "TD", "TH"].includes(inlineElement.tagName) &&
|
|
|
|
|
|
range.toString() === "" && range.startContainer.nodeType === 3 && protyle.toolbar.getCurrentType().length > 0 &&
|
|
|
|
|
|
inlineElement.textContent.replace(Constants.ZWSP, "").length >= inputData.length && // 为空的时候需要等于
|
|
|
|
|
|
!protyle.toolbar.isNewEmptyInline) {
|
|
|
|
|
|
const position = getSelectionOffset(inlineElement, protyle.wysiwyg.element, range);
|
|
|
|
|
|
let dataLength = inputData.length;
|
|
|
|
|
|
if (inputData === "<" || inputData === ">") {
|
|
|
|
|
|
// 使用 inlineElement.innerHTML 会出现 https://ld246.com/article/1627185027423 中的第2个问题
|
|
|
|
|
|
dataLength = 4;
|
|
|
|
|
|
}
|
|
|
|
|
|
// ctrl+k 会产生 ZWSP,需要移除
|
|
|
|
|
|
const html = inlineElement.innerHTML.replace(Constants.ZWSP, "");
|
|
|
|
|
|
if (position.start === inlineElement.textContent.length) {
|
|
|
|
|
|
// 使用 inlineElement.textContent **$a$b** 中数学公式消失
|
|
|
|
|
|
inlineElement.innerHTML = html.substr(0, html.length - dataLength);
|
|
|
|
|
|
const textNode = document.createTextNode(inputData);
|
|
|
|
|
|
outElement.after(textNode);
|
|
|
|
|
|
range.selectNodeContents(textNode);
|
|
|
|
|
|
range.collapse(false);
|
|
|
|
|
|
} else if (position.start === inputData.length) {
|
|
|
|
|
|
inlineElement.innerHTML = html.substr(dataLength);
|
|
|
|
|
|
const textNode = document.createTextNode(inputData);
|
|
|
|
|
|
outElement.before(textNode);
|
|
|
|
|
|
range.selectNodeContents(textNode);
|
|
|
|
|
|
range.collapse(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private setEmptyOutline(protyle: IProtyle, element: HTMLElement) {
|
|
|
|
|
|
// 图片移除选择状态应放在前面,否则 https://github.com/siyuan-note/siyuan/issues/4173
|
|
|
|
|
|
const selectImgElement = protyle.wysiwyg.element.querySelector(".img--select");
|
|
|
|
|
|
if (selectImgElement) {
|
|
|
|
|
|
selectImgElement.classList.remove("img--select");
|
|
|
|
|
|
}
|
|
|
|
|
|
let nodeElement = element;
|
|
|
|
|
|
if (!element.getAttribute("data-node-id")) {
|
|
|
|
|
|
const tempElement = hasClosestBlock(element);
|
|
|
|
|
|
if (!tempElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
nodeElement = tempElement;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (protyle.model) {
|
|
|
|
|
|
getAllModels().outline.forEach(item => {
|
|
|
|
|
|
if (item.blockId === protyle.block.rootID) {
|
|
|
|
|
|
if (nodeElement.getAttribute("data-type") === "NodeHeading") {
|
|
|
|
|
|
item.setCurrent(nodeElement.getAttribute("data-node-id"));
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const headingElement = getPreviousHeading(nodeElement);
|
|
|
|
|
|
if (headingElement) {
|
|
|
|
|
|
item.setCurrent(headingElement.getAttribute("data-node-id"));
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
item.setCurrent("");
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private bindEvent(protyle: IProtyle) {
|
|
|
|
|
|
this.element.addEventListener("focusout", () => {
|
|
|
|
|
|
if (getSelection().rangeCount > 0 && !this.element.contains(getSelection().getRangeAt(0).startContainer)) {
|
|
|
|
|
|
// 对 ctrl+tab 切换后 range 已经在新页面中才会触发的 focusout 进行忽略,因为在切换之前已经 dispatchEvent 了。
|
|
|
|
|
|
if (!protyle.toolbar.range) {
|
|
|
|
|
|
protyle.toolbar.range = this.element.ownerDocument.createRange();
|
|
|
|
|
|
protyle.toolbar.range.setStart(getContenteditableElement(this.element) || this.element, 0);
|
|
|
|
|
|
protyle.toolbar.range.collapse(true);
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
protyle.toolbar.range = getEditorRange(this.element);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("copy", (event: ClipboardEvent & { target: HTMLElement }) => {
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4600
|
|
|
|
|
|
if (event.target.tagName === "PROTYLE-HTML") {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
const range = getEditorRange(protyle.wysiwyg.element);
|
|
|
|
|
|
const nodeElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (!nodeElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2022-05-29 10:06:02 +08:00
|
|
|
|
const selectImgElement = nodeElement.querySelector(".img--select");
|
2022-05-26 15:18:53 +08:00
|
|
|
|
let selectElements = Array.from(protyle.wysiwyg.element.querySelectorAll(".protyle-wysiwyg--select"));
|
|
|
|
|
|
if (selectElements.length === 0 && range.toString() === "" && !range.cloneContents().querySelector("img") &&
|
|
|
|
|
|
!selectImgElement) {
|
|
|
|
|
|
nodeElement.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
selectElements = [nodeElement];
|
|
|
|
|
|
}
|
|
|
|
|
|
let html = "";
|
|
|
|
|
|
let textPlain = "";
|
|
|
|
|
|
if (selectElements.length > 0) {
|
|
|
|
|
|
if (selectElements[0].getAttribute("data-type") === "NodeListItem" && selectElements[0].parentElement.childElementCount - 1 === selectElements.length) {
|
|
|
|
|
|
html = selectElements[0].parentElement.outerHTML;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.forEach(item => {
|
|
|
|
|
|
const topElement = getTopAloneElement(item);
|
|
|
|
|
|
html += removeEmbed(topElement);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const tempElement = document.createElement("div");
|
|
|
|
|
|
if (range.toString() !== "" && range.startContainer.isSameNode(range.endContainer) && range.startContainer.nodeType === 3
|
|
|
|
|
|
&& range.endOffset === range.startContainer.textContent.length && range.startOffset === 0) {
|
|
|
|
|
|
if (range.startContainer.parentElement.parentElement.getAttribute("data-type") === "NodeHeading") {
|
|
|
|
|
|
// 复制标题 https://github.com/siyuan-note/insider/issues/297
|
|
|
|
|
|
tempElement.append(range.startContainer.parentElement.parentElement.cloneNode(true));
|
|
|
|
|
|
} else if (!["DIV", "TD", "TH"].includes(range.startContainer.parentElement.tagName)) {
|
|
|
|
|
|
// 复制行内元素 https://github.com/siyuan-note/insider/issues/191
|
|
|
|
|
|
tempElement.append(range.startContainer.parentElement.cloneNode(true));
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 直接复制块 https://github.com/siyuan-note/insider/issues/318
|
|
|
|
|
|
tempElement.append(range.cloneContents());
|
|
|
|
|
|
}
|
|
|
|
|
|
html = tempElement.innerHTML;
|
|
|
|
|
|
} else if (selectImgElement) {
|
|
|
|
|
|
html = selectImgElement.outerHTML;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
tempElement.append(range.cloneContents());
|
|
|
|
|
|
const inlineMathElement = hasClosestByAttribute(range.commonAncestorContainer, "data-type", "inline-math");
|
|
|
|
|
|
if (inlineMathElement) {
|
|
|
|
|
|
// 表格内复制数学公式 https://ld246.com/article/1631708573504
|
|
|
|
|
|
html = inlineMathElement.outerHTML;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
html = tempElement.innerHTML;
|
|
|
|
|
|
}
|
|
|
|
|
|
// 不能使用 commonAncestorContainer https://ld246.com/article/1643282894693
|
|
|
|
|
|
if (hasClosestByAttribute(range.startContainer, "data-type", "NodeCodeBlock") ||
|
|
|
|
|
|
hasClosestByMatchTag(range.startContainer, "CODE")) {
|
|
|
|
|
|
textPlain = tempElement.textContent.replace(Constants.ZWSP, "");
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
event.clipboardData.setData("text/plain", textPlain || protyle.lute.BlockDOM2StdMd(html).trimEnd());
|
|
|
|
|
|
event.clipboardData.setData("text/html", html + Constants.ZWSP);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("cut", (event: ClipboardEvent & { target: HTMLElement }) => {
|
|
|
|
|
|
if (event.target.tagName === "PROTYLE-HTML") {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
|
|
|
|
protyle.breadcrumb.hide();
|
|
|
|
|
|
const range = getEditorRange(protyle.wysiwyg.element);
|
|
|
|
|
|
let nodeElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (!nodeElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2022-05-29 10:06:02 +08:00
|
|
|
|
const selectImgElement = nodeElement.querySelector(".img--select");
|
2022-05-26 15:18:53 +08:00
|
|
|
|
let selectElements = Array.from(protyle.wysiwyg.element.querySelectorAll(".protyle-wysiwyg--select"));
|
|
|
|
|
|
if (selectElements.length === 0 && range.toString() === "" && !range.cloneContents().querySelector("img") &&
|
|
|
|
|
|
!selectImgElement) {
|
|
|
|
|
|
nodeElement.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
selectElements = [nodeElement];
|
|
|
|
|
|
}
|
|
|
|
|
|
let html = "";
|
|
|
|
|
|
if (selectElements.length > 0) {
|
|
|
|
|
|
if (selectElements[0].getAttribute("data-type") === "NodeListItem" && selectElements[0].parentElement.childElementCount - 1 === selectElements.length) {
|
|
|
|
|
|
html = selectElements[0].parentElement.outerHTML;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.forEach(item => {
|
|
|
|
|
|
const topElement = getTopAloneElement(item);
|
|
|
|
|
|
html += removeEmbed(topElement);
|
|
|
|
|
|
});
|
|
|
|
|
|
if (selectElements[0].getAttribute("data-type") === "NodeListItem") {
|
|
|
|
|
|
html = `<div data-subtype="${selectElements[0].getAttribute("data-subtype")}" data-node-id="${Lute.NewNodeID()}" data-type="NodeList" class="list">${html}<div class="protyle-attr" contenteditable="false">${Constants.ZWSP}</div></div>`;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
removeBlock(protyle, nodeElement, range);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const id = nodeElement.getAttribute("data-node-id");
|
|
|
|
|
|
const oldHTML = nodeElement.outerHTML;
|
|
|
|
|
|
const tempElement = document.createElement("div");
|
|
|
|
|
|
// 首次选中标题时,range.startContainer 会为空
|
|
|
|
|
|
let startContainer = range.startContainer;
|
|
|
|
|
|
if (startContainer.nodeType === 3 && startContainer.textContent === "") {
|
|
|
|
|
|
const nextSibling = hasNextSibling(range.startContainer);
|
|
|
|
|
|
if (nextSibling) {
|
|
|
|
|
|
startContainer = nextSibling;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 选中整个标题 https://github.com/siyuan-note/siyuan/issues/4329
|
|
|
|
|
|
const headElement = hasClosestByAttribute(startContainer, "data-type", "NodeHeading");
|
|
|
|
|
|
if (headElement && range.toString() === headElement.firstElementChild.textContent) {
|
|
|
|
|
|
const doOperations: IOperation[] = [{
|
|
|
|
|
|
action: "delete",
|
|
|
|
|
|
id: headElement.getAttribute("data-node-id")
|
|
|
|
|
|
}];
|
|
|
|
|
|
const undoOperations: IOperation[] = [{
|
|
|
|
|
|
action: "insert",
|
|
|
|
|
|
id: headElement.getAttribute("data-node-id"),
|
|
|
|
|
|
data: headElement.outerHTML,
|
|
|
|
|
|
previousID: headElement.previousElementSibling?.getAttribute("data-node-id"),
|
|
|
|
|
|
parentID: headElement.parentElement?.getAttribute("data-node-id") || protyle.block.parentID
|
|
|
|
|
|
}];
|
|
|
|
|
|
if ((headElement.parentElement.childElementCount === 3 && headElement.parentElement.classList.contains("li")) ||
|
|
|
|
|
|
(headElement.parentElement.childElementCount === 2 && (headElement.parentElement.classList.contains("bq") || headElement.parentElement.classList.contains("sb")))) {
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4040
|
|
|
|
|
|
const emptyId = Lute.NewNodeID();
|
|
|
|
|
|
const emptyElement = genEmptyElement(false, false, emptyId);
|
|
|
|
|
|
doOperations.push({
|
|
|
|
|
|
id: emptyId,
|
|
|
|
|
|
data: emptyElement.outerHTML,
|
|
|
|
|
|
action: "insert",
|
|
|
|
|
|
parentID: headElement.parentElement.getAttribute("data-node-id")
|
|
|
|
|
|
});
|
|
|
|
|
|
undoOperations.push({
|
|
|
|
|
|
id: emptyId,
|
|
|
|
|
|
action: "delete",
|
|
|
|
|
|
});
|
|
|
|
|
|
headElement.before(emptyElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
transaction(protyle, doOperations, undoOperations);
|
2022-06-06 20:10:54 +08:00
|
|
|
|
focusSideBlock(headElement);
|
2022-05-26 15:18:53 +08:00
|
|
|
|
tempElement.append(headElement);
|
|
|
|
|
|
} else if (range.toString() !== "" && startContainer.isSameNode(range.endContainer) && range.startContainer.nodeType === 3
|
|
|
|
|
|
&& range.endOffset === range.endContainer.textContent.length && range.startOffset === 0 &&
|
|
|
|
|
|
!["DIV", "TD", "TH"].includes(range.startContainer.parentElement.tagName)) {
|
|
|
|
|
|
// 选中整个内联元素
|
|
|
|
|
|
tempElement.append(range.startContainer.parentElement);
|
|
|
|
|
|
} else if (selectImgElement) {
|
|
|
|
|
|
tempElement.append(selectImgElement);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (range.cloneContents().querySelectorAll("td, th").length > 0) {
|
|
|
|
|
|
// 表格内多格子 cut https://github.com/siyuan-note/insider/issues/564
|
|
|
|
|
|
const wbrElement = document.createElement("wbr");
|
|
|
|
|
|
range.insertNode(wbrElement);
|
|
|
|
|
|
range.setStartAfter(wbrElement);
|
|
|
|
|
|
tempElement.append(range.extractContents());
|
|
|
|
|
|
nodeElement.outerHTML = protyle.lute.SpinBlockDOM(nodeElement.outerHTML);
|
|
|
|
|
|
nodeElement = protyle.wysiwyg.element.querySelector(`[data-node-id="${id}"]`) as HTMLElement;
|
|
|
|
|
|
focusByWbr(nodeElement, range);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const inlineMathElement = hasClosestByAttribute(range.commonAncestorContainer, "data-type", "inline-math");
|
|
|
|
|
|
if (inlineMathElement) {
|
|
|
|
|
|
// 表格内剪切数学公式 https://ld246.com/article/1631708573504
|
|
|
|
|
|
tempElement.append(inlineMathElement);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
tempElement.append(range.extractContents());
|
|
|
|
|
|
let parentElement: Element | false;
|
|
|
|
|
|
// https://ld246.com/article/1647689760545
|
|
|
|
|
|
if (nodeElement.classList.contains("table")) {
|
|
|
|
|
|
parentElement = hasClosestByMatchTag(range.startContainer, "TD") || hasClosestByMatchTag(range.startContainer, "TH");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
parentElement = getContenteditableElement(nodeElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (parentElement) {
|
|
|
|
|
|
Array.from(parentElement.children).forEach(item => {
|
|
|
|
|
|
if (item.textContent === "") {
|
|
|
|
|
|
item.remove();
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
html = tempElement.innerHTML;
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4321
|
|
|
|
|
|
if (!nodeElement.classList.contains("table")) {
|
|
|
|
|
|
const editableElement = getContenteditableElement(nodeElement);
|
|
|
|
|
|
if (editableElement && editableElement.textContent === "") {
|
|
|
|
|
|
editableElement.innerHTML = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
nodeElement.setAttribute("updated", dayjs().format("YYYYMMDDHHmmss"));
|
|
|
|
|
|
if (nodeElement.getAttribute("data-type") === "NodeCodeBlock") {
|
|
|
|
|
|
range.insertNode(document.createElement("wbr"));
|
|
|
|
|
|
getContenteditableElement(nodeElement).removeAttribute("data-render");
|
|
|
|
|
|
highlightRender(nodeElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (nodeElement.parentElement.parentElement) {
|
|
|
|
|
|
// 选中 heading 时,使用删除的 transaction
|
|
|
|
|
|
updateTransaction(protyle, id, nodeElement.outerHTML, oldHTML);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
protyle.hint.render(protyle);
|
|
|
|
|
|
event.clipboardData.setData("text/plain", protyle.lute.BlockDOM2StdMd(html));
|
|
|
|
|
|
event.clipboardData.setData("text/html", Constants.ZWSP + html);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
let beforeContextmenuRange: Range;
|
|
|
|
|
|
this.element.addEventListener("contextmenu", (event) => {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
const selectElements = protyle.wysiwyg.element.querySelectorAll(".protyle-wysiwyg--select");
|
|
|
|
|
|
if (selectElements.length > 1) {
|
|
|
|
|
|
// 多选块
|
|
|
|
|
|
hideElements(["util"], protyle);
|
|
|
|
|
|
protyle.gutter.renderMenu(protyle, selectElements[0]);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, event.clientX, event.clientY);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const target = event.target as HTMLElement;
|
|
|
|
|
|
const embedElement = hasClosestByAttribute(target, "data-type", "NodeBlockQueryEmbed");
|
|
|
|
|
|
if (embedElement) {
|
|
|
|
|
|
if (getSelection().rangeCount === 0) {
|
|
|
|
|
|
focusSideBlock(embedElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
protyle.gutter.renderMenu(protyle, embedElement);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, event.clientX, event.clientY);
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (protyle.disabled) {
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
protyle.toolbar.range = getEditorRange(protyle.element);
|
|
|
|
|
|
const type = target.getAttribute("data-type");
|
|
|
|
|
|
if (type === "block-ref") {
|
|
|
|
|
|
refMenu(protyle, target);
|
2022-06-12 22:06:19 +08:00
|
|
|
|
const rect = target.getBoundingClientRect();
|
2022-06-01 23:52:22 +08:00
|
|
|
|
setPosition(window.siyuan.menus.menu.element, rect.left, rect.top + 13, 26);
|
2022-05-26 15:18:53 +08:00
|
|
|
|
// 阻止 popover
|
|
|
|
|
|
target.removeAttribute("data-type");
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
target.setAttribute("data-type", "block-ref");
|
|
|
|
|
|
}, 620);
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (type === "file-annotation-ref") {
|
|
|
|
|
|
protyle.toolbar.showFileAnnotationRef(protyle, target);
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (type === "a") {
|
2022-06-01 23:52:22 +08:00
|
|
|
|
linkMenu(protyle, target);
|
2022-06-12 22:06:19 +08:00
|
|
|
|
const rect = target.getBoundingClientRect();
|
2022-06-01 23:52:22 +08:00
|
|
|
|
setPosition(window.siyuan.menus.menu.element, rect.left, rect.top + 13, 26);
|
|
|
|
|
|
if (target.getAttribute("data-href")?.startsWith("siyuan://blocks")) {
|
2022-05-26 15:18:53 +08:00
|
|
|
|
// 阻止 popover
|
|
|
|
|
|
target.setAttribute("prevent-popover", "true");
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
target.removeAttribute("prevent-popover");
|
|
|
|
|
|
}, 620);
|
|
|
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (target.tagName === "IMG" && hasClosestByClassName(target, "img")) {
|
|
|
|
|
|
imgMenu(protyle, protyle.toolbar.range, target.parentElement.parentElement, {
|
2022-06-09 11:28:35 +08:00
|
|
|
|
clientX: event.clientX + 4,
|
2022-05-26 15:18:53 +08:00
|
|
|
|
clientY: event.clientY
|
|
|
|
|
|
});
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
const nodeElement = hasClosestBlock(target);
|
|
|
|
|
|
if (!nodeElement) {
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!isNotEditBlock(nodeElement) &&
|
|
|
|
|
|
(isMobile() || beforeContextmenuRange && nodeElement.contains(beforeContextmenuRange.startContainer))) {
|
|
|
|
|
|
if (!isMobile() || protyle.toolbar?.element.classList.contains("fn__none")) {
|
|
|
|
|
|
contentMenu(protyle, nodeElement);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, event.clientX, event.clientY + 13, 26);
|
|
|
|
|
|
protyle.toolbar?.element.classList.add("fn__none");
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (protyle.toolbar.range.toString() === "") {
|
|
|
|
|
|
hideElements(["util"], protyle);
|
|
|
|
|
|
protyle.gutter.renderMenu(protyle, nodeElement);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, event.clientX, event.clientY);
|
|
|
|
|
|
protyle.toolbar?.element.classList.add("fn__none");
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("pointerdown", () => {
|
|
|
|
|
|
if (getSelection().rangeCount > 0) {
|
|
|
|
|
|
beforeContextmenuRange = getSelection().getRangeAt(0);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
beforeContextmenuRange = undefined;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
2022-06-27 21:37:58 +08:00
|
|
|
|
let mousewheelTime = 0
|
2022-05-26 15:18:53 +08:00
|
|
|
|
this.element.addEventListener("mousewheel", (event: WheelEvent) => {
|
|
|
|
|
|
// https://ld246.com/article/1648865235549
|
2022-06-27 21:37:58 +08:00
|
|
|
|
if (event.timeStamp - mousewheelTime > 6000 &&
|
|
|
|
|
|
event.deltaY < 0 && !protyle.scroll.element.classList.contains("fn__none") &&
|
2022-05-26 15:18:53 +08:00
|
|
|
|
protyle.contentElement.clientHeight === protyle.contentElement.scrollHeight &&
|
|
|
|
|
|
protyle.wysiwyg.element.firstElementChild.getAttribute("data-eof") !== "true") {
|
|
|
|
|
|
fetchPost("/api/filetree/getDoc", {
|
|
|
|
|
|
id: protyle.wysiwyg.element.firstElementChild.getAttribute("data-node-id"),
|
|
|
|
|
|
mode: 1,
|
|
|
|
|
|
k: protyle.options.key || "",
|
|
|
|
|
|
size: Constants.SIZE_GET,
|
|
|
|
|
|
}, getResponse => {
|
|
|
|
|
|
onGet(getResponse, protyle, [Constants.CB_GET_BEFORE, Constants.CB_GET_UNCHANGEID]);
|
|
|
|
|
|
});
|
2022-06-27 21:37:58 +08:00
|
|
|
|
mousewheelTime = 0;
|
2022-05-26 15:18:53 +08:00
|
|
|
|
}
|
2022-06-27 21:37:58 +08:00
|
|
|
|
mousewheelTime = event.timeStamp;
|
2022-05-26 15:18:53 +08:00
|
|
|
|
if (event.deltaX === 0) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2022-06-27 21:37:58 +08:00
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4099
|
2022-05-26 15:18:53 +08:00
|
|
|
|
const tableElement = hasClosestByClassName(event.target as HTMLElement, "table");
|
|
|
|
|
|
if (tableElement) {
|
|
|
|
|
|
const tableSelectElement = tableElement.querySelector(".table__select") as HTMLElement;
|
|
|
|
|
|
if (tableSelectElement?.style.width) {
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
window.siyuan.menus.menu.remove();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}, {passive: true});
|
|
|
|
|
|
|
|
|
|
|
|
let overAttr = false;
|
|
|
|
|
|
this.element.addEventListener("mouseover", (event: MouseEvent & { target: Element }) => {
|
|
|
|
|
|
const attrElement = hasClosestByClassName(event.target, "protyle-attr");
|
|
|
|
|
|
if (attrElement) {
|
|
|
|
|
|
overAttr = true;
|
|
|
|
|
|
attrElement.parentElement.classList.add("protyle-wysiwyg--hl");
|
|
|
|
|
|
return;
|
|
|
|
|
|
} else if (overAttr) {
|
|
|
|
|
|
const hlElement = protyle.wysiwyg.element.querySelector(".protyle-wysiwyg--hl");
|
|
|
|
|
|
if (hlElement) {
|
|
|
|
|
|
hlElement.classList.remove("protyle-wysiwyg--hl");
|
|
|
|
|
|
}
|
|
|
|
|
|
overAttr = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (hasClosestByClassName(event.target, "protyle-action") || !protyle.options.render.gutter) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const nodeElement = hasClosestBlock(event.target);
|
|
|
|
|
|
if (nodeElement && (nodeElement.classList.contains("list") || nodeElement.classList.contains("li"))) {
|
|
|
|
|
|
// 光标在列表下部应显示右侧的元素,而不是列表本身。放在 globalShortcut 中的 mousemove 下处理
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (nodeElement) {
|
|
|
|
|
|
const embedElement = hasClosestByAttribute(nodeElement, "data-type", "NodeBlockQueryEmbed");
|
|
|
|
|
|
if (embedElement) {
|
|
|
|
|
|
protyle.gutter.render(embedElement, this.element);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
protyle.gutter.render(nodeElement, this.element);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("mousedown", (event: MouseEvent) => {
|
|
|
|
|
|
if (event.button === 2 || window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
// 右键
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/3026
|
|
|
|
|
|
hideElements(["select"], protyle);
|
|
|
|
|
|
}
|
|
|
|
|
|
const target = event.target as HTMLElement;
|
|
|
|
|
|
if (hasClosestByClassName(target, "protyle-action")) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const documentSelf = document;
|
|
|
|
|
|
const rect = protyle.element.getBoundingClientRect();
|
|
|
|
|
|
const mostLeft = rect.left + parseInt(protyle.wysiwyg.element.style.paddingLeft) + 1;
|
|
|
|
|
|
const mostRight = mostLeft + protyle.wysiwyg.element.firstElementChild.clientWidth - 1;
|
|
|
|
|
|
const mostBottom = rect.bottom;
|
|
|
|
|
|
const y = event.clientY;
|
|
|
|
|
|
// 图片、iframe、video 缩放
|
|
|
|
|
|
if (target.classList.contains("protyle-action__drag")) {
|
|
|
|
|
|
const nodeElement = hasClosestBlock(target);
|
|
|
|
|
|
if (!nodeElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
let isCenter = true;
|
|
|
|
|
|
if (["NodeIFrame", "NodeWidget", "NodeVideo"].includes(nodeElement.getAttribute("data-type"))) {
|
|
|
|
|
|
nodeElement.classList.add("iframe--drag");
|
|
|
|
|
|
if (nodeElement.style.textAlign === "left" || nodeElement.style.textAlign === "right") {
|
|
|
|
|
|
isCenter = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (target.parentElement.parentElement.getAttribute("data-type") === "img") {
|
|
|
|
|
|
target.parentElement.parentElement.classList.add("img--drag");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const id = nodeElement.getAttribute("data-node-id");
|
|
|
|
|
|
const html = nodeElement.outerHTML;
|
|
|
|
|
|
const x = event.clientX;
|
|
|
|
|
|
const dragElement = target.previousElementSibling as HTMLElement;
|
|
|
|
|
|
const dragWidth = dragElement.clientWidth;
|
|
|
|
|
|
const dragHeight = dragElement.clientHeight;
|
|
|
|
|
|
documentSelf.onmousemove = (moveEvent: MouseEvent) => {
|
|
|
|
|
|
if (dragElement.tagName === "IMG") {
|
|
|
|
|
|
dragElement.parentElement.parentElement.style.width = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
if (moveEvent.clientX > x - dragWidth + 8 && moveEvent.clientX < mostRight) {
|
|
|
|
|
|
if ((dragElement.tagName === "IMG" && dragElement.parentElement.parentElement.style.display !== "block") || !isCenter) {
|
|
|
|
|
|
dragElement.style.width = (dragWidth + (moveEvent.clientX - x)) + "px";
|
|
|
|
|
|
} else {
|
|
|
|
|
|
dragElement.style.width = (dragWidth + (moveEvent.clientX - x) * 2) + "px";
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (dragElement.tagName !== "IMG") {
|
|
|
|
|
|
if (moveEvent.clientY > y - dragHeight + 8 && moveEvent.clientY < mostBottom) {
|
|
|
|
|
|
dragElement.style.height = (dragHeight + (moveEvent.clientY - y)) + "px";
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
dragElement.parentElement.parentElement.style.maxWidth = (parseInt(dragElement.style.width) + 10) + "px";
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
documentSelf.onmouseup = () => {
|
|
|
|
|
|
documentSelf.onmousemove = null;
|
|
|
|
|
|
documentSelf.onmouseup = null;
|
|
|
|
|
|
documentSelf.ondragstart = null;
|
|
|
|
|
|
documentSelf.onselectstart = null;
|
|
|
|
|
|
documentSelf.onselect = null;
|
|
|
|
|
|
if (target.classList.contains("protyle-action__drag") && nodeElement) {
|
|
|
|
|
|
updateTransaction(protyle, id, nodeElement.outerHTML, html);
|
|
|
|
|
|
}
|
|
|
|
|
|
nodeElement.classList.remove("iframe--drag");
|
|
|
|
|
|
target.parentElement.parentElement.classList.remove("img--drag");
|
|
|
|
|
|
};
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
// table cell select
|
|
|
|
|
|
let tableBlockElement: HTMLElement | false;
|
|
|
|
|
|
if (target.tagName === "TH" || target.tagName === "TD" || target.firstElementChild?.tagName === "TABLE" || target.classList.contains("table__resize") || target.classList.contains("table__select")) {
|
|
|
|
|
|
tableBlockElement = hasClosestBlock(target);
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableBlockElement.querySelector(".table__select").removeAttribute("style");
|
|
|
|
|
|
window.siyuan.menus.menu.remove();
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
}
|
|
|
|
|
|
// 后续拖拽操作写在多选节点中
|
|
|
|
|
|
}
|
|
|
|
|
|
// table col resize
|
|
|
|
|
|
if (target.classList.contains("table__resize")) {
|
|
|
|
|
|
const nodeElement = hasClosestBlock(target);
|
|
|
|
|
|
if (!nodeElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const html = nodeElement.outerHTML;
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4455
|
|
|
|
|
|
if (getSelection().rangeCount > 0) {
|
|
|
|
|
|
getSelection().getRangeAt(0).collapse(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
|
nodeElement.firstElementChild.style.webkitUserModify = "read-only";
|
|
|
|
|
|
nodeElement.style.cursor = "col-resize";
|
|
|
|
|
|
target.removeAttribute("style");
|
|
|
|
|
|
const id = nodeElement.getAttribute("data-node-id");
|
|
|
|
|
|
const x = event.clientX;
|
|
|
|
|
|
const colElement = nodeElement.querySelectorAll("table col")[parseInt(target.getAttribute("data-col-index"))] as HTMLElement;
|
|
|
|
|
|
// 兼容 1.8.2
|
|
|
|
|
|
(nodeElement.querySelectorAll("table th")[parseInt(target.getAttribute("data-col-index"))] as HTMLElement).style.width = "";
|
|
|
|
|
|
const oldWidth = colElement.clientWidth;
|
|
|
|
|
|
const hasScroll = nodeElement.firstElementChild.clientWidth < nodeElement.firstElementChild.scrollWidth;
|
|
|
|
|
|
documentSelf.onmousemove = (moveEvent: MouseEvent) => {
|
|
|
|
|
|
if (nodeElement.style.textAlign === "center" && !hasScroll) {
|
|
|
|
|
|
colElement.style.width = (oldWidth + (moveEvent.clientX - x) * 2) + "px";
|
|
|
|
|
|
} else {
|
|
|
|
|
|
colElement.style.width = (oldWidth + (moveEvent.clientX - x)) + "px";
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
documentSelf.onmouseup = () => {
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
|
nodeElement.firstElementChild.style.webkitUserModify = "";
|
|
|
|
|
|
nodeElement.style.cursor = "";
|
|
|
|
|
|
documentSelf.onmousemove = null;
|
|
|
|
|
|
documentSelf.onmouseup = null;
|
|
|
|
|
|
documentSelf.ondragstart = null;
|
|
|
|
|
|
documentSelf.onselectstart = null;
|
|
|
|
|
|
documentSelf.onselect = null;
|
|
|
|
|
|
if (nodeElement) {
|
|
|
|
|
|
updateTransaction(protyle, id, nodeElement.outerHTML, html);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 多选节点
|
|
|
|
|
|
let x = event.clientX;
|
|
|
|
|
|
if (event.clientX > mostRight) {
|
|
|
|
|
|
x = mostRight;
|
|
|
|
|
|
} else if (event.clientX < mostLeft) {
|
|
|
|
|
|
x = mostLeft;
|
|
|
|
|
|
}
|
|
|
|
|
|
const mostTop = rect.top + protyle.breadcrumb.element.parentElement.clientHeight;
|
|
|
|
|
|
|
|
|
|
|
|
let mouseElement: Element;
|
|
|
|
|
|
let moveCellElement: HTMLElement;
|
|
|
|
|
|
documentSelf.onmousemove = (moveEvent: MouseEvent) => {
|
|
|
|
|
|
const moveTarget = moveEvent.target as HTMLElement;
|
|
|
|
|
|
// table cell select
|
|
|
|
|
|
if (tableBlockElement && tableBlockElement.contains(moveTarget) && !hasClosestByClassName(tableBlockElement, "protyle-wysiwyg__embed")) {
|
|
|
|
|
|
if ((moveTarget.tagName === "TH" || moveTarget.tagName === "TD") && !moveTarget.isSameNode(target) && (!moveCellElement || !moveCellElement.isSameNode(moveTarget))) {
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
|
tableBlockElement.firstElementChild.style.webkitUserModify = "read-only";
|
|
|
|
|
|
let width = target.offsetLeft + target.clientWidth - moveTarget.offsetLeft;
|
|
|
|
|
|
let left = moveTarget.offsetLeft;
|
|
|
|
|
|
if (target.offsetLeft === moveTarget.offsetLeft) {
|
|
|
|
|
|
width = Math.max(target.clientWidth, moveTarget.clientWidth);
|
|
|
|
|
|
} else if (target.offsetLeft < moveTarget.offsetLeft) {
|
|
|
|
|
|
width = moveTarget.offsetLeft + moveTarget.clientWidth - target.offsetLeft;
|
|
|
|
|
|
left = target.offsetLeft;
|
|
|
|
|
|
}
|
|
|
|
|
|
let height = target.offsetTop + target.clientHeight - moveTarget.offsetTop;
|
|
|
|
|
|
let top = moveTarget.offsetTop;
|
|
|
|
|
|
if (target.offsetTop === moveTarget.offsetTop) {
|
|
|
|
|
|
height = Math.max(target.clientHeight, moveTarget.clientHeight);
|
|
|
|
|
|
} else if (target.offsetTop < moveTarget.offsetTop) {
|
|
|
|
|
|
height = moveTarget.offsetTop + moveTarget.clientHeight - target.offsetTop;
|
|
|
|
|
|
top = target.offsetTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
// https://github.com/siyuan-note/insider/issues/1015
|
|
|
|
|
|
Array.from(tableBlockElement.querySelectorAll("th, td")).find((item: HTMLElement) => {
|
|
|
|
|
|
const updateWidth = item.offsetLeft < left + width && item.offsetLeft + item.clientWidth > left + width;
|
|
|
|
|
|
const updateWidth2 = item.offsetLeft < left && item.offsetLeft + item.clientWidth > left;
|
|
|
|
|
|
if (item.offsetTop < top && item.offsetTop + item.clientHeight > top) {
|
|
|
|
|
|
if ((item.offsetLeft + 6 > left && item.offsetLeft + item.clientWidth - 6 < left + width) || updateWidth || updateWidth2) {
|
|
|
|
|
|
height = top + height - item.offsetTop;
|
|
|
|
|
|
top = item.offsetTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (updateWidth) {
|
|
|
|
|
|
width = item.offsetLeft + item.clientWidth - left;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (updateWidth2) {
|
|
|
|
|
|
width = left + width - item.offsetLeft;
|
|
|
|
|
|
left = item.offsetLeft;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (item.offsetTop < top + height && item.offsetTop + item.clientHeight > top + height) {
|
|
|
|
|
|
if ((item.offsetLeft + 6 > left && item.offsetLeft + item.clientWidth - 6 < left + width) || updateWidth || updateWidth2) {
|
|
|
|
|
|
height = item.clientHeight + item.offsetTop - top;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (updateWidth) {
|
|
|
|
|
|
width = item.offsetLeft + item.clientWidth - left;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (updateWidth2) {
|
|
|
|
|
|
width = left + width - item.offsetLeft;
|
|
|
|
|
|
left = item.offsetLeft;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (updateWidth2 && item.offsetTop + 6 > top && item.offsetTop + item.clientHeight - 6 < top + height) {
|
|
|
|
|
|
width = left + width - item.offsetLeft;
|
|
|
|
|
|
left = item.offsetLeft;
|
|
|
|
|
|
} else if (updateWidth && item.offsetTop + 6 > top && item.offsetTop + item.clientHeight - 6 < top + height) {
|
|
|
|
|
|
width = item.offsetLeft + item.clientWidth - left;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.add("select");
|
|
|
|
|
|
tableBlockElement.querySelector(".table__select").setAttribute("style", `left:${left - tableBlockElement.firstElementChild.scrollLeft}px;top:${top}px;height:${height}px;width:${width + 1}px;`);
|
|
|
|
|
|
moveCellElement = moveTarget;
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
protyle.selectElement.classList.remove("fn__none");
|
|
|
|
|
|
// 向左选择,遇到 gutter 就不会弹出 toolbar
|
|
|
|
|
|
hideElements(["gutter"], protyle);
|
|
|
|
|
|
let newTop = 0;
|
|
|
|
|
|
let newLeft = 0;
|
|
|
|
|
|
let newWidth = 0;
|
|
|
|
|
|
let newHeight = 0;
|
|
|
|
|
|
if (moveEvent.clientX < x) {
|
|
|
|
|
|
if (moveEvent.clientX < mostLeft) {
|
|
|
|
|
|
// 向左越界
|
|
|
|
|
|
newLeft = mostLeft;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 向左
|
|
|
|
|
|
newLeft = moveEvent.clientX;
|
|
|
|
|
|
}
|
|
|
|
|
|
newWidth = x - newLeft;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (moveEvent.clientX > mostRight) {
|
|
|
|
|
|
// 向右越界
|
|
|
|
|
|
newLeft = x;
|
|
|
|
|
|
newWidth = mostRight - newLeft;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 向右
|
|
|
|
|
|
newLeft = x;
|
|
|
|
|
|
newWidth = moveEvent.clientX - x;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (moveEvent.clientY > y) {
|
|
|
|
|
|
if (moveEvent.clientY > mostBottom) {
|
|
|
|
|
|
// 向下越界
|
|
|
|
|
|
newTop = y;
|
|
|
|
|
|
newHeight = mostBottom - y;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 向下
|
|
|
|
|
|
newTop = y;
|
|
|
|
|
|
newHeight = moveEvent.clientY - y;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (moveEvent.clientY < mostTop) {
|
|
|
|
|
|
// 向上越界
|
|
|
|
|
|
newTop = mostTop;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 向上
|
|
|
|
|
|
newTop = moveEvent.clientY;
|
|
|
|
|
|
}
|
|
|
|
|
|
newHeight = y - newTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
protyle.selectElement.setAttribute("style", `background-color: ${protyle.selectElement.style.backgroundColor};top:${newTop}px;height:${newHeight}px;left:${newLeft + 2}px;width:${newWidth - 2}px;`);
|
|
|
|
|
|
const newMouseElement = document.elementFromPoint(moveEvent.clientX, moveEvent.clientY);
|
|
|
|
|
|
if (mouseElement && mouseElement.isSameNode(newMouseElement) && !mouseElement.classList.contains("protyle-wysiwyg") &&
|
|
|
|
|
|
!mouseElement.classList.contains("list") && !mouseElement.classList.contains("bq") && !mouseElement.classList.contains("sb")) {
|
|
|
|
|
|
// 性能优化,同一个p元素不进行选中计算
|
|
|
|
|
|
return;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
mouseElement = newMouseElement;
|
|
|
|
|
|
}
|
|
|
|
|
|
hideElements(["select"], protyle);
|
|
|
|
|
|
let firstElement = document.elementFromPoint(newLeft - 1, newTop);
|
|
|
|
|
|
if (!firstElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (firstElement.classList.contains("protyle-wysiwyg") || firstElement.classList.contains("list") || firstElement.classList.contains("sb") || firstElement.classList.contains("bq")) {
|
|
|
|
|
|
firstElement = document.elementFromPoint(newLeft - 1, newTop + 16);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!firstElement || firstElement.classList.contains("protyle-wysiwyg")) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const firstBlockElement = hasClosestBlock(firstElement);
|
|
|
|
|
|
if (!firstBlockElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
let selectElements: Element[] = [];
|
|
|
|
|
|
let currentElement: Element | boolean = firstBlockElement;
|
|
|
|
|
|
let hasJump = false;
|
|
|
|
|
|
while (currentElement) {
|
|
|
|
|
|
if (currentElement && !currentElement.classList.contains("protyle-attr")) {
|
|
|
|
|
|
const currentRect = currentElement.getBoundingClientRect();
|
|
|
|
|
|
if (currentRect.height > 0 && currentRect.top < newTop + newHeight && currentRect.left < newLeft + newWidth) {
|
|
|
|
|
|
if (hasJump) {
|
|
|
|
|
|
// 父节点的下个节点在选中范围内才可使用父节点作为选中节点
|
|
|
|
|
|
if (currentElement.nextElementSibling && !currentElement.nextElementSibling.classList.contains("protyle-attr")) {
|
|
|
|
|
|
const nextRect = currentElement.nextElementSibling.getBoundingClientRect();
|
|
|
|
|
|
if (nextRect.top < newTop + newHeight && nextRect.left < newLeft + newWidth) {
|
|
|
|
|
|
selectElements = [currentElement];
|
|
|
|
|
|
currentElement = currentElement.nextElementSibling;
|
|
|
|
|
|
hasJump = false;
|
|
|
|
|
|
} else if (currentElement.parentElement.classList.contains("sb")) {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement);
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement);
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.push(currentElement);
|
|
|
|
|
|
currentElement = currentElement.nextElementSibling;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (currentElement.parentElement.classList.contains("sb")) {
|
|
|
|
|
|
// 跳出超级块横向排版中的未选中元素
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement);
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
} else if (currentRect.height === 0 && currentRect.width === 0 && currentElement.parentElement.getAttribute("fold") === "1") {
|
|
|
|
|
|
currentElement = currentElement.parentElement;
|
|
|
|
|
|
selectElements = [];
|
|
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement);
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (selectElements.length === 1 && !selectElements[0].classList.contains("list") && !selectElements[0].classList.contains("bq") && !selectElements[0].classList.contains("sb")) {
|
|
|
|
|
|
// 只有一个 p 时不选中
|
|
|
|
|
|
protyle.selectElement.style.backgroundColor = "transparent";
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.forEach(item => {
|
|
|
|
|
|
if (!hasClosestByClassName(item, "protyle-wysiwyg__embed")) {
|
|
|
|
|
|
item.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
protyle.selectElement.style.backgroundColor = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
documentSelf.onmouseup = (mouseUpEvent) => {
|
|
|
|
|
|
documentSelf.onmousemove = null;
|
|
|
|
|
|
documentSelf.onmouseup = null;
|
|
|
|
|
|
documentSelf.ondragstart = null;
|
|
|
|
|
|
documentSelf.onselectstart = null;
|
|
|
|
|
|
documentSelf.onselect = null;
|
|
|
|
|
|
protyle.selectElement.classList.add("fn__none");
|
|
|
|
|
|
protyle.selectElement.removeAttribute("style");
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
|
tableBlockElement.firstElementChild.style.webkitUserModify = "";
|
|
|
|
|
|
const tableSelectElement = tableBlockElement.querySelector(".table__select") as HTMLElement;
|
|
|
|
|
|
if (tableSelectElement.getAttribute("style")) {
|
|
|
|
|
|
if (getSelection().rangeCount > 0) {
|
|
|
|
|
|
getSelection().getRangeAt(0).collapse(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
window.siyuan.menus.menu.remove();
|
|
|
|
|
|
window.siyuan.menus.menu.append(new MenuItem({
|
|
|
|
|
|
label: window.siyuan.languages.mergeCell,
|
|
|
|
|
|
click: () => {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
const selectCellElements: HTMLTableCellElement[] = [];
|
|
|
|
|
|
const colIndexList: number[] = [];
|
|
|
|
|
|
const colCount = tableBlockElement.querySelectorAll("th").length;
|
|
|
|
|
|
let fnNoneMax = 0;
|
|
|
|
|
|
const scrollLeft = tableBlockElement.firstElementChild.scrollLeft;
|
|
|
|
|
|
let isTHead = false;
|
|
|
|
|
|
let isTBody = false;
|
|
|
|
|
|
tableBlockElement.querySelectorAll("th, td").forEach((item: HTMLTableCellElement, index: number) => {
|
|
|
|
|
|
if (item.classList.contains("fn__none")) {
|
|
|
|
|
|
// 合并的元素中间有 fn__none 的元素
|
|
|
|
|
|
if (item.previousElementSibling && item.previousElementSibling.isSameNode(selectCellElements[selectCellElements.length - 1])) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
if (!isTHead && item.parentElement.parentElement.tagName === "THEAD") {
|
|
|
|
|
|
isTHead = true;
|
|
|
|
|
|
} else if (!isTBody && item.parentElement.parentElement.tagName === "TBODY") {
|
|
|
|
|
|
isTBody = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (index < fnNoneMax && colIndexList.includes((index + 1) % colCount)) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
if (!isTHead && item.parentElement.parentElement.tagName === "THEAD") {
|
|
|
|
|
|
isTHead = true;
|
|
|
|
|
|
} else if (!isTBody && item.parentElement.parentElement.tagName === "TBODY") {
|
|
|
|
|
|
isTBody = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (item.offsetLeft + 6 > tableSelectElement.offsetLeft + scrollLeft && item.offsetLeft + item.clientWidth - 6 < tableSelectElement.offsetLeft + scrollLeft + tableSelectElement.clientWidth &&
|
|
|
|
|
|
item.offsetTop + 6 > tableSelectElement.offsetTop && item.offsetTop + item.clientHeight - 6 < tableSelectElement.offsetTop + tableSelectElement.clientHeight) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
if (!isTHead && item.parentElement.parentElement.tagName === "THEAD") {
|
|
|
|
|
|
isTHead = true;
|
|
|
|
|
|
} else if (!isTBody && item.parentElement.parentElement.tagName === "TBODY") {
|
|
|
|
|
|
isTBody = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
colIndexList.push((index + 1) % colCount);
|
|
|
|
|
|
// https://github.com/siyuan-note/insider/issues/1014
|
|
|
|
|
|
fnNoneMax = Math.max((item.rowSpan - 1) * colCount + index + 1, fnNoneMax);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
const oldHTML = tableBlockElement.outerHTML;
|
|
|
|
|
|
let cellElement = selectCellElements[0];
|
|
|
|
|
|
let colSpan = cellElement.colSpan;
|
|
|
|
|
|
let index = 1;
|
|
|
|
|
|
while (cellElement.nextElementSibling && cellElement.nextElementSibling.isSameNode(selectCellElements[index])) {
|
|
|
|
|
|
cellElement = cellElement.nextElementSibling as HTMLTableCellElement;
|
|
|
|
|
|
if (!cellElement.classList.contains("fn__none")) { // https://github.com/siyuan-note/insider/issues/1007#issuecomment-1046195608
|
|
|
|
|
|
colSpan += cellElement.colSpan;
|
|
|
|
|
|
}
|
|
|
|
|
|
index++;
|
|
|
|
|
|
}
|
|
|
|
|
|
let html = "";
|
|
|
|
|
|
let rowElement: Element = selectCellElements[0].parentElement;
|
|
|
|
|
|
let rowSpan = selectCellElements[0].rowSpan;
|
|
|
|
|
|
selectCellElements.forEach((item, index) => {
|
|
|
|
|
|
let cellHTML = item.innerHTML.trim();
|
|
|
|
|
|
if (cellHTML.endsWith("<br>")) {
|
|
|
|
|
|
cellHTML = cellHTML.substr(0, cellHTML.length - 4);
|
|
|
|
|
|
}
|
|
|
|
|
|
html += cellHTML + ((!cellHTML || index === selectCellElements.length - 1) ? "" : "<br>");
|
|
|
|
|
|
if (index !== 0) {
|
|
|
|
|
|
if (!rowElement.isSameNode(item.parentElement)) {
|
|
|
|
|
|
if (!item.classList.contains("fn__none")) { // https://github.com/siyuan-note/insider/issues/1011
|
|
|
|
|
|
rowSpan += item.rowSpan;
|
|
|
|
|
|
}
|
|
|
|
|
|
rowElement = item.parentElement;
|
|
|
|
|
|
if (selectCellElements[0].parentElement.parentElement.tagName === "THEAD" && item.parentElement.parentElement.tagName !== "THEAD") {
|
|
|
|
|
|
selectCellElements[0].parentElement.parentElement.insertAdjacentElement("beforeend", item.parentElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
item.classList.add("fn__none");
|
|
|
|
|
|
item.innerHTML = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// https://github.com/siyuan-note/insider/issues/1017
|
|
|
|
|
|
if (isTHead && isTBody) {
|
|
|
|
|
|
rowElement = rowElement.parentElement.nextElementSibling.firstElementChild;
|
|
|
|
|
|
while (rowElement && rowElement.parentElement.tagName !== "THEAD") {
|
|
|
|
|
|
let colSpanCount = 0;
|
|
|
|
|
|
let noneCount = 0;
|
|
|
|
|
|
Array.from(rowElement.children).forEach((item: HTMLTableCellElement) => {
|
|
|
|
|
|
colSpanCount += item.colSpan - 1;
|
|
|
|
|
|
if (item.classList.contains("fn__none")) {
|
|
|
|
|
|
noneCount++;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
if (colSpanCount !== noneCount) {
|
|
|
|
|
|
selectCellElements[0].parentElement.parentElement.insertAdjacentElement("beforeend", rowElement);
|
|
|
|
|
|
rowElement = rowElement.parentElement.nextElementSibling.firstElementChild;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 合并背景色不会修改,需要等计算完毕
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
selectCellElements[0].innerHTML = html;
|
|
|
|
|
|
selectCellElements[0].colSpan = colSpan;
|
|
|
|
|
|
selectCellElements[0].rowSpan = rowSpan;
|
|
|
|
|
|
updateTransaction(protyle, tableBlockElement.getAttribute("data-node-id"), tableBlockElement.outerHTML, oldHTML);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}).element);
|
|
|
|
|
|
window.siyuan.menus.menu.append(new MenuItem({
|
|
|
|
|
|
label: window.siyuan.languages.remove,
|
|
|
|
|
|
icon: "iconTrashcan",
|
|
|
|
|
|
click() {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
const selectCellElements: HTMLTableCellElement[] = [];
|
|
|
|
|
|
const scrollLeft = tableBlockElement.firstElementChild.scrollLeft;
|
|
|
|
|
|
tableBlockElement.querySelectorAll("th, td").forEach((item: HTMLTableCellElement) => {
|
|
|
|
|
|
if (!item.classList.contains("fn__none") &&
|
|
|
|
|
|
item.offsetLeft + 6 > tableSelectElement.offsetLeft + scrollLeft && item.offsetLeft + item.clientWidth - 6 < tableSelectElement.offsetLeft + scrollLeft + tableSelectElement.clientWidth &&
|
|
|
|
|
|
item.offsetTop + 6 > tableSelectElement.offsetTop && item.offsetTop + item.clientHeight - 6 < tableSelectElement.offsetTop + tableSelectElement.clientHeight) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
const oldHTML = tableBlockElement.outerHTML;
|
|
|
|
|
|
selectCellElements.forEach(item => {
|
|
|
|
|
|
item.innerHTML = "";
|
|
|
|
|
|
});
|
|
|
|
|
|
updateTransaction(protyle, tableBlockElement.getAttribute("data-node-id"), tableBlockElement.outerHTML, oldHTML);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}).element);
|
2022-06-12 22:06:19 +08:00
|
|
|
|
window.siyuan.menus.menu.append(new MenuItem({
|
|
|
|
|
|
icon: "iconAlignLeft",
|
|
|
|
|
|
accelerator: window.siyuan.config.keymap.editor.general.alignLeft.custom,
|
|
|
|
|
|
label: window.siyuan.languages.alignLeft,
|
|
|
|
|
|
click: () => {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
const selectCellElements: HTMLTableCellElement[] = [];
|
|
|
|
|
|
const scrollLeft = tableBlockElement.firstElementChild.scrollLeft;
|
|
|
|
|
|
tableBlockElement.querySelectorAll("th, td").forEach((item: HTMLTableCellElement) => {
|
|
|
|
|
|
if (!item.classList.contains("fn__none") &&
|
|
|
|
|
|
item.offsetLeft + 6 > tableSelectElement.offsetLeft + scrollLeft && item.offsetLeft + item.clientWidth - 6 < tableSelectElement.offsetLeft + scrollLeft + tableSelectElement.clientWidth &&
|
|
|
|
|
|
item.offsetTop + 6 > tableSelectElement.offsetTop && item.offsetTop + item.clientHeight - 6 < tableSelectElement.offsetTop + tableSelectElement.clientHeight &&
|
|
|
|
|
|
(selectCellElements.length === 0 || (selectCellElements.length > 0 && item.offsetTop === selectCellElements[0].offsetTop))) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
setTableAlign(protyle, selectCellElements, tableBlockElement, "left", getEditorRange(tableBlockElement));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}).element);
|
|
|
|
|
|
window.siyuan.menus.menu.append(new MenuItem({
|
|
|
|
|
|
icon: "iconAlignCenter",
|
|
|
|
|
|
accelerator: window.siyuan.config.keymap.editor.general.alignCenter.custom,
|
|
|
|
|
|
label: window.siyuan.languages.alignCenter,
|
|
|
|
|
|
click: () => {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
const selectCellElements: HTMLTableCellElement[] = [];
|
|
|
|
|
|
const scrollLeft = tableBlockElement.firstElementChild.scrollLeft;
|
|
|
|
|
|
tableBlockElement.querySelectorAll("th, td").forEach((item: HTMLTableCellElement) => {
|
|
|
|
|
|
if (!item.classList.contains("fn__none") &&
|
|
|
|
|
|
item.offsetLeft + 6 > tableSelectElement.offsetLeft + scrollLeft && item.offsetLeft + item.clientWidth - 6 < tableSelectElement.offsetLeft + scrollLeft + tableSelectElement.clientWidth &&
|
|
|
|
|
|
item.offsetTop + 6 > tableSelectElement.offsetTop && item.offsetTop + item.clientHeight - 6 < tableSelectElement.offsetTop + tableSelectElement.clientHeight &&
|
|
|
|
|
|
(selectCellElements.length === 0 || (selectCellElements.length > 0 && item.offsetTop === selectCellElements[0].offsetTop))) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
setTableAlign(protyle, selectCellElements, tableBlockElement, "center", getEditorRange(tableBlockElement));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}).element);
|
|
|
|
|
|
window.siyuan.menus.menu.append(new MenuItem({
|
|
|
|
|
|
icon: "iconAlignRight",
|
|
|
|
|
|
accelerator: window.siyuan.config.keymap.editor.general.alignRight.custom,
|
|
|
|
|
|
label: window.siyuan.languages.alignRight,
|
|
|
|
|
|
click: () => {
|
|
|
|
|
|
if (tableBlockElement) {
|
|
|
|
|
|
const selectCellElements: HTMLTableCellElement[] = [];
|
|
|
|
|
|
const scrollLeft = tableBlockElement.firstElementChild.scrollLeft;
|
|
|
|
|
|
tableBlockElement.querySelectorAll("th, td").forEach((item: HTMLTableCellElement) => {
|
|
|
|
|
|
if (!item.classList.contains("fn__none") &&
|
|
|
|
|
|
item.offsetLeft + 6 > tableSelectElement.offsetLeft + scrollLeft && item.offsetLeft + item.clientWidth - 6 < tableSelectElement.offsetLeft + scrollLeft + tableSelectElement.clientWidth &&
|
|
|
|
|
|
item.offsetTop + 6 > tableSelectElement.offsetTop && item.offsetTop + item.clientHeight - 6 < tableSelectElement.offsetTop + tableSelectElement.clientHeight &&
|
|
|
|
|
|
(selectCellElements.length === 0 || (selectCellElements.length > 0 && item.offsetTop === selectCellElements[0].offsetTop))) {
|
|
|
|
|
|
selectCellElements.push(item);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
tableBlockElement.querySelector("table").classList.remove("select");
|
|
|
|
|
|
tableSelectElement.removeAttribute("style");
|
|
|
|
|
|
setTableAlign(protyle, selectCellElements, tableBlockElement, "right", getEditorRange(tableBlockElement));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}).element);
|
2022-05-26 15:18:53 +08:00
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, mouseUpEvent.clientX - 16, mouseUpEvent.clientY - 46);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 划选后不能存在跨块的 range https://github.com/siyuan-note/siyuan/issues/4473
|
|
|
|
|
|
if (getSelection().rangeCount > 0) {
|
|
|
|
|
|
const range = getSelection().getRangeAt(0);
|
|
|
|
|
|
if (range.toString() === "" ||
|
|
|
|
|
|
window.siyuan.shiftIsPressed // https://ld246.com/article/1650096678723
|
|
|
|
|
|
) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (protyle.wysiwyg.element.querySelector(".protyle-wysiwyg--select")) {
|
|
|
|
|
|
range.collapse(true);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const startBlockElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
let endBlockElement: false | HTMLElement;
|
|
|
|
|
|
if (mouseUpEvent.detail === 3 && range.endContainer.nodeType !== 3 && (range.endContainer as HTMLElement).tagName === "DIV" && range.endOffset === 0) {
|
|
|
|
|
|
// 三击选中段落块时,rangeEnd 会在下一个块
|
|
|
|
|
|
if ((range.endContainer as HTMLElement).classList.contains("protyle-attr")) {
|
|
|
|
|
|
// 三击在悬浮层中会选择到 attr https://github.com/siyuan-note/siyuan/issues/4636
|
|
|
|
|
|
range.setEndAfter(range.endContainer.previousSibling.lastChild);
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
endBlockElement = hasClosestBlock(range.endContainer);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (startBlockElement && endBlockElement && !endBlockElement.isSameNode(startBlockElement)) {
|
|
|
|
|
|
range.collapse(true);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("paste", (event: ClipboardEvent & { target: HTMLElement }) => {
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4600
|
|
|
|
|
|
if (event.target.tagName === "PROTYLE-HTML") {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const range = getEditorRange(this.element);
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/3257
|
|
|
|
|
|
if (range.toString() === "" && range.startContainer.nodeType === 3 && !protyle.toolbar.isNewEmptyInline &&
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/4118
|
|
|
|
|
|
range.startOffset >= range.startContainer.textContent.length && !hasNextSibling(range.startContainer) &&
|
|
|
|
|
|
protyle.toolbar.getCurrentType(range).length > 0) {
|
|
|
|
|
|
// 粗体等行内元素粘贴
|
|
|
|
|
|
range.setEndAfter(range.startContainer.parentElement);
|
|
|
|
|
|
range.collapse(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
paste(protyle, event);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// 输入法测试点 https://github.com/siyuan-note/siyuan/issues/3027
|
|
|
|
|
|
let isComposition = false; // for iPhone
|
|
|
|
|
|
this.element.addEventListener("compositionstart", (event) => {
|
|
|
|
|
|
// 搜狗输入法划选输入后无 data https://github.com/siyuan-note/siyuan/issues/4672
|
|
|
|
|
|
const range = getEditorRange(protyle.wysiwyg.element);
|
|
|
|
|
|
const nodeElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (nodeElement && typeof protyle.wysiwyg.lastHTMLs[nodeElement.getAttribute("data-node-id")] === "undefined") {
|
|
|
|
|
|
range.insertNode(document.createElement("wbr"));
|
|
|
|
|
|
protyle.wysiwyg.lastHTMLs[nodeElement.getAttribute("data-node-id")] = nodeElement.outerHTML;
|
|
|
|
|
|
nodeElement.querySelector("wbr").remove();
|
|
|
|
|
|
}
|
|
|
|
|
|
isComposition = true;
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("compositionend", (event: InputEvent) => {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
isComposition = false;
|
|
|
|
|
|
const range = getEditorRange(this.element);
|
|
|
|
|
|
const blockElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (blockElement && blockElement.getAttribute("data-type") === "NodeHTMLBlock") {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!blockElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if ("" !== event.data) {
|
|
|
|
|
|
this.escapeInline(protyle, range, event);
|
|
|
|
|
|
// 小鹤音形 ;k 不能使用 setTimeout;
|
|
|
|
|
|
// wysiwyg.element contenteditable 为 false 时,连拼 needRender 必须为 false
|
|
|
|
|
|
// hr 渲染;任务列表、粗体、数学公示结尾 needRender 必须为 true
|
|
|
|
|
|
input(protyle, blockElement, range, true);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const id = blockElement.getAttribute("data-node-id");
|
|
|
|
|
|
if (protyle.wysiwyg.lastHTMLs[id]) {
|
|
|
|
|
|
updateTransaction(protyle, id, blockElement.outerHTML, protyle.wysiwyg.lastHTMLs[id]);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("input", (event: InputEvent) => {
|
|
|
|
|
|
const target = event.target as HTMLElement;
|
|
|
|
|
|
if (target.tagName === "VIDEO" || target.tagName === "AUDIO") {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const range = getEditorRange(this.element);
|
|
|
|
|
|
const blockElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (!blockElement) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (blockElement && blockElement.getAttribute("data-type") === "NodeHTMLBlock") {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (event.isComposing || isComposition ||
|
|
|
|
|
|
// https://github.com/siyuan-note/siyuan/issues/337 编辑器内容拖拽问题
|
|
|
|
|
|
event.inputType === "deleteByDrag" || event.inputType === "insertFromDrop"
|
|
|
|
|
|
) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
this.escapeInline(protyle, range, event);
|
|
|
|
|
|
if (/^\d{1}$/.test(event.data) || event.data === "‘" || event.data === "“") {
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
input(protyle, blockElement, range, true); // 搜狗拼音数字后面句号变为点;Mac 反向双引号无法输入
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (event.data === ":") {
|
|
|
|
|
|
protyle.hint.enableEmoji = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
input(protyle, blockElement, range, true);
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("keyup", (event) => {
|
|
|
|
|
|
const range = getEditorRange(this.element).cloneRange();
|
|
|
|
|
|
if (event.key !== "PageUp" && event.key !== "PageDown" && event.key !== "Home" && event.key !== "End" && event.key.indexOf("Arrow") === -1 &&
|
|
|
|
|
|
event.key !== "Alt" && event.key !== "Shift" && event.key !== "CapsLock" && event.key !== "Escape" && event.key !== "Meta" && !/^F\d{1,2}$/.test(event.key) &&
|
|
|
|
|
|
(!event.isComposing || (event.isComposing && range.toString() !== "")) // https://github.com/siyuan-note/siyuan/issues/4341
|
|
|
|
|
|
) {
|
|
|
|
|
|
// 搜狗输入法不走 keydown,需重新记录历史状态
|
|
|
|
|
|
const nodeElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (nodeElement && typeof protyle.wysiwyg.lastHTMLs[nodeElement.getAttribute("data-node-id")] === "undefined") {
|
|
|
|
|
|
range.insertNode(document.createElement("wbr"));
|
|
|
|
|
|
protyle.wysiwyg.lastHTMLs[nodeElement.getAttribute("data-node-id")] = nodeElement.outerHTML;
|
|
|
|
|
|
nodeElement.querySelector("wbr").remove();
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 需放在 lastHTMLs 后,否则 https://github.com/siyuan-note/siyuan/issues/4388
|
|
|
|
|
|
if (this.preventKeyup) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ((event.shiftKey || isCtrl(event)) && !event.isComposing && range.toString() !== "") {
|
|
|
|
|
|
// 工具栏
|
|
|
|
|
|
protyle.toolbar.render(protyle, range, event);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (event.eventPhase !== 3 && !event.shiftKey && (event.key.indexOf("Arrow") > -1 || event.key === "Home" || event.key === "End" || event.key === "PageUp" || event.key === "PageDown") && !event.isComposing) {
|
|
|
|
|
|
// 不可编辑的块处理
|
|
|
|
|
|
const nodeElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
if (nodeElement) {
|
|
|
|
|
|
if (isNotEditBlock(nodeElement) && !this.element.querySelector(".protyle-wysiwyg--select")) {
|
|
|
|
|
|
nodeElement.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
}
|
|
|
|
|
|
this.setEmptyOutline(protyle, nodeElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.element.addEventListener("dblclick", (event: MouseEvent & { target: HTMLElement }) => {
|
|
|
|
|
|
if (event.target.tagName === "IMG") {
|
|
|
|
|
|
previewImage((event.target as HTMLImageElement).src, protyle.block.rootID);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
let shiftStartElement: HTMLElement;
|
|
|
|
|
|
this.element.addEventListener("click", (event: MouseEvent & { target: HTMLElement }) => {
|
|
|
|
|
|
hideElements(["hint", "util"], protyle);
|
|
|
|
|
|
if (!window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
shiftStartElement = undefined;
|
|
|
|
|
|
}
|
|
|
|
|
|
this.setEmptyOutline(protyle, event.target);
|
|
|
|
|
|
const tableElement = hasClosestByClassName(event.target, "table");
|
|
|
|
|
|
this.element.querySelectorAll(".table").forEach(item => {
|
|
|
|
|
|
if (!tableElement || !item.isSameNode(tableElement)) {
|
|
|
|
|
|
item.querySelector("table").classList.remove("select");
|
|
|
|
|
|
item.querySelector(".table__select").removeAttribute("style");
|
|
|
|
|
|
}
|
|
|
|
|
|
if (tableElement && tableElement.isSameNode(item) && item.querySelector(".table__select").getAttribute("style")) {
|
|
|
|
|
|
// 防止合并单元格的菜单消失
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
const range = getEditorRange(this.element);
|
|
|
|
|
|
// 需放在嵌入块之前,否则嵌入块内的引用、链接、pdf 双链无法点击打开 https://ld246.com/article/1630479789513
|
|
|
|
|
|
const blockRefElement = hasClosestByAttribute(event.target, "data-type", "block-ref");
|
|
|
|
|
|
const aElement = hasClosestByAttribute(event.target, "data-type", "a");
|
|
|
|
|
|
if (blockRefElement ||
|
|
|
|
|
|
(aElement && aElement.getAttribute("data-href").startsWith("siyuan://blocks/"))
|
|
|
|
|
|
) {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
hideElements(["dialog", "toolbar"], protyle);
|
|
|
|
|
|
if (range.toString() !== "" && !window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
// 选择不打开引用
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
let refBlockId: string;
|
|
|
|
|
|
if (blockRefElement) {
|
|
|
|
|
|
refBlockId = blockRefElement.getAttribute("data-id");
|
|
|
|
|
|
} else if (aElement) {
|
|
|
|
|
|
refBlockId = aElement.getAttribute("data-href").substring(16, 38);
|
|
|
|
|
|
}
|
2022-06-01 10:12:14 +08:00
|
|
|
|
|
|
|
|
|
|
fetchPost("/api/block/checkBlockFold", {id: refBlockId}, (foldResponse) => {
|
|
|
|
|
|
if (isMobile()) {
|
|
|
|
|
|
openMobileFileById(refBlockId, !foldResponse.data, foldResponse.data ? [Constants.CB_GET_ALL, Constants.CB_GET_HL] : [Constants.CB_GET_HL]);
|
|
|
|
|
|
} else {
|
2022-05-26 15:18:53 +08:00
|
|
|
|
if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: refBlockId,
|
|
|
|
|
|
position: "bottom",
|
|
|
|
|
|
hasContext: !foldResponse.data,
|
|
|
|
|
|
action: foldResponse.data ? [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL] : [Constants.CB_GET_FOCUS],
|
|
|
|
|
|
zoomIn: foldResponse.data
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (window.siyuan.altIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: refBlockId,
|
|
|
|
|
|
position: "right",
|
|
|
|
|
|
hasContext: !foldResponse.data,
|
|
|
|
|
|
action: foldResponse.data ? [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL] : [Constants.CB_GET_FOCUS],
|
|
|
|
|
|
zoomIn: foldResponse.data
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: refBlockId,
|
|
|
|
|
|
hasContext: !foldResponse.data,
|
|
|
|
|
|
action: foldResponse.data ? [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL] : [Constants.CB_GET_HL],
|
|
|
|
|
|
keepCursor: true,
|
|
|
|
|
|
zoomIn: foldResponse.data
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: refBlockId,
|
|
|
|
|
|
hasContext: !foldResponse.data,
|
|
|
|
|
|
action: foldResponse.data ? [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL] : [Constants.CB_GET_FOCUS],
|
|
|
|
|
|
zoomIn: foldResponse.data
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2022-06-01 10:12:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
});
|
2022-05-26 15:18:53 +08:00
|
|
|
|
if (protyle.model) {
|
|
|
|
|
|
// 打开双链需记录到后退中 https://github.com/siyuan-note/insider/issues/801
|
|
|
|
|
|
let blockElement: HTMLElement | false;
|
|
|
|
|
|
if (blockRefElement) {
|
|
|
|
|
|
blockElement = hasClosestBlock(blockRefElement);
|
|
|
|
|
|
} else if (aElement) {
|
|
|
|
|
|
blockElement = hasClosestBlock(aElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (blockElement) {
|
|
|
|
|
|
pushBack(protyle, getEditorRange(this.element), blockElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const fileElement = hasClosestByAttribute(event.target, "data-type", "file-annotation-ref");
|
|
|
|
|
|
if (fileElement && range.toString() === "") {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
const fileIds = fileElement.getAttribute("data-id").split("/");
|
|
|
|
|
|
const linkAddress = `assets/${fileIds[1]}`;
|
|
|
|
|
|
if (isMobile()) {
|
|
|
|
|
|
if (typeof window.JSAndroid === "undefined") {
|
|
|
|
|
|
window.open(linkAddress);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
window.JSAndroid.openExternal(linkAddress);
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
openBy(linkAddress, "folder");
|
|
|
|
|
|
} else if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
openBy(linkAddress, "app");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
openAsset(linkAddress, fileIds[2], "right");
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (aElement && !window.siyuan.altIsPressed) {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
const linkAddress = aElement.getAttribute("data-href");
|
|
|
|
|
|
if (isMobile()) {
|
|
|
|
|
|
if (window.siyuan.config.system.container === "ios") {
|
|
|
|
|
|
window.location.href = linkAddress;
|
|
|
|
|
|
} else if (window.siyuan.config.system.container === "android" && window.JSAndroid) {
|
|
|
|
|
|
window.JSAndroid.openExternal(linkAddress);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
window.open(linkAddress);
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (isLocalPath(linkAddress)) {
|
|
|
|
|
|
const linkPathname = linkAddress.split("?page")[0];
|
|
|
|
|
|
if (Constants.SIYUAN_ASSETS_EXTS.includes(pathPosix().extname(linkPathname)) &&
|
|
|
|
|
|
(!linkPathname.endsWith(".pdf") ||
|
|
|
|
|
|
(linkPathname.endsWith(".pdf") && !linkAddress.startsWith("file://")))
|
|
|
|
|
|
) {
|
|
|
|
|
|
if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
openBy(linkAddress, "folder");
|
|
|
|
|
|
} else if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
openBy(linkAddress, "app");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
const page = getSearch("page", linkAddress);
|
|
|
|
|
|
openAsset(linkPathname, page === "" ? undefined : parseInt(page), "right");
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
/// #if !BROWSER
|
|
|
|
|
|
if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
openBy(linkAddress, "folder");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
openBy(linkAddress, "app");
|
|
|
|
|
|
}
|
|
|
|
|
|
/// #else
|
|
|
|
|
|
if (window.siyuan.config.system.container === "ios") {
|
|
|
|
|
|
window.location.href = linkAddress;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
window.open(linkAddress);
|
|
|
|
|
|
}
|
|
|
|
|
|
/// #endif
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
/// #if !BROWSER
|
|
|
|
|
|
shell.openExternal(linkAddress).catch((e) => {
|
|
|
|
|
|
console.log("openExternal error:" + e);
|
|
|
|
|
|
});
|
|
|
|
|
|
/// #else
|
|
|
|
|
|
if (window.siyuan.config.system.container === "ios") {
|
|
|
|
|
|
window.location.href = linkAddress;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
window.open(linkAddress);
|
|
|
|
|
|
}
|
|
|
|
|
|
/// #endif
|
|
|
|
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const tagElement = hasClosestByAttribute(event.target, "data-type", "tag");
|
|
|
|
|
|
if (tagElement && !window.siyuan.altIsPressed && protyle.model) {
|
|
|
|
|
|
openGlobalSearch(`#${tagElement.textContent}#`, !window.siyuan.ctrlIsPressed);
|
|
|
|
|
|
hideElements(["dialog"]);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const embedItemElement = hasClosestByClassName(event.target, "protyle-wysiwyg__embed");
|
|
|
|
|
|
if (embedItemElement) {
|
2022-06-23 01:19:04 +08:00
|
|
|
|
const embedId = embedItemElement.getAttribute("data-id");
|
2022-05-26 15:18:53 +08:00
|
|
|
|
if (isMobile()) {
|
2022-06-19 00:34:15 +08:00
|
|
|
|
openMobileFileById(embedId, false, [Constants.CB_GET_ALL]);
|
|
|
|
|
|
} else if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: embedId,
|
|
|
|
|
|
position: "bottom",
|
|
|
|
|
|
hasContext: false,
|
|
|
|
|
|
action: [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL],
|
|
|
|
|
|
zoomIn: true
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (window.siyuan.altIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: embedId,
|
|
|
|
|
|
position: "right",
|
|
|
|
|
|
hasContext: false,
|
|
|
|
|
|
action: [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL],
|
|
|
|
|
|
zoomIn: true
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
openFileById({
|
|
|
|
|
|
id: embedId,
|
|
|
|
|
|
hasContext: false,
|
|
|
|
|
|
action: [Constants.CB_GET_FOCUS, Constants.CB_GET_ALL],
|
|
|
|
|
|
keepCursor: true,
|
|
|
|
|
|
zoomIn: true
|
|
|
|
|
|
});
|
2022-05-26 15:18:53 +08:00
|
|
|
|
} else if (!protyle.disabled) {
|
|
|
|
|
|
window.siyuan.blockPanels.push(new BlockPanel({
|
|
|
|
|
|
targetElement: embedItemElement,
|
2022-06-19 00:34:15 +08:00
|
|
|
|
nodeIds: [embedId],
|
2022-05-26 15:18:53 +08:00
|
|
|
|
}));
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const attrBookmarkElement = hasClosestByClassName(event.target, "protyle-attr--bookmark");
|
|
|
|
|
|
if (attrBookmarkElement) {
|
|
|
|
|
|
openAttr(attrBookmarkElement.parentElement.parentElement, protyle, "bookmark");
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const attrNameElement = hasClosestByClassName(event.target, "protyle-attr--name");
|
|
|
|
|
|
if (attrNameElement) {
|
|
|
|
|
|
openAttr(attrNameElement.parentElement.parentElement, protyle, "name");
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const attrAliasElement = hasClosestByClassName(event.target, "protyle-attr--alias");
|
|
|
|
|
|
if (attrAliasElement) {
|
|
|
|
|
|
openAttr(attrAliasElement.parentElement.parentElement, protyle, "alias");
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const attrMemoElement = hasClosestByClassName(event.target, "protyle-attr--memo");
|
|
|
|
|
|
if (attrMemoElement) {
|
|
|
|
|
|
openAttr(attrMemoElement.parentElement.parentElement, protyle, "memo");
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (hasTopClosestByClassName(event.target, "protyle-action__copy")) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const editElement = hasClosestByClassName(event.target, "protyle-action__edit");
|
|
|
|
|
|
if (editElement && !protyle.disabled) {
|
|
|
|
|
|
protyle.toolbar.showRender(protyle, editElement.parentElement.parentElement);
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const menuElement = hasClosestByClassName(event.target, "protyle-action__menu");
|
|
|
|
|
|
if (menuElement && !protyle.disabled) {
|
|
|
|
|
|
protyle.gutter.renderMenu(protyle, menuElement.parentElement.parentElement);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
const rect = menuElement.getBoundingClientRect();
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, rect.left - window.siyuan.menus.menu.element.clientWidth, rect.top);
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const reloadElement = hasClosestByClassName(event.target, "protyle-action__reload");
|
|
|
|
|
|
if (reloadElement) {
|
|
|
|
|
|
const embedReloadElement = hasClosestByAttribute(reloadElement, "data-type", "NodeBlockQueryEmbed");
|
|
|
|
|
|
if (embedReloadElement) {
|
|
|
|
|
|
embedReloadElement.removeAttribute("data-render");
|
|
|
|
|
|
blockRender(protyle, embedReloadElement);
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const languageElement = hasClosestByClassName(event.target, "protyle-action__language");
|
|
|
|
|
|
if (languageElement && !protyle.disabled) {
|
|
|
|
|
|
protyle.toolbar.showCodeLanguage(protyle, languageElement);
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 需放在属性后,否则数学公式无法点击属性;需放在 action 后,否则嵌入块的的 action 无法打开;需放在嵌入块后,否则嵌入块中的数学公式会被打开
|
|
|
|
|
|
const mathElement = hasClosestByAttribute(event.target, "data-subtype", "math");
|
|
|
|
|
|
if (!window.siyuan.shiftIsPressed && !window.siyuan.ctrlIsPressed && mathElement && !protyle.disabled) {
|
|
|
|
|
|
protyle.toolbar.showRender(protyle, mathElement);
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const actionElement = hasClosestByClassName(event.target, "protyle-action");
|
|
|
|
|
|
if (actionElement) {
|
|
|
|
|
|
const type = actionElement.parentElement.parentElement.getAttribute("data-type");
|
|
|
|
|
|
if (type === "img" && !protyle.disabled) {
|
|
|
|
|
|
imgMenu(protyle, range, actionElement.parentElement.parentElement, {
|
2022-06-09 11:28:35 +08:00
|
|
|
|
clientX: event.clientX + 4,
|
2022-05-26 15:18:53 +08:00
|
|
|
|
clientY: event.clientY
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (!protyle.disabled && actionElement.parentElement.classList.contains("li")) {
|
|
|
|
|
|
if (window.siyuan.altIsPressed) {
|
|
|
|
|
|
// 展开/折叠当前层级的所有列表项
|
|
|
|
|
|
if (actionElement.parentElement.parentElement.classList.contains("protyle-wysiwyg")) {
|
|
|
|
|
|
// 缩放列表项 https://ld246.com/article/1653123034794
|
|
|
|
|
|
setFold(protyle, actionElement.parentElement);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
let hasFold = true;
|
|
|
|
|
|
const oldHTML = actionElement.parentElement.parentElement.outerHTML;
|
|
|
|
|
|
Array.from(actionElement.parentElement.parentElement.children).find((listItemElement) => {
|
|
|
|
|
|
if (listItemElement.classList.contains("li")) {
|
|
|
|
|
|
if (listItemElement.getAttribute("fold") !== "1" && listItemElement.childElementCount > 3) {
|
|
|
|
|
|
hasFold = false;
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
Array.from(actionElement.parentElement.parentElement.children).find((listItemElement) => {
|
|
|
|
|
|
if (listItemElement.classList.contains("li")) {
|
|
|
|
|
|
if (hasFold) {
|
|
|
|
|
|
listItemElement.removeAttribute("fold");
|
|
|
|
|
|
} else if (listItemElement.childElementCount > 3) {
|
|
|
|
|
|
listItemElement.setAttribute("fold", "1");
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
updateTransaction(protyle, actionElement.parentElement.parentElement.getAttribute("data-node-id"), actionElement.parentElement.parentElement.outerHTML, oldHTML);
|
|
|
|
|
|
}
|
|
|
|
|
|
hideElements(["gutter"], protyle);
|
|
|
|
|
|
} else if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
openAttr(actionElement.parentElement, protyle);
|
|
|
|
|
|
} else if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
zoomOut(protyle, actionElement.parentElement.getAttribute("data-node-id"));
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (actionElement.classList.contains("protyle-action--task")) {
|
|
|
|
|
|
const html = actionElement.parentElement.outerHTML;
|
|
|
|
|
|
if (actionElement.parentElement.classList.contains("protyle-task--done")) {
|
|
|
|
|
|
actionElement.querySelector("use").setAttribute("xlink:href", "#iconUncheck");
|
|
|
|
|
|
actionElement.parentElement.classList.remove("protyle-task--done");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
actionElement.querySelector("use").setAttribute("xlink:href", "#iconCheck");
|
|
|
|
|
|
actionElement.parentElement.classList.add("protyle-task--done");
|
|
|
|
|
|
}
|
|
|
|
|
|
actionElement.parentElement.setAttribute("updated", dayjs().format("YYYYMMDDHHmmss"));
|
|
|
|
|
|
updateTransaction(protyle, actionElement.parentElement.getAttribute("data-node-id"), actionElement.parentElement.outerHTML, html);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
protyle.gutter.renderMenu(protyle, actionElement.parentElement);
|
|
|
|
|
|
window.siyuan.menus.menu.element.classList.remove("fn__none");
|
|
|
|
|
|
setPosition(window.siyuan.menus.menu.element, event.clientX - window.siyuan.menus.menu.element.clientWidth - 16, event.clientY - 16);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const selectElement = hasClosestByClassName(event.target, "hr") ||
|
|
|
|
|
|
hasClosestByClassName(event.target, "iframe");
|
|
|
|
|
|
if (!window.siyuan.shiftIsPressed && !window.siyuan.ctrlIsPressed && selectElement) {
|
|
|
|
|
|
selectElement.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const imgElement = hasTopClosestByClassName(event.target, "img");
|
|
|
|
|
|
if (!window.siyuan.shiftIsPressed && !window.siyuan.ctrlIsPressed && imgElement) {
|
|
|
|
|
|
imgElement.classList.add("img--select");
|
|
|
|
|
|
range.setStartAfter(imgElement);
|
|
|
|
|
|
range.collapse(true);
|
|
|
|
|
|
focusByRange(range);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 点击空白
|
|
|
|
|
|
if (event.target.contains(this.element) && this.element.lastElementChild && !protyle.disabled) {
|
|
|
|
|
|
const lastRect = this.element.lastElementChild.getBoundingClientRect();
|
|
|
|
|
|
if (event.y > lastRect.bottom) {
|
|
|
|
|
|
const lastEditElement = getContenteditableElement(getLastBlock(this.element.lastElementChild));
|
|
|
|
|
|
if (!lastEditElement ||
|
|
|
|
|
|
(this.element.lastElementChild.getAttribute("data-type") !== "NodeParagraph" && protyle.wysiwyg.element.getAttribute("data-doc-type") !== "NodeListItem") ||
|
|
|
|
|
|
(this.element.lastElementChild.getAttribute("data-type") === "NodeParagraph" && getContenteditableElement(lastEditElement).textContent !== "")) {
|
|
|
|
|
|
const emptyElement = genEmptyElement(false, false);
|
|
|
|
|
|
this.element.insertAdjacentElement("beforeend", emptyElement);
|
|
|
|
|
|
transaction(protyle, [{
|
|
|
|
|
|
action: "insert",
|
|
|
|
|
|
data: emptyElement.outerHTML,
|
|
|
|
|
|
id: emptyElement.getAttribute("data-node-id"),
|
|
|
|
|
|
previousID: emptyElement.previousElementSibling.getAttribute("data-node-id"),
|
|
|
|
|
|
parentID: protyle.block.parentID
|
|
|
|
|
|
}], [{
|
|
|
|
|
|
action: "delete",
|
|
|
|
|
|
id: emptyElement.getAttribute("data-node-id")
|
|
|
|
|
|
}]);
|
|
|
|
|
|
const emptyEditElement = getContenteditableElement(emptyElement) as HTMLInputElement;
|
|
|
|
|
|
range.selectNodeContents(emptyEditElement);
|
|
|
|
|
|
range.collapse(true);
|
|
|
|
|
|
emptyEditElement.focus();
|
|
|
|
|
|
} else if (lastEditElement) {
|
|
|
|
|
|
range.selectNodeContents(lastEditElement);
|
|
|
|
|
|
range.collapse(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
// 选中后,在选中的文字上点击需等待 range 更新
|
|
|
|
|
|
const newRange = getEditorRange(this.element);
|
|
|
|
|
|
if (newRange.toString().replace(Constants.ZWSP, "") !== "") {
|
|
|
|
|
|
protyle.toolbar.render(protyle, newRange);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
hideElements(["toolbar"], protyle);
|
|
|
|
|
|
}
|
|
|
|
|
|
pushBack(protyle, newRange);
|
|
|
|
|
|
}, isMobile() ? 520 : 0); // Android 双击慢了出不来
|
|
|
|
|
|
|
|
|
|
|
|
protyle.toolbar.isNewEmptyInline = false;
|
|
|
|
|
|
protyle.hint.enableEmoji = false;
|
|
|
|
|
|
if (window.siyuan.shiftIsPressed) {
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
// shift 多选
|
|
|
|
|
|
let startElement = hasClosestBlock(range.startContainer);
|
|
|
|
|
|
let endElement = hasClosestBlock(event.target);
|
|
|
|
|
|
if (startElement && endElement && startElement.isSameNode(endElement)) {
|
|
|
|
|
|
startElement = hasClosestBlock(range.endContainer);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (startElement && endElement && (!startElement.isSameNode(endElement) || (shiftStartElement && shiftStartElement.isSameNode(startElement)))) {
|
|
|
|
|
|
let toDown = true;
|
|
|
|
|
|
range.collapse(true);
|
|
|
|
|
|
if (shiftStartElement) {
|
|
|
|
|
|
startElement = shiftStartElement;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
shiftStartElement = startElement;
|
|
|
|
|
|
}
|
|
|
|
|
|
let startTop = startElement.getBoundingClientRect().top;
|
|
|
|
|
|
let endTop = endElement.getBoundingClientRect().top;
|
|
|
|
|
|
if (startTop > endTop) {
|
|
|
|
|
|
const tempElement = endElement;
|
|
|
|
|
|
endElement = startElement;
|
|
|
|
|
|
startElement = tempElement;
|
|
|
|
|
|
const tempTop = endTop;
|
|
|
|
|
|
endTop = startTop;
|
|
|
|
|
|
startTop = tempTop;
|
|
|
|
|
|
toDown = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
let selectElements: Element[] = [];
|
|
|
|
|
|
let currentElement: HTMLElement = startElement;
|
|
|
|
|
|
let hasJump = false;
|
|
|
|
|
|
while (currentElement) {
|
|
|
|
|
|
if (currentElement && !currentElement.classList.contains("protyle-attr")) {
|
|
|
|
|
|
if (currentElement.getBoundingClientRect().top <= endTop) {
|
|
|
|
|
|
if (hasJump) {
|
|
|
|
|
|
// 父节点的下个节点在选中范围内才可使用父节点作为选中节点
|
|
|
|
|
|
if (currentElement.nextElementSibling && !currentElement.nextElementSibling.classList.contains("protyle-attr")) {
|
|
|
|
|
|
if (currentElement.nextElementSibling.getBoundingClientRect().top <= endTop) {
|
|
|
|
|
|
selectElements = [currentElement];
|
|
|
|
|
|
currentElement = currentElement.nextElementSibling as HTMLElement;
|
|
|
|
|
|
hasJump = false;
|
|
|
|
|
|
} else if (currentElement.parentElement.classList.contains("sb")) {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement) as HTMLElement;
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement) as HTMLElement;
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.push(currentElement);
|
|
|
|
|
|
currentElement = currentElement.nextElementSibling as HTMLElement;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if (currentElement.parentElement.classList.contains("sb")) {
|
|
|
|
|
|
// 跳出超级块横向排版中的未选中元素
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement) as HTMLElement;
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
currentElement = hasClosestBlock(currentElement.parentElement) as HTMLElement;
|
|
|
|
|
|
hasJump = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (selectElements.length === 1 && !selectElements[0].classList.contains("list") && !selectElements[0].classList.contains("bq") && !selectElements[0].classList.contains("sb")) {
|
|
|
|
|
|
// 单个 p 不选中
|
|
|
|
|
|
shiftStartElement = undefined;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
selectElements.forEach(item => {
|
|
|
|
|
|
item.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
});
|
|
|
|
|
|
if (toDown) {
|
|
|
|
|
|
focusBlock(selectElements[selectElements.length - 1], protyle.wysiwyg.element, false);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
focusBlock(selectElements[0], protyle.wysiwyg.element, false);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (this.element.querySelector(".protyle-wysiwyg--select") && range.toString() !== "") {
|
|
|
|
|
|
// 选中块后,文字不能被选中。需在 shift click 之后,防止shift点击单个块出现文字选中
|
|
|
|
|
|
range.collapse(false);
|
|
|
|
|
|
focusByRange(range);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (window.siyuan.ctrlIsPressed) {
|
|
|
|
|
|
let ctrlElement = hasClosestBlock(event.target);
|
|
|
|
|
|
if (ctrlElement) {
|
|
|
|
|
|
ctrlElement = getTopAloneElement(ctrlElement) as HTMLElement;
|
|
|
|
|
|
if (ctrlElement.classList.contains("protyle-wysiwyg--select")) {
|
|
|
|
|
|
ctrlElement.classList.remove("protyle-wysiwyg--select");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
ctrlElement.classList.add("protyle-wysiwyg--select");
|
|
|
|
|
|
}
|
|
|
|
|
|
ctrlElement.querySelectorAll(".protyle-wysiwyg--select").forEach(item => {
|
|
|
|
|
|
item.classList.remove("protyle-wysiwyg--select");
|
|
|
|
|
|
});
|
|
|
|
|
|
const ctrlParentElement = hasClosestByClassName(ctrlElement, "protyle-wysiwyg--select");
|
|
|
|
|
|
if (ctrlParentElement && !ctrlElement.isSameNode(ctrlParentElement)) {
|
|
|
|
|
|
ctrlParentElement.classList.remove("protyle-wysiwyg--select");
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 面包屑定位
|
|
|
|
|
|
protyle.breadcrumb.render(protyle);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|