import {Tab} from "../Tab"; import {Model} from "../Model"; import {getDisplayName} from "../../util/pathName"; import {Tree} from "../../util/Tree"; import {getDockByType, setPanelFocus} from "../util"; import {fetchPost} from "../../util/fetch"; import {Constants} from "../../constants"; import {updateHotkeyTip} from "../../protyle/util/compatibility"; import {openFileById} from "../../editor/util"; import {Protyle} from "../../protyle"; export class Backlink extends Model { public element: HTMLElement; public inputsElement: NodeListOf; public type: "pin" | "local"; public blockId: string; public rootId: string; // "local" 必传 private tree: Tree; private notebookId: string; private mTree: Tree; private editors: Protyle[] = []; private status: { [key: string]: { scrollTop: number, mScrollTop: number, backlinkOpenIds: string[], backlinkMOpenIds: string[], backlinkMStatus: number // 0 全展开,1 展开一半箭头向下,2 展开一半箭头向上,3 全收起 } } = {}; constructor(options: { tab: Tab, blockId: string, rootId?: string, type: "pin" | "local" }) { super({ id: options.tab.id, callback() { if (this.type === "local") { fetchPost("/api/block/checkBlockExist", {id: this.blockId}, existResponse => { if (!existResponse.data) { this.parent.parent.removeTab(this.parent.id); } }); } }, msgCallback(data) { if (data && this.type === "local") { switch (data.cmd) { case "rename": if (this.blockId === data.data.id) { this.parent.updateTitle(data.data.title); } break; case "unmount": if (this.notebookId === data.data.box) { this.parent.parent.removeTab(this.parent.id); } break; case "remove": if (this.path?.indexOf(getDisplayName(data.data.path, false, true)) === 0) { this.parent.parent.removeTab(this.parent.id); } break; } } } }); this.blockId = options.blockId; this.rootId = options.rootId; this.type = options.type; this.element = options.tab.panelElement; this.element.classList.add("fn__flex-column", "file-tree", "sy__backlink"); this.element.innerHTML = `
`; this.inputsElement = this.element.querySelectorAll("input"); this.inputsElement.forEach((item) => { item.addEventListener("keydown", (event: KeyboardEvent) => { if (!event.isComposing && event.key === "Enter") { this.searchBacklinks(); } }); item.addEventListener("input", (event: KeyboardEvent) => { const inputElement = event.target as HTMLInputElement; if (inputElement.value === "") { inputElement.classList.remove("search__input--block"); } else { inputElement.classList.add("search__input--block"); } }); }); this.tree = new Tree({ element: this.element.querySelector(".backlinkList") as HTMLElement, data: null, click: (element) => { this.toggleItem(element, false); }, ctrlClick(element) { openFileById({ id: element.getAttribute("data-node-id"), keepCursor: true, action: [Constants.CB_GET_CONTEXT] }); }, altClick(element) { openFileById({ id: element.getAttribute("data-node-id"), position: "right", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); }, shiftClick(element) { openFileById({ id: element.getAttribute("data-node-id"), position: "bottom", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); }, toggleClick: (liElement) => { this.toggleItem(liElement, false); } }); this.mTree = new Tree({ element: this.element.querySelector(".backlinkMList") as HTMLElement, data: null, click: (element) => { this.toggleItem(element, true); }, ctrlClick(element) { openFileById({ id: element.getAttribute("data-node-id"), keepCursor: true, action: [Constants.CB_GET_CONTEXT] }); }, altClick(element) { openFileById({ id: element.getAttribute("data-node-id"), position: "right", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); }, shiftClick(element) { openFileById({ id: element.getAttribute("data-node-id"), position: "bottom", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); }, toggleClick: (liElement) => { this.toggleItem(liElement, true); }, blockExtHTML: `` }); this.tree.element.addEventListener("scroll", () => { this.tree.element.querySelectorAll(".protyle-gutters").forEach(item => { item.classList.add("fn__none"); item.innerHTML = ""; }); this.tree.element.querySelectorAll(".protyle-wysiwyg--hl").forEach((hlItem) => { hlItem.classList.remove("protyle-wysiwyg--hl"); }); }); this.mTree.element.addEventListener("scroll", () => { this.mTree.element.querySelectorAll(".protyle-gutters").forEach(item => { item.classList.add("fn__none"); item.innerHTML = ""; }); this.mTree.element.querySelectorAll(".protyle-wysiwyg--hl").forEach((hlItem) => { hlItem.classList.remove("protyle-wysiwyg--hl"); }); }); // 为了快捷键的 dispatch this.element.querySelector('[data-type="collapse"]').addEventListener("click", () => { this.tree.element.querySelectorAll(".protyle").forEach(item => { item.classList.add("fn__none"); }); this.tree.element.querySelectorAll(".b3-list-item__arrow").forEach(item => { item.classList.remove("b3-list-item__arrow--open"); }); }); this.element.addEventListener("click", (event) => { if (this.type === "local") { setPanelFocus(this.element.parentElement.parentElement); } else { setPanelFocus(this.element.firstElementChild); } let target = event.target as HTMLElement; while (target && !target.isEqualNode(this.element)) { if (target.classList.contains("block__icon")) { const type = target.getAttribute("data-type"); switch (type) { case "refresh": this.refresh(); break; case "mCollapse": this.mTree.element.querySelectorAll(".protyle").forEach(item => { item.classList.add("fn__none"); }); this.mTree.element.querySelectorAll(".b3-list-item__arrow").forEach(item => { item.classList.remove("b3-list-item__arrow--open"); }); break; case "min": getDockByType("backlink").toggleModel("backlink"); break; case "layout": if (this.mTree.element.style.flex) { if (this.mTree.element.style.height === "0px") { this.tree.element.classList.remove("fn__none"); this.mTree.element.removeAttribute("style"); target.setAttribute("aria-label", window.siyuan.languages.up); target.querySelector("use").setAttribute("xlink:href", "#iconUp"); } else { this.tree.element.classList.remove("fn__none"); this.mTree.element.removeAttribute("style"); target.setAttribute("aria-label", window.siyuan.languages.down); target.querySelector("use").setAttribute("xlink:href", "#iconDown"); } } else { if (target.getAttribute("aria-label") === window.siyuan.languages.down) { this.tree.element.classList.remove("fn__none"); this.mTree.element.setAttribute("style", "flex:none;height:0px"); target.setAttribute("aria-label", window.siyuan.languages.up); target.querySelector("use").setAttribute("xlink:href", "#iconUp"); } else { this.tree.element.classList.add("fn__none"); this.mTree.element.setAttribute("style", `flex:none;height:${this.element.clientHeight - this.tree.element.previousElementSibling.clientHeight * 2}px`); target.setAttribute("aria-label", window.siyuan.languages.down); target.querySelector("use").setAttribute("xlink:href", "#iconDown"); } } this.tree.element.dispatchEvent(new CustomEvent("scroll")); this.mTree.element.dispatchEvent(new CustomEvent("scroll")); break; } } target = target.parentElement; } }); this.searchBacklinks(); if (this.type === "pin") { setPanelFocus(this.element.firstElementChild); } } private toggleItem(liElement: HTMLElement, isMention: boolean) { const svgElement = liElement.firstElementChild.firstElementChild; const docId = liElement.getAttribute("data-node-id"); if (svgElement.classList.contains("b3-list-item__arrow--open")) { svgElement.classList.remove("b3-list-item__arrow--open"); this.editors.find((item, index) => { if (item.protyle.block.rootID === docId) { item.destroy(); this.editors.splice(index, 1); return true; } }); liElement.nextElementSibling?.remove(); } else { svgElement.classList.add("b3-list-item__arrow--open"); fetchPost(isMention ? "/api/ref/getBackmentionDoc" : "/api/ref/getBacklinkDoc", { defID: this.blockId, refTreeID: docId }, (response) => { const editorElement = document.createElement("div"); editorElement.style.minHeight = "auto"; editorElement.setAttribute("data-defid", this.blockId); editorElement.setAttribute("data-ismention", isMention ? "true" : "false"); liElement.after(editorElement); const editor = new Protyle(editorElement, { blockId: "", backlinkData: isMention ? response.data.backmentions : response.data.backlinks, render: { background: false, title: false, gutter: true, scroll: false, breadcrumb: false, } }); editor.protyle.block.rootID = docId; this.editors.push(editor); }); } } private refresh() { fetchPost("/api/ref/refreshBacklink", { id: this.blockId, }, () => { this.searchBacklinks(); }); } private searchBacklinks(reload = false) { const element = this.element.querySelector('.block__icon[data-type="refresh"] svg'); if (element.classList.contains("fn__rotate") && !reload) { return; } element.classList.add("fn__rotate"); fetchPost("/api/ref/getBacklink2", { k: this.inputsElement[0].value, mk: this.inputsElement[1].value, id: this.blockId, }, response => { this.saveStatus(); this.render(response.data); }); } public saveStatus() { this.status[this.blockId] = { scrollTop: this.tree.element.scrollTop, mScrollTop: this.mTree.element.scrollTop, backlinkOpenIds: [], backlinkMOpenIds: [], backlinkMStatus: 3 // 0 全展开,1 展开一半箭头向下,2 展开一半箭头向上,3 全收起 } this.tree.element.querySelectorAll(".b3-list-item__arrow--open").forEach(item => { this.status[this.blockId].backlinkOpenIds.push(item.parentElement.parentElement.getAttribute("data-node-id")) }) this.mTree.element.querySelectorAll(".b3-list-item__arrow--open").forEach(item => { this.status[this.blockId].backlinkMOpenIds.push(item.parentElement.parentElement.getAttribute("data-node-id")) }) if (this.mTree.element.style.flex) { if (this.mTree.element.style.height === "0px") { this.status[this.blockId].backlinkMStatus = 3; } else { this.status[this.blockId].backlinkMStatus = 0; } } else { if (this.mTree.element.previousElementSibling.querySelector('[data-type="layout"]').getAttribute("aria-label") === window.siyuan.languages.down) { this.status[this.blockId].backlinkMStatus = 1; } else { this.status[this.blockId].backlinkMStatus = 2; } } } public render(data: { box: string, backlinks: IBlockTree[], backmentions: IBlockTree[], linkRefsCount: number, mentionsCount: number, k: string, mk: string }) { if (!data) { data = { box: "", backlinks: [], backmentions: [], linkRefsCount: 0, mentionsCount: 0, k: "", mk: "" }; } this.editors.forEach(item => { item.destroy(); }); this.editors = []; this.element.querySelector('.block__icon[data-type="refresh"] svg').classList.remove("fn__rotate"); this.notebookId = data.box; this.inputsElement[0].value = data.k; this.inputsElement[1].value = data.mk; this.tree.updateData(data.backlinks); this.mTree.updateData(data.backmentions); const countElement = this.element.querySelector(".listCount"); if (data.linkRefsCount === 0) { countElement.classList.add("fn__none"); } else { countElement.classList.remove("fn__none"); countElement.textContent = data.linkRefsCount.toString(); } const mCountElement = this.element.querySelector(".listMCount"); if (data.mentionsCount === 0) { mCountElement.classList.add("fn__none"); } else { mCountElement.classList.remove("fn__none"); mCountElement.textContent = data.mentionsCount.toString(); } if (!this.status[this.blockId]) { this.status[this.blockId] = { scrollTop: 0, mScrollTop: 0, backlinkOpenIds: [], backlinkMOpenIds: [], backlinkMStatus: 3 } if (data.mentionsCount === 0) { this.status[this.blockId].backlinkMStatus = 3; } else { this.status[this.blockId].backlinkMOpenIds = [data.backmentions[0].id] if (data.linkRefsCount === 0) { this.status[this.blockId].backlinkMStatus = 0; } else { this.status[this.blockId].backlinkOpenIds = [data.backlinks[0].id] this.status[this.blockId].backlinkMStatus = 1; } } } // restore status this.status[this.blockId].backlinkOpenIds.forEach(item => { const liElement = this.tree.element.querySelector(`.b3-list-item[data-node-id="${item}"]`) as HTMLElement if (liElement) { this.toggleItem(liElement, false); } }) this.status[this.blockId].backlinkMOpenIds.forEach(item => { const liElement = this.mTree.element.querySelector(`.b3-list-item[data-node-id="${item}"]`) as HTMLElement if (liElement) { this.toggleItem(liElement, true); } }) // 0 全展开,1 展开一半箭头向下,2 展开一半箭头向上,3 全收起 const layoutElement = this.mTree.element.previousElementSibling.querySelector('[data-type="layout"]'); if (this.status[this.blockId].backlinkMStatus === 2 || this.status[this.blockId].backlinkMStatus === 1) { this.tree.element.classList.remove("fn__none"); this.mTree.element.removeAttribute("style"); if (this.status[this.blockId].backlinkMStatus === 1) { layoutElement.setAttribute("aria-label", window.siyuan.languages.down); layoutElement.querySelector("use").setAttribute("xlink:href", "#iconDown"); } else { layoutElement.setAttribute("aria-label", window.siyuan.languages.up); layoutElement.querySelector("use").setAttribute("xlink:href", "#iconUp"); } }else if (this.status[this.blockId].backlinkMStatus === 3) { this.tree.element.classList.remove("fn__none"); this.mTree.element.setAttribute("style", "flex:none;height:0px"); layoutElement.setAttribute("aria-label", window.siyuan.languages.up); layoutElement.querySelector("use").setAttribute("xlink:href", "#iconUp"); } else { this.tree.element.classList.add("fn__none"); this.mTree.element.setAttribute("style", `flex:none;height:${this.element.clientHeight - this.tree.element.previousElementSibling.clientHeight * 2}px`); layoutElement.setAttribute("aria-label", window.siyuan.languages.down); layoutElement.querySelector("use").setAttribute("xlink:href", "#iconDown"); } setTimeout(() => { this.tree.element.scrollTop = this.status[this.blockId].scrollTop; this.mTree.element.scrollTop = this.status[this.blockId].mScrollTop; }, Constants.TIMEOUT_BLOCKLOAD); } }