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: data.properties.width || 300,
height: data.properties.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: data.properties.width || 250,
height: data.properties.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' || data.type == 'extended_read_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);
}
}
// }, 50);
});
}
// 开启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;
changeLetterSpacing();
};
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;
}
if (data.type === 'group_node') {
data.properties.width = 300;
data.properties.height = 450;
}
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;
padding: 5px;
margin: 0;
}
.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 {
font-size: 10px;
}
.lf-dnd-item>.lf-dnd-shape {
background-size: 90%;
height: 30px;
}
.lf-node-text--auto-wrap-inner {
font-weight: bold;
font-size: 18px;
}
</style>

沪公网安备 31011002002714 号