import {Tab} from "../Tab"; import {Model} from "../Model"; import {Tree} from "../../util/Tree"; import {setPanelFocus} from "../util"; import {getDockByType} from "../tabUtil"; import {fetchPost} from "../../util/fetch"; import {Constants} from "../../constants"; import {updateHotkeyTip} from "../../protyle/util/compatibility"; import {openFileById} from "../../editor/util"; import {Protyle} from "../../protyle"; import {MenuItem} from "../../menus/Menu"; import {App} from "../../index"; import {isSupportCSSHL, searchMarkRender} from "../../protyle/render/searchMarkRender"; export class Backlink extends Model { public element: HTMLElement; public inputsElement: NodeListOf; public type: "pin" | "local"; public blockId: string; public rootId: string; // "local" 必传 public tree: Tree; private notebookId: string; public mTree: Tree; public editors: Protyle[] = []; public status: { [key: string]: { sort: string, mSort: string, scrollTop: number, mScrollTop: number, backlinkOpenIds: string[], backlinkMOpenIds: string[], backlinkMStatus: number // 0 全展开,1 展开一半箭头向下,2 展开一半箭头向上,3 全收起 } } = {}; constructor(options: { app: App, tab: Tab, blockId: string, rootId?: string, type: "pin" | "local" }) { super({ app: options.app, 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.rootId === data.data.id) { this.parent.updateTitle(data.data.title); } break; case "unmount": if (this.notebookId === data.data.box && this.type === "local") { this.parent.parent.removeTab(this.parent.id); } break; case "removeDoc": if (data.data.ids.includes(this.rootId) && this.type === "local") { 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("blur", (event: KeyboardEvent) => { const inputElement = event.target as HTMLInputElement; inputElement.classList.add("fn__none"); const filterIconElement = inputElement.nextElementSibling; if (inputElement.value) { filterIconElement.classList.add("block__icon--active"); filterIconElement.setAttribute("aria-label", window.siyuan.languages.filter + " " + inputElement.value); } else { filterIconElement.classList.remove("block__icon--active"); filterIconElement.setAttribute("aria-label", window.siyuan.languages.filter); } }); 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); this.setFocus(); this.mTree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, ctrlClick: (element) => { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), action: [Constants.CB_GET_CONTEXT] }); this.mTree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, altClick(element) { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), position: "right", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); this.mTree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, shiftClick(element) { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), position: "bottom", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); this.mTree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, toggleClick: (liElement) => { this.toggleItem(liElement, false); this.setFocus(); this.mTree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); } }); this.mTree = new Tree({ element: this.element.querySelector(".backlinkMList") as HTMLElement, data: null, click: (element) => { this.toggleItem(element, true); this.setFocus(); this.tree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, ctrlClick(element) { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), action: [Constants.CB_GET_CONTEXT] }); this.tree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, altClick(element) { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), position: "right", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); this.tree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, shiftClick(element) { openFileById({ app: options.app, id: element.getAttribute("data-node-id"), position: "bottom", action: [Constants.CB_GET_FOCUS, Constants.CB_GET_CONTEXT] }); this.tree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, toggleClick: (liElement) => { this.toggleItem(liElement, true); this.setFocus(); this.tree.element.querySelector(".b3-list-item--focus")?.classList.remove("b3-list-item--focus"); }, 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.querySelector('[data-type="expand"]').addEventListener("click", () => { Array.from(this.tree.element.firstElementChild.children).forEach((item: HTMLElement) => { if (item.tagName === "LI" && !item.querySelector(".b3-list-item__arrow--open")) { this.toggleItem(item, false); } }); }); this.element.addEventListener("click", (event) => { this.setFocus(); let target = event.target as HTMLElement; while (target && !target.isEqualNode(this.element)) { if (target.classList.contains("block__icon") && target.parentElement.parentElement.isSameNode(this.element)) { const type = target.getAttribute("data-type"); switch (type) { case "refresh": this.refresh(); break; case "mExpand": Array.from(this.mTree.element.firstElementChild.children).forEach((item: HTMLElement) => { if (item.tagName === "LI" && !item.querySelector(".b3-list-item__arrow--open")) { this.toggleItem(item, true); } }); 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 "search": target.previousElementSibling.classList.remove("fn__none"); (target.previousElementSibling as HTMLInputElement).select(); break; case "sort": case "mSort": this.showSortMenu(type, target.getAttribute("data-sort")); window.siyuan.menus.menu.popup({x: event.clientX, y: event.clientY}); event.stopPropagation(); 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(true); } private setFocus() { if (this.type === "local") { setPanelFocus(this.element.parentElement.parentElement); } else { setPanelFocus(this.element); } } private showSortMenu(type: string, sort: string) { const clickEvent = (currentSort: string) => { (type === "sort" ? this.tree : this.mTree).element.previousElementSibling.querySelector(`[data-type="${type}"]`).setAttribute("data-sort", currentSort); this.searchBacklinks(); }; window.siyuan.menus.menu.remove(); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "0" ? "iconSelect" : undefined, label: window.siyuan.languages.fileNameASC, click: () => { clickEvent("0"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "1" ? "iconSelect" : undefined, label: window.siyuan.languages.fileNameDESC, click: () => { clickEvent("1"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "4" ? "iconSelect" : undefined, label: window.siyuan.languages.fileNameNatASC, click: () => { clickEvent("4"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "5" ? "iconSelect" : undefined, label: window.siyuan.languages.fileNameNatDESC, click: () => { clickEvent("5"); } }).element); window.siyuan.menus.menu.append(new MenuItem({type: "separator"}).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "9" ? "iconSelect" : undefined, label: window.siyuan.languages.createdASC, click: () => { clickEvent("9"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "10" ? "iconSelect" : undefined, label: window.siyuan.languages.createdDESC, click: () => { clickEvent("10"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "2" ? "iconSelect" : undefined, label: window.siyuan.languages.modifiedASC, click: () => { clickEvent("2"); } }).element); window.siyuan.menus.menu.append(new MenuItem({ icon: sort === "3" ? "iconSelect" : undefined, label: window.siyuan.languages.modifiedDESC, click: () => { clickEvent("3"); } }).element); } private toggleItem(liElement: HTMLElement, isMention: boolean) { const svgElement = liElement.firstElementChild?.firstElementChild; if (!svgElement || svgElement.getAttribute("disabled")) { return; } svgElement.setAttribute("disabled", "disabled"); 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 && liElement.nextElementSibling && item.protyle.element.isSameNode(liElement.nextElementSibling)) { item.destroy(); this.editors.splice(index, 1); liElement.nextElementSibling.remove(); return true; } }); svgElement.removeAttribute("disabled"); } else { const keyword = isMention ? this.inputsElement[1].value : this.inputsElement[0].value; fetchPost(isMention ? "/api/ref/getBackmentionDoc" : "/api/ref/getBacklinkDoc", { defID: this.blockId, refTreeID: docId, highlight: !isSupportCSSHL(), keyword, }, (response) => { svgElement.removeAttribute("disabled"); svgElement.classList.add("b3-list-item__arrow--open"); 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(this.app, editorElement, { blockId: docId, backlinkData: isMention ? response.data.backmentions : response.data.backlinks, render: { background: false, gutter: true, scroll: false, breadcrumb: false, } }); editor.protyle.notebookId = liElement.getAttribute("data-notebook-id"); searchMarkRender(editor.protyle, response.data.keywords); this.editors.push(editor); }); } } public refresh() { const element = this.element.querySelector('.block__icon[data-type="refresh"] svg'); if (!this.blockId || element.classList.contains("fn__rotate")) { return; } element.classList.add("fn__rotate"); fetchPost("/api/ref/refreshBacklink", { id: this.blockId, }, () => { element.classList.remove("fn__rotate"); this.searchBacklinks(); }); } private searchBacklinks(init = false) { const element = this.element.querySelector('.block__icon[data-type="refresh"] svg'); if (element.classList.contains("fn__rotate")) { return; } element.classList.add("fn__rotate"); fetchPost("/api/ref/getBacklink2", { sort: this.tree.element.previousElementSibling.querySelector('[data-type="sort"]').getAttribute("data-sort"), mSort: this.mTree.element.previousElementSibling.querySelector('[data-type="mSort"]').getAttribute("data-sort"), k: this.inputsElement[0].value, mk: this.inputsElement[1].value, id: this.blockId, }, response => { if (!init) { this.saveStatus(); } this.render(response.data); }); } public saveStatus() { this.status[this.blockId] = { sort: this.tree.element.previousElementSibling.querySelector('[data-type="sort"]').getAttribute("data-sort"), mSort: this.mTree.element.previousElementSibling.querySelector('[data-type="mSort"]').getAttribute("data-sort"), 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] = { sort: "3", mSort: "3", scrollTop: 0, mScrollTop: 0, backlinkOpenIds: [], backlinkMOpenIds: [], backlinkMStatus: 3 }; if (data.mentionsCount === 0 || window.siyuan.config.editor.backmentionExpandCount === -1) { this.status[this.blockId].backlinkMStatus = 3; } else { Array.from({length: window.siyuan.config.editor.backmentionExpandCount}).forEach((item, index) => { if (data.backmentions[index]) { this.status[this.blockId].backlinkMOpenIds.push(data.backmentions[index].id); } }); if (data.mentionsCount === 0) { this.status[this.blockId].backlinkMStatus = 3; } else { if (data.linkRefsCount === 0) { this.status[this.blockId].backlinkMStatus = 0; } else { this.status[this.blockId].backlinkMStatus = 1; } } } if (data.linkRefsCount > 0) { Array.from({length: window.siyuan.config.editor.backlinkExpandCount}).forEach((item, index) => { if (data.backlinks[index]) { this.status[this.blockId].backlinkOpenIds.push(data.backlinks[index].id); } }); } } // 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"); } this.tree.element.previousElementSibling.querySelector('[data-type="sort"]').setAttribute("data-sort", this.status[this.blockId].sort); this.mTree.element.previousElementSibling.querySelector('[data-type="mSort"]').setAttribute("data-sort", this.status[this.blockId].mSort); setTimeout(() => { this.tree.element.scrollTop = this.status[this.blockId].scrollTop; this.mTree.element.scrollTop = this.status[this.blockId].mScrollTop; }, Constants.TIMEOUT_LOAD); } }