Widget:可视化推荐阅读编辑器
<script src="https://cdn.jsdelivr.net/npm/@logicflow/core/dist/index.min.js"></script>
<link href="https://cdn.jsdelivr.net/npm/@logicflow/core/lib/style/index.min.css" rel="stylesheet">
<script src="https://cdn.jsdelivr.net/npm/@logicflow/extension/dist/index.min.js"></script> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@logicflow/extension/lib/style/index.min.css" />
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
<body>
</body>
<script>
const { HtmlResize, dynamicGroup, DndPanel, Menu, Control, MiniMap, DynamicGroup, SelectionSelect } = Extension; const { LogicFlow, h, BaseText } = Core;
// 暂定五种节点:主线、世界深处、约会剧情、活动剧情、手机信息 // 定义主线节点 class CommonModel extends HtmlResize.model { initNodeData(data) { super.initNodeData(data); this.width = 200; this.height = 50; } getTextStyle() { const style = super.getTextStyle(); style.color = this.properties.textColor || '#aa7952'; style.fontSize = 15 return style; } } class MainLineNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; // node_md.properties 获取节点属性 let el = document.createElement("div"); el.className = "main-line-node node-div"; el.id = node_md.id; // html 节点绑定节点唯一ID;即可通过id 获取对应dom元素 并进行相关业务操作 el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const MainLineBox = { type: "main_line_node", view: MainLineNode, model: CommonModel, };
// 定义世界深处节点 class WorldDeepNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "world-deep-node node-div"; el.id = node_md.id; // html 节点绑定节点唯一ID;即可通过id 获取对应dom元素 并进行相关业务操作 el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const WorldDeepBox = { type: "world_deep_node", view: WorldDeepNode, model: CommonModel, };
// 定义约会剧情节点 class DatePlotNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "date-plot-node node-div"; el.id = node_md.id; // html 节点绑定节点唯一ID;即可通过id 获取对应dom元素 并进行相关业务操作 el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const DatePlotBox = { type: "date_plot_node", view: DatePlotNode, model: CommonModel, };
// 定义活动剧情节点 class ActivityPlotNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "activity-plot-node node-div"; el.id = node_md.id; // html 节点绑定节点唯一ID;即可通过id 获取对应dom元素 并进行相关业务操作 el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const ActivityPlotBox = { type: "activity_plot_node", view: ActivityPlotNode, model: CommonModel, };
// 定义手机信息节点 class PhoneInfoNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "phone-info-node node-div"; el.id = node_md.id; // html 节点绑定节点唯一ID;即可通过id 获取对应dom元素 并进行相关业务操作 el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const PhoneInfoBox = { type: "phone_info_node", view: PhoneInfoNode, model: CommonModel, };
// 定义分组节点 class GroupModel extends dynamicGroup.model { initNodeData(data) { super.initNodeData(data); // 设置默认样式 this.properties = { ...this.properties, collapsible: false, width: 300, height: 450 } this.collapsible = false; } getNodeStyle() { const style = super.getNodeStyle(); style.fill = "transparent"; style.strokeWidth = 0; return style; } // 处理高亮样式 getAddableOutlineStyle() { const style = super.getAddableOutlineStyle(); style.stroke = "#FF9800"; style.strokeDasharray = "5 5"; return style; } // 节点内文本样式 // getTextStyle() { // const style = super.getTextStyle(); // style.fontSize = 20; // style.color = this.properties.textColor || '#aa7952'; // 修改文字颜色 // return style; // } } class GroupView extends dynamicGroup.view { // 由于分组节点没有内置自定义html方法,因此重写getShape方法,加入自定义div元素 getShape() { const { model } = this.props const { width, height, x, y, radius } = model
const rectSvg = super.getShape(); return h('g', {}, [ rectSvg, h('g', {}, [ // 写入自定义内容-用于设置分组节点样式、背景图片 h('foreignObject', { x: x - width / 2, y: y - height / 2, width: width, height: height, radius: radius, }, [ // 注意:foreignObject内的HTML元素需要指定xhtml命名空间 h('div', { xmlns: 'http://www.w3.org/2000/xhtml', class: 'group-node' }) ]) ]) ]); }
// getText() { // const { model, graphModel } = this.props // const { editConfigModel } = graphModel
// // 当 节点文本模式非 TEXT 时,不显示文本 // if (editConfigModel.nodeTextMode !== 'text') return null // // 文本被编辑的时候,显示编辑框,不显示文本。 // if (model.state === 'TEXT_EDIT') return null
// if (model.text) { // let draggable = false // if (editConfigModel.nodeTextDraggable && model.text.draggable) { // draggable = true // } // return h('BaseText', { // editable: editConfigModel.nodeTextEdit && (model.text.editable ?? true), // model: model, // graphModel: graphModel, // draggable: draggable, // }) // // ( // // < // // editable={ // // editConfigModel.nodeTextEdit && (model.text.editable ?? true) // // } // // model={model} // // graphModel={graphModel} // // draggable={draggable} // // /> // // ) // } // return null // } } const GroupBox = { type: "group_node", model: GroupModel, view: GroupView, }
// 定义主标题 class MainTitleModel extends HtmlResize.model { initNodeData(data) { super.initNodeData(data); this.width = 200; this.height = 50; } getTextStyle() { const style = super.getTextStyle(); style.fontSize = 24; style.color = this.properties.textColor || '#aa7952'; return style; } } class MainTitleNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "main-title-node title-node"; el.id = node_md.id; el.setAttribute("data-text", node_md.text.value); el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const MainTitleBox = { type: "main_title", model: MainTitleModel, view: MainTitleNode, }
// 定义副标题 class SubTitleModel extends HtmlResize.model { initNodeData(data) { super.initNodeData(data); this.width = 200; this.height = 50; } getTextStyle() { const style = super.getTextStyle(); style.fontSize = 20; style.color = this.properties.textColor || '#aa7952'; // style.overflowMode = 'autoWrap' return style; } } class SubTitleNode extends HtmlResize.view { setHtml(rootEl) { let node_md = this.props.model; let el = document.createElement("div"); el.className = "sub-title-node title-node"; el.id = node_md.id; el.setAttribute("data-text", node_md.text.value); el.innerHTML = ""; // 需要先把之前渲染的子节点清除掉。 rootEl.innerHTML = ""; rootEl.appendChild(el); } } const SubTitleBox = { type: "sub_title", model: SubTitleModel, view: SubTitleNode, }
// 定义拓展阅读节点 class ExtendedReadModel extends dynamicGroup.model { initNodeData(data) { super.initNodeData(data); // 设置默认样式 this.properties = { ...this.properties, collapsible: false, width: 250, height: 150 } this.collapsible = false; } getNodeStyle() { const style = super.getNodeStyle(); style.fill = "transparent"; style.strokeWidth = 0; return style; } // 处理高亮样式 getAddableOutlineStyle() { const style = super.getAddableOutlineStyle(); style.stroke = "#FF9800"; style.strokeDasharray = "5 5"; return style; } } class ExtendedReadNode extends dynamicGroup.view { // 由于分组节点没有内置自定义html方法,因此重写getShape方法,加入自定义div元素 getShape() { const { model } = this.props const { width, height, x, y, radius } = model
const rectSvg = super.getShape(); return h('g', {}, [ rectSvg, h('g', {}, [ // 写入自定义内容-用于设置分组节点样式、背景图片 h('foreignObject', { x: x - width / 2, y: y - height / 2, width: width, height: height, radius: radius, }, [ // 注意:foreignObject内的HTML元素需要指定xhtml命名空间 h('div', { xmlns: 'http://www.w3.org/2000/xhtml', class: 'extended-read-node' }) ]) ]) ]); } } const ExtendedReadBox = { type: "extended_read_node", view: ExtendedReadNode, model: ExtendedReadModel, };
// 定义hover面板 class HoverNode { static pluginName = 'hoverNode'
constructor({ lf, LogicFlow }) { this.lf = lf; this.allowHover = false; // 默认关闭 }
render(lf, container) { this.container = container; lf.on("node:mouseenter", ({ data }) => { if (!this.allowHover) return; if (data.properties.recommond || data.properties.pageUrl) { // 创建一个div节点,将recommond内容渲染为html let panelDiv = document.createElement("div"); panelDiv.id = "node-page-panel"; // 计算悬浮框位置 panelDiv.style.left = this.lf.getTransform().SCALE_X * (data.x + data.properties.width / 2) + this.lf.getTransform().TRANSLATE_X + "px"; panelDiv.style.top = this.lf.getTransform().SCALE_Y * (data.y - data.properties.height / 2) + this.lf.getTransform().TRANSLATE_Y + "px"; panelDiv.innerHTML = `
<button class="node-button" ${data.properties.pageUrl ? : 'disabled'} style="${data.properties.pageUrl ? 'cursor: pointer; color: #aa7952;' : 'color: #aa79524d;'}" onclick="window.open('${data.properties.pageUrl}')">跳转</button> `; this.container.appendChild(panelDiv); panelDiv.addEventListener("mouseleave", () => { this.container.removeChild(panelDiv); }); let panelButton = $('.node-button')[0]; panelButton.addEventListener("mousedown", () => { panelButton.style.backgroundImage = "url(https://patchwiki.biligame.com/images/lysk/b/ba/8wn6pkomoljy6s809xc50g5gkt6aq6l.png)"; }); panelButton.addEventListener("mouseup", () => { panelButton.style.backgroundImage = "url(https://patchwiki.biligame.com/images/lysk/7/7f/8cfam9dgpmzkz2om9j2bjyyqeu9q9sf.png)"; }); panelButton.addEventListener("mouseleave", () => { panelButton.style.backgroundImage = "url(https://patchwiki.biligame.com/images/lysk/7/7f/8cfam9dgpmzkz2om9j2bjyyqeu9q9sf.png)"; }); }
}); lf.on("node:mouseleave", ({ data, e }) => { if (!this.allowHover) return; if (data.type == 'group_node') return; setTimeout(() => { let mouseX = e.offsetX; let mouseY = e.offsetY; let panelDiv = document.querySelector("#node-page-panel"); if (panelDiv) { // 查看鼠标是否在#node-page-panel节点上 if (mouseX >= panelDiv.offsetLeft && mouseX <= panelDiv.offsetLeft + panelDiv.offsetWidth && mouseY >= panelDiv.offsetTop && mouseY <= panelDiv.offsetTop + panelDiv.offsetHeight) { return; } else { this.container.removeChild(panelDiv); } } }, 100); }); } // 开启hover面板 openHoverPanel() { this.allowHover = true; } // 关闭hover面板 closeHoverPanel() { this.allowHover = false; } }
// TODO 虚线样式 // class SequenceModel extends PolylineEdgeModel { // // 设置边样式 // getEdgeStyle() { // const style = super.getEdgeStyle(); // const { properties } = this; // if (properties.isstrokeDashed) { // style.strokeDasharray = '4, 4'; // } // return style; // } // }
// export default { // type: 'sequence', // view: PolylineEdge, // model: SequenceModel, // };
// TODO 修改节点文本颜色 // graphModel.getNodeModelById(currentNodeId).setProperties({ textColor: '#fff' });
const lf = new LogicFlow({ container: document.querySelector("#container"), // grid: true, // edgeType: 'bezier', allowResize: true, disabledTools: ['multipleSelect'], // multipleSelectKey:"shift", keyboard: { enabled: true, }, background: { backgroundImage: "url(https://patchwiki.biligame.com/images/lysk/4/40/0ibuxl73pgr095hvneel8bf1kwxp6r7.jpg)", backgroundRepeat: "repeat", }, idGenerator: (type) => { //重写全局id let edgeTypes = ['line', 'polyline', 'bezier']; // 重写node节点id if (!edgeTypes.includes(type)) { let id_num = graphModel.nodes.length + 1; for (let i = 0; i < graphModel.nodes.length; i++) { if (id_num == graphModel.nodes[i].id.split('_')[1]) { id_num++; i--; } } return 'node_' + id_num; } }, plugins: [DndPanel, Menu, Control, DynamicGroup, SelectionSelect, HoverNode], //MiniMap pluginsOptions: { miniMap: { isShowCloseIcon: true, }, }, });
//注册自定义节点 lf.batchRegister([MainLineBox, WorldDeepBox, DatePlotBox, ActivityPlotBox, PhoneInfoBox, GroupBox, MainTitleBox, SubTitleBox, ExtendedReadBox]);//
lf.extension.dndPanel.setPatternItems([ { type: 'main_line_node', text: '主线节点', label: '主线节点', properties: {}, icon: '' }, { type: 'world_deep_node', text: '世界深处', label: '世界深处', properties: {}, icon: '' }, { type: 'date_plot_node', text: '倾心之约', label: '倾心之约', properties: {}, icon: '' }, { type: 'activity_plot_node', text: '活动剧情', label: '活动剧情', properties: {}, icon: '' }, { type: 'phone_info_node', text: '手机信息', label: '手机信息', properties: {}, icon: '' }, { type: 'group_node', label: '分组', text: , icon: '', }, { type: 'main_title', label: '主标题', text: '主标题', icon: '', }, { type: 'sub_title', label: '副标题', text: '副标题', icon: '', }, { type: 'extended_read_node', label: '拓展阅读', text: , icon: '', }, ])
lf.extension.menu.setMenuConfig({ nodeMenu: [ { text: "删除", callback(node) { lf.deleteNode(node.id); }, }, ], // 覆盖默认的节点右键菜单 });
lf.extension.control.removeItem('zoom-out'); lf.extension.control.removeItem('zoom-in'); lf.extension.control.removeItem('reset'); lf.extension.control.addItem({ // 小地图导航 key: 'mini-map', iconClass: "custom-minimap", title: "", text: "导航", onClick: (lf, ev) => { lf.extension.miniMap.show(); lf.extension.miniMap.updatePosition('right-top'); }, }); lf.extension.control.addItem({ //导入json iconClass: 'input-json', title: "", text: "导入json", onClick: (lf, ev) => { $('#input-json')[0].style.display = ; $('#model-json-input')[0].value = ; }, }); lf.extension.control.addItem({ //打印json iconClass: 'log-json', title: "", text: "导出json", onClick: (lf, ev) => { const modelData = graphModel.modelToGraphData(); //图形->数据 $('#show-json')[0].style.display = ; $('.json-content')[0].innerText = JSON.stringify(modelData); }, }); lf.extension.control.addItem({ //保存 iconClass: 'save-json', title: "", text: "保存", onClick: (lf, ev) => { const request = indexedDB.open('readDatabase'); request.onerror = function (event) { console.error('Database error: ', event.target.errorCode); }; request.onsuccess = function (event) { const db = event.target.result; const newData = { id: 'readData', data: JSON.stringify(graphModel.modelToGraphData()) }; addData(db, newData); }; }, }); lf.extension.control.addItem({ // 开启框选 iconClass: "selection-open", title: "", text: "开启框选", onClick: (lf, ev) => { // 开启框选 graphModel.editConfigModel.updateEditConfig({ stopMoveGraph: true, }) lf.openSelectionSelect(); showCustomAlert('框选已开启!'); }, }); lf.extension.control.addItem({ // 关闭框选 iconClass: "selection-close", title: "", text: "关闭框选", onClick: (lf, ev) => { // 开启框选 graphModel.editConfigModel.updateEditConfig({ stopMoveGraph: false, }) lf.closeSelectionSelect(); showCustomAlert('框选已关闭!'); }, }); lf.extension.control.addItem({ // 开启悬浮框 iconClass: "hoverpanel-open", title: "", text: "开启悬浮框", onClick: (lf, ev) => { lf.extension.hoverNode.openHoverPanel(); showCustomAlert('节点悬浮框已开启!'); }, }); lf.extension.control.addItem({ // 关闭悬浮框 iconClass: "hoverpanel-close", title: "", text: "关闭悬浮框", onClick: (lf, ev) => { lf.extension.hoverNode.closeHoverPanel(); showCustomAlert('节点悬浮框已关闭!'); }, });
// 消息提示框 function showCustomAlert(message) { $('#buttons-tip')[0].innerText = message; $('#buttons-tip')[0].style.display = ; setTimeout(() => { $('#buttons-tip')[0].style.display = 'none'; }, 1000); }
// 复制json到剪贴板 $('.copy-json')[0].addEventListener('click', (event) => { async function copyToClipboard(text) { try { await navigator.clipboard.writeText(text); alert('已复制到剪贴板!'); } catch (err) { console.error('Failed to copy text: ', err); } } copyToClipboard($('.json-content')[0].innerText); }) // 关闭json弹窗 $('.close-json')[0].addEventListener('click', (event) => { $('#show-json')[0].style.display = 'none'; }) // 确认导入 $('.confirm-input')[0].addEventListener('click', (event) => { graphData = JSON.parse($('#model-json-input')[0].value); lf.render(graphData); $('#input-json')[0].style.display = 'none'; console.log('流程数据:', graphData) }) $('.cancle-input')[0].addEventListener('click', (event) => { $('#input-json')[0].style.display = 'none'; $('#model-json-input')[0].value = ; })
// 输入节点属性json $('#node-recommond-input')[0].addEventListener('input', (event) => { graphModel.getNodeModelById(currentNodeId).setProperties({ recommond: event.target.value }); }) $('#node-url-input')[0].addEventListener('input', (event) => { graphModel.getNodeModelById(currentNodeId).setProperties({ pageUrl: event.target.value }); }) // 关闭属性栏 $('.close-panel')[0].addEventListener('click', (event) => { $('.right-panel')[0].style.display = 'none'; })
let graphData, graphModel; const types = ['main_line_node', 'world_deep_node', 'date_plot_node', 'activity_plot_node', 'phone_info_node']
// 从indexedDB中读取数据 const loadDatafromDB = () => { const request = indexedDB.open('readDatabase'); request.onerror = function (event) { console.error('Database error: ', event.target.errorCode); }; request.onsuccess = function (event) { const db = request.result; const transaction = db.transaction(['readDataStore'], 'readonly'); const store = transaction.objectStore('readDataStore'); const result = store.get('readData'); result.onsuccess = function (event) { const data = result.result; if (data) { graphData = JSON.parse(data.data); } else { graphData = {}; } console.log('流程数据:', graphData); lf.render(graphData); //渲染 graphModel = lf.graphModel; }; result.onerror = function (event) { console.error('Error retrieving data.', event.target.errorCode); }; } request.onupgradeneeded = function (event) { let db = event.target.result; if (!db.objectStoreNames.contains("readDataStore")) { db.createObjectStore("readDataStore", { keyPath: "id" }); } }; } loadDatafromDB();
// 向indexedDB中写入数据readData function addData(db, data) { const transaction = db.transaction(['readDataStore'], 'readwrite'); const store = transaction.objectStore('readDataStore'); const request = store.put(data); request.onsuccess = function (event) { alert('数据缓存成功!'); }; request.onerror = function (event) { console.error('Error adding data.', event.target.errorCode); }; }
// 节点点击事件 var currentNodeId = ; lf.on("node:click", (node) => { console.log('节点数据:', node.data); currentNodeId = node.data.id; // 显示属性栏 if (types.includes(node.data.type)) { $('.right-panel')[0].style.display = ; const currentNodeModel = graphModel.getNodeModelById(currentNodeId); $('#current-node')[0].innerText = `${currentNodeModel.text.value}(${currentNodeModel.id})`; $('#node-recommond-input')[0].value = currentNodeModel.properties.recommond || ; $('#node-url-input')[0].value = currentNodeModel.properties.pageUrl || ; } });
// 节点添加事件 lf.on("node:dnd-add", ({ data }) => { // 修改分组节点的文本位置 setTimeout(() => { if (data.type == 'main_title') { let text = graphModel.getNodeModelById(data.id).text; text.x = data.x - data.properties.width / 2 + 20 + text.value.length * 17; text.y = data.y - data.properties.height / 2 + 20; } if (data.type == 'sub_title') { let text = graphModel.getNodeModelById(data.id).text; text.x = data.x - data.properties.width / 2 + 12 + text.value.length * 15; text.y = data.y - data.properties.height / 2 + 17; } changeLetterSpacing(); }, 100) })
// 更改文本 lf.on("text:update", ({ data }) => { if (data.type == 'main_title') { // 修改文本节点位置并更新文本内容 let oldText = $(`#${data.id}`)[0].getAttribute('data-text'); lf.getNodeModelById(data.id).moveText((data.text.length - oldText.length) * 17, 0) $(`#${data.id}`)[0].setAttribute('data-text', data.text); } if (data.type == 'sub_title') { // 修改文本节点位置并更新文本内容 let oldText = $(`#${data.id}`)[0].getAttribute('data-text'); lf.getNodeModelById(data.id).moveText((data.text.length - oldText.length) * 15, 0) $(`#${data.id}`)[0].setAttribute('data-text', data.text); } setTimeout(() => { changeLetterSpacing(); }, 100) }) const changeLetterSpacing = () => { let titleNodes = $('.title-node'); for (let i = 0; i < titleNodes.length; i++) { titleNodes[i].parentElement.parentElement.nextSibling.childNodes[0].setAttribute('letter-spacing', 10) } } // 节点缩放事件 lf.on("node:resize", (e) => { // 重写分组节点的文本位置 if (e.data.type == 'main_title') { let height = e.data.properties.height; let width = e.data.properties.width; let text = e.model.text text.x = e.data.x - width / 2 + 20 + text.value.length * 17; text.y = e.data.y - height / 2 + 20; } if (e.data.type == 'sub_title') { let height = e.data.properties.height; let width = e.data.properties.width; let text = e.model.text text.x = e.data.x - width / 2 + 12 + text.value.length * 15; text.y = e.data.y - height / 2 + 17; } })
lf.on('node:mouseenter', ({ data, e }) => { if (types.includes(data.type)) { $(`#${data.id}`)[0].style.backgroundImage = 'url(https://patchwiki.biligame.com/images/lysk/1/1b/sxyq7pmcdcwe6uazcy7uqk89losay2s.png)'; $(`#${data.id}`)[0].style.backgroundSize = '100% 110%'; } }) lf.on('node:mouseleave', ({ data, e }) => { if (types.includes(data.type)) { $(`#${data.id}`)[0].style.backgroundImage = 'url(https://patchwiki.biligame.com/images/lysk/5/51/241yaplcz3ortw7m1n5o2fn37sotr7o.png)'; $(`#${data.id}`)[0].style.backgroundSize = '100% 90%'; } })
</script>
<style>
/* .main-line-node, .world-deep-node, .date-plot-node, .activity-plot-node, .phone-info-node */ .node-div { width: 100%; height: 100%; background-image: url(https://patchwiki.biligame.com/images/lysk/5/51/241yaplcz3ortw7m1n5o2fn37sotr7o.png); background-size: 100% 90%; background-repeat: no-repeat; background-position: center; cursor: pointer; box-sizing: border-box; }
.main-title-node::after { content: attr(data-text); position: absolute; line-height: 100%; left: 20px; top: 55px; font-size: 24px; letter-spacing: 10px; transform: scaleY(-1); transform-origin: top; background: linear-gradient(to bottom, transparent 30%, #aa79525c); -webkit-background-clip: text; background-clip: text; -webkit-text-fill-color: transparent; }
.sub-title-node { background-image: url(https://patchwiki.biligame.com/images/lysk/c/c1/c2xu5ggcepuj340avoyfrf782dxudm7.png); background-size: contain; background-repeat: no-repeat; width: 100%; height: 100%; }
.sub-title-node::after { content: attr(data-text); position: absolute; line-height: 100%; left: 12px; top: 50px; font-size: 20px; letter-spacing: 10px; transform: scaleY(-1); transform-origin: top; background: linear-gradient(to bottom, transparent 30%, #aa79525c); -webkit-background-clip: text; background-clip: text; -webkit-text-fill-color: transparent; }
.group-node { width: 100%; height: 100%; border-width: 19px 20px 16px 10px; border-style: solid; border-image-source: url(https://patchwiki.biligame.com/images/lysk/0/00/5xlyzapkfe69krklaup1dwxxsod8f67.png); border-image-slice: 4% 6% 2% 3% fill; border-image-repeat: round; cursor: pointer; box-sizing: border-box; }
.extended-read-node { width: 100%; height: 100%; background-image: url(https://patchwiki.biligame.com/images/lysk/5/57/cne0haqqiyjdpkww3qg54mxwfq2d3mo.png); background-size: 100% 100%; }
#node-page-panel { position: absolute; width: 250px; height: 150px; padding: 20px 25px; background-image: url(https://patchwiki.biligame.com/images/lysk/c/cb/cfp9bf0jkzu4ffyxeubrf4csbmbcxq4.png); background-size: 100% 100%; display: flex; flex-direction: column; justify-content: space-between; align-items: flex-end; }
#node-page-panel>.node-recommond { width: 100%; flex: 1; color: #aa7952; font-weight: bold; letter-spacing: 2px; }
#node-page-panel>.node-button { width: 90px; height: 25px; border: none; font-size: 13px; font-weight: bold; padding: 0; background-image: url(https://patchwiki.biligame.com/images/lysk/7/7f/8cfam9dgpmzkz2om9j2bjyyqeu9q9sf.png); background-size: 100% 113%; background-position: -2px 0px; background-repeat: no-repeat; background-color: transparent; letter-spacing: 4px; }
#container { width: 100%; height: calc(100vh - 20px); }
#buttons-tip { position: absolute; width: 500px; background-color: #fdaeae; border-radius: 5px; text-align: center; padding: 10px; top: 0; left: calc((100vw - 500px) / 2); }
.overlay { position: absolute; width: 80%; height: 80%; font-size: 10px; overflow: auto; top: 10%; left: 10%; background: #e1dddd; padding: 20px; }
.right-panel { position: absolute; height: 450px; width: 200px; right: 0; bottom: 0; background-color: #ddd; padding: 10px; }
.lf-dndpanel { top: 60px; }
.lf-control { right: unset !important; }
.custom-minimap { background: url(); }
.input-json { background: url(); }
.log-json { background: url(); }
.save-json { background: url(); }
.selection-open { background: url(); }
.selection-close { background: url(); }
.hoverpanel-open { background: url(); }
.hoverpanel-close { background: url(); }
.lf-dnd-item>.lf-dnd-shape { background-size: 100%; }
.lf-node-text--auto-wrap-inner { font-weight: bold; font-size: 18px; }
</style>