// ==UserScript==
// @name 改善 JVS 开发体验
// @namespace https://github.com/11ze
// @match *://*/*
// @icon https://jvsoss.bctools.cn/jvs-public/jvs-auth-mgr/2_1_8/1/application/2024/03/08/2024-03-08950809363759403008-86d5bf057d4bdc12ed6d4341e84e118.png
// @grant GM_addStyle
// @license MIT
// @author 11ze
// @version 0.2.21
// @description 2025-03-04
// ==/UserScript==
// 检查是否包含 jvs-ui 的 link 标签
const isJVS = () => {
const links = document.getElementsByTagName('link');
for (const link of links) {
const matchList = [
'jvs-ui',
'edf-ui',
];
if (link.href && matchList.some((match) => link.href.includes(match))) {
console.log('%c「改善 JVS 开发体验」已检测到 JVS 环境', 'color: #0099ff;');
return true;
}
}
return false;
};
(function () {
('use strict');
if (!isJVS()) {
return;
}
setInterval(() => {
const operations = [
expandNames,
changeTitle,
enterAppCenter,
enterTabDesign,
adjustInterfaceAndComponentStyle,
addButtonToOpenNewLogicDesign,
addButtonToOpenNewLogicDesignForNestedLogic,
addButtonToCopyDesignName,
addButtonToCopyComponentName,
expandFormButton,
addButtonToClearAllFields,
expandLogicVariableButton,
addButtonToOpenNewFormOrListDesign,
highlightApps,
expandFormDesignAllComponentSettings,
autoExpandComponentLibraryCategory,
];
for (const operation of operations) {
try {
operation();
} catch (error) {
console.error('「改善 JVS 开发体验」' + operation.name + ' 运行错误:');
console.error(error);
}
}
}, 400);
// changeTitle
const envList = [
{ ip: '8.138.', env: '开发站' },
{ ip: 'jyy-test.', env: '测试站' },
{ ip: '47.107.', env: '正式站' },
];
const designMapping = {
逻辑设计: '逻',
列表设计: '列',
表单设计: '表',
流程设计: '流',
};
// log
const designSetting = {
逻辑设计: { color: 'blue', shortname: '逻辑' },
列表设计: { color: 'orange', shortname: '列表' },
表单设计: { color: 'green', shortname: '表单' },
流程设计: { color: 'purple', shortname: '流程' },
};
const logsLocalStorageKey = '__11ze_JVS_LOG_LOGS_';
const logOptionsLocalStorageKey = '__11ze_JVS_LOG_OPTIONS_';
// value 是日志对象的 key
const logOptions = [
{
label: '设计',
value: 'tabType',
options: ['全部', '逻辑', '表单', '列表', '流程'],
selected: '全部',
},
{
label: '操作',
value: 'type',
options: ['全部', '打开', '保存'],
selected: '打开',
},
];
const logSaveDays = 365;
window.designSetting = designSetting;
window.logsLocalStorageKey = logsLocalStorageKey;
window.logSaveDays = logSaveDays;
window.logOptionsLocalStorageKey = logOptionsLocalStorageKey;
window.logOptions = logOptions;
window.appNameSelectorList = [
// 应用左上角
'#app > div > div > div.jvs-layout > div.jvs-left > div > div.el-menu-scrollbar.el-scrollbar > div.el-scrollbar__wrap > div > div.app-item-info.app-item-info-hide > div > span',
// 应用左上角
'#app > div > div > div.jvs-layout > div.jvs-left > div > div.el-menu-scrollbar.el-scrollbar > div.el-scrollbar__wrap > div > div.app-item-info > div > span',
// 逻辑设计、列表设计、表单设计
'#app > div > div > div.design-header-box > div.header-left',
// 新版 JVS 列表设计、表单设计
'div.design-header-box > div.header-left',
];
function getQueryParamMapping(url) {
if (!url) {
return {};
}
// 先根据 # 分割,再平铺成基于 ? 分割的一维数组
const urlParts = url.split('#');
const urlQuery = urlParts.map((part) => part.split('?')[1]).join('&');
if (!urlQuery) {
return {};
}
const params = urlQuery.split('&');
const mapping = {};
for (let i = 0; i < params.length; i++) {
const param = params[i];
const id = param.split('=')[0];
const value = param.split('=')[1];
mapping[id] = value;
}
return mapping;
}
window.getQueryParamMapping = getQueryParamMapping;
function getUrl() {
return location.href;
}
window.getUrl = getUrl;
function getJvsAppId() {
const urlParams = getQueryParamMapping(getUrl());
return urlParams['jvsAppId'];
}
window.getJvsAppId = getJvsAppId;
function getTabType() {
// #tab-design > span
const typeDom = document.querySelector('#tab-design > span');
if (!typeDom) {
return '';
}
return typeDom.textContent;
}
window.getTabType = getTabType;
window.appModeMapKey = '__11ze_JVS_APP_MODE_MAP__';
function getAppModelMap() {
return JSON.parse(localStorage.getItem(window.appModeMapKey) ?? '{}');
}
window.getAppModelMap = getAppModelMap;
function getMode() {
const systemList = document.querySelector('.system-list');
if (!systemList) {
return '';
}
const systemListItems = systemList.querySelectorAll('li');
for (const systemListItem of systemListItems) {
if (systemListItem.innerText.includes('模式')) {
const mode = systemListItem.innerText.trim();
const jvsAppId = window.getJvsAppId();
if (jvsAppId) {
const appModeMap = window.getAppModelMap();
appModeMap[jvsAppId] = mode;
localStorage.setItem(window.appModeMapKey, JSON.stringify(appModeMap));
}
return mode;
}
}
return '';
}
window.getMode = getMode;
function getModeFromHistory() {
const urlParams = getQueryParamMapping(getUrl());
if (!urlParams) {
return '';
}
const jvsAppId = urlParams['jvsAppId'];
if (!jvsAppId) {
return '';
}
const appModeMap = window.getAppModelMap();
return appModeMap[jvsAppId];
}
window.getModeFromHistory = getModeFromHistory;
function getModeColor(mode) {
const modeColorMapping = {
开发模式: 'black',
测试模式: 'green',
正式模式: 'red',
};
return modeColorMapping[mode] ?? 'black';
}
window.getModeColor = getModeColor;
/**
* 展开应用名称和侧边栏功能名称
*/
function expandNames() {
const appNameSelectorList = [
// 首页
'#app > div > div > div.jvs-layout > div.jvs-main > div.el-scrollbar > div.el-scrollbar__wrap > div > div > div.top-outer-container > div.el-row-bottom-container.el-row > div > div > div > div > p',
// 旧应用中心
'#template > div.template-manage-content > div.template-manage-box > div.my-template-list > div > div > div.content > div.content-header > h5',
// 新首页 > 常用应用
'#app > div > div > div.jvs-layout > div.jvs-main > div.el-scrollbar > div.el-scrollbar__wrap > div > div > div.top-outer-container > div > div:nth-child(2) > div > div:nth-child(4) > div > div.card-body.el-col.el-col-24 > div > div > p',
// 新首页 > 应用中心
'#app > div > div > div.jvs-layout.jvs-layout-tempOpen > div.template-content-box > div.container.el-row > div:nth-child(2) > div > div > div > div:nth-child(1) > p',
// 应用左上角
'#app > div > div > div.jvs-layout > div.jvs-left > div > div.el-menu-scrollbar.el-scrollbar > div.el-scrollbar__wrap > div > div.app-item-info.app-item-info-hide > div > span',
// 应用左上角
'#app > div > div > div.jvs-layout > div.jvs-left > div > div.el-menu-scrollbar.el-scrollbar > div.el-scrollbar__wrap > div > div.app-item-info > div > span',
// 逻辑设计、列表设计、表单设计
'#app > div > div > div.design-header-box > div.header-left',
];
for (let i = 0; i < appNameSelectorList.length; i++) {
document.querySelectorAll(appNameSelectorList[i]).forEach((el) => {
el.style.whiteSpace = 'normal';
});
}
}
/**
* 修改浏览器标签页标题
*/
function changeTitle() {
function getEnvironment() {
const url = location.href;
for (let i = 0; i < envList.length; i++) {
if (url.includes(envList[i].ip)) {
return envList[i].env;
}
}
return '';
}
function getAppName() {
// 逻辑设计
// 把 selector 放到 getAppName 获取不到,先保留下面的处理
const title = document.querySelector(
'#app > div > div > div.design-header-box > div.header-left > span:nth-child(3)'
);
if (title) {
return title.textContent.trim();
}
for (let i = 1; i < window.appNameSelectorList.length; i++) {
const allTextElements = document.querySelectorAll(window.appNameSelectorList[i]);
for (let j = 0; j < allTextElements.length; j++) {
const text = allTextElements[j].innerHTML;
if (!text.includes('<')) {
return text.trim();
}
if (i === 3) {
let splitText = text.split('</svg>')[1];
if (!splitText) {
continue;
}
splitText = splitText.split('center;">')[1];
if (!splitText) {
continue;
}
splitText = splitText.split('<')[0];
return splitText.trim();
}
const splitText = text.split('</svg>')[1];
if (!splitText) {
continue;
}
if (splitText.includes('<')) {
return splitText.split('<')[0].trim();
}
return splitText.trim();
}
}
if (location.href.includes('doc.html')) {
return '接口文档';
}
return '';
}
window.getAppName = getAppName;
function getTabType() {
const typeDom = document.querySelector('#tab-design > span');
if (!typeDom) {
return '';
}
if (designMapping[typeDom.textContent]) {
return designMapping[typeDom.textContent];
}
return typeDom.textContent.trim();
}
const appName = getAppName();
const tabType = getTabType();
function changeFavicon(iconURL) {
const links = document.querySelectorAll("link[rel*='icon']"); // 获取现有的 favicon 元素
if (!links) {
// 如果不存在,则创建一个新的 link 元素
const link = document.createElement('link');
link.rel = 'shortcut icon'; // 或 'icon'
link.type = 'image/x-icon'; // 设置类型,虽然并非所有浏览器都强制要求
document.head.appendChild(link);
}
links.forEach(function (link) {
link.href = iconURL; // 设置新的图标 URL
});
}
if (tabType) {
document.title = appName;
changeFavicon(window.iconMap[tabType]);
} else {
let prefix = getMode();
if (!prefix) {
prefix = getEnvironment();
}
document.title = prefix + '|' + (appName ? appName : '未打开应用');
}
}
/**
* 首次进入平台首页时自动进入应用中心
*/
function enterAppCenter() {
const selector =
'#app > div > div > div.jvs-tags > div > div > div.top-nav > ul > li:nth-child(2) > span';
const element = document.querySelector(selector);
const url = location.href;
if (element) {
if (
element.innerText === '应用中心' &&
url.includes('wel/index') &&
!element.hasAttribute('app-center-clicked-11ze')
) {
element.click();
element.setAttribute('app-center-clicked-11ze', 'true');
}
}
}
window.secondTabDesignClicked11ze = false;
/**
* 在设计页面自动点击 tab,如【表单设计】
*/
function enterTabDesign() {
const selector = '#tab-design';
const element = document.querySelector(selector);
if (!element) {
return;
}
if (element.getAttribute('second-tab-design-clicked-11ze')) {
if (window.secondTabDesignClicked11ze) {
return;
}
window.secondTabDesignClicked11ze = true;
element.click();
return;
}
element.click();
element.setAttribute('second-tab-design-clicked-11ze', 'true');
}
/**
* 调整界面、组件样式
*/
function adjustInterfaceAndComponentStyle() {
// 旧版 JVS,逻辑设计,所有在用可拖拽组件,改颜色
const draggableComponents = document.querySelectorAll(
'div.jvs-rule-node.ef-node-container.jtk-droppable'
);
const typeToColorList = [
{
types: [
'数据模型',
'跳过数据权限',
'删除数据',
'新增数据',
'查询单条',
'查询所有',
'更新模型',
'统计条数',
],
color: '#ffcbda',
},
{
types: ['逻辑引擎', '逻辑应用'],
color: '#d4e3fc',
},
{
types: ['循环容器', '对数组对象进行遍历'],
color: '#c8f0c7',
},
{
types: ['中止程序', '提示消息'],
color: '#fef7d7',
},
{
types: [
'对象变量',
'数组变量',
'对象数组变量',
'固定变量',
'对象结构',
'公式值',
'等变量',
'结构示例',
],
color: '#e6e6fa',
},
];
for (const component of draggableComponents) {
const text = component.innerText.trim();
for (const typeToColor of typeToColorList) {
if (typeToColor.types.some((t) => text.includes(t))) {
component.style.backgroundColor = typeToColor.color;
component.style.borderColor = typeToColor.color;
break;
}
}
}
// 新版 JVS,逻辑设计,所有在用可拖拽组件,改颜色
const newDraggableComponents = document.querySelectorAll('.jvs-rule-node.ef-node-container');
for (const component of newDraggableComponents) {
// 获取组件所有文本
const text = component.textContent.trim();
for (const typeToColor of typeToColorList) {
if (typeToColor.types.some((t) => text.includes(t))) {
component.style.backgroundColor = typeToColor.color;
component.style.borderColor = typeToColor.color;
break;
}
}
}
// 设置侧边栏可选组件的颜色
const sidebarComponents = document.querySelectorAll('.getItem');
for (const component of sidebarComponents) {
const text = component.innerText.trim();
for (const typeToColor of typeToColorList) {
if (typeToColor.types.some((t) => text.includes(t))) {
component.style.backgroundColor = typeToColor.color;
component.style.borderColor = typeToColor.color;
break;
}
}
}
// 新版 JVS,表单设计,组件的名称全部显示出来
const formComponents = document.querySelectorAll('.formitem2');
for (const component of formComponents) {
const parent = component.parentElement;
if (!parent.getAttribute('draggable')) {
continue;
}
component.classList.add('active-formitem2');
}
}
/**
* 从日志或 url 生成跳转链接
*
* @param {*} id 设计 id
* @param {*} isFromUrl 是否是从 url 中获取
* @returns string | null
*/
function getUrlFromLogs(id, isFromUrl) {
if (!id) {
return null;
}
const logs = window.getLogs();
for (let i = logs.length - 1; i >= 0; i--) {
const log = logs[i];
if (log.id === id) {
return log.url;
}
}
if (!isFromUrl) {
return null;
}
const url = location.href; // http://xxx?id=xxx&xxx
return url.replace(/id=([^&]*)/, `id=${id}`);
}
/**
* 从日志和 url 生成跳转链接
* @returns string | null
*/
function getUrlFromLogsAndUrl(logicName, jvsAppId) {
if (!logicName) {
return null;
}
const logs = window.getLogs();
for (let i = logs.length - 1; i >= 0; i--) {
const log = logs[i];
if (log.jvsAppId !== jvsAppId) {
continue;
}
if (log.designName === logicName) {
return log.url;
}
if (log.designName.includes(logicName)) {
return log.url;
}
if (logicName.includes(log.designName)) {
return log.url;
}
}
return null;
}
/**
* 逻辑设计,检查到【逻辑调用】组件时,自动添加一个按钮用于查看对应的逻辑设计
*/
function addButtonToOpenNewLogicDesign() {
const buttonClass = 'ze-look-logic-button';
const selector = '.el-form-item__label';
const labels = document.querySelectorAll(selector);
for (const label of labels) {
if (!label.innerText.includes('逻辑引擎远程调用key')) {
continue;
}
const logicKey = label.nextElementSibling.querySelector('.el-input__inner').title;
const newUrl = getUrlFromLogs(logicKey, true);
if (!newUrl) {
continue;
}
const existedButton = label.querySelector('.' + buttonClass);
if (existedButton) {
if (existedButton.getAttribute('target-key') === logicKey) {
continue;
}
existedButton.remove();
}
const newButton = document.createElement('button');
newButton.className =
buttonClass + ' modern-button el-button el-button--primary el-button--mini button-11ze';
newButton.innerHTML = '查看';
newButton.setAttribute('target-key', logicKey);
newButton.onclick = function () {
window.open(newUrl, '_blank');
};
newButton.style.marginLeft = '10px';
// 将按钮直接添加到 label 元素中
label.appendChild(newButton);
}
}
/**
* 新版逻辑嵌套组件,检查到【逻辑嵌套】组件时,自动添加一个按钮用于查看对应的逻辑设计
* 从已打开过的逻辑设计中获取跳转链接
*/
function addButtonToOpenNewLogicDesignForNestedLogicLater() {
const buttonClass = 'ze-look-logic-button';
const selector = '.el-form-item__label';
const labels = document.querySelectorAll(selector);
for (const label of labels) {
if (!label.innerText.includes('选择逻辑引擎')) {
continue;
}
const logicNameElement = label.nextElementSibling.querySelector(
'.el-select-dropdown__item.selected > span'
);
if (!logicNameElement) {
continue;
}
const logicName = logicNameElement.innerText.trim();
const jvsAppId = window.getJvsAppId();
const newUrl = getUrlFromLogsAndUrl(logicName, jvsAppId);
if (!newUrl) {
continue;
}
const existedButton = label.querySelector('.' + buttonClass);
if (existedButton) {
if (existedButton.getAttribute('target-key') === logicName) {
continue;
}
existedButton.remove();
}
const newButton = document.createElement('button');
newButton.className =
buttonClass + ' modern-button el-button el-button--primary el-button--mini button-11ze';
newButton.innerHTML = '查看';
newButton.setAttribute('target-key', logicName);
newButton.onclick = function () {
window.open(newUrl, '_blank');
};
newButton.style.marginLeft = '10px';
// 将按钮直接添加到 label 元素中
label.appendChild(newButton);
}
}
/**
* 新版逻辑嵌套组件,检查到【逻辑嵌套】组件时,自动添加一个按钮用于查看对应的逻辑设计
* 从左上角的 icon 中获取跳转页面
*/
function addButtonToOpenNewLogicDesignForNestedLogic() {
const buttonClass = 'ze-look-logic-button';
const selector = '.el-form-item__label';
const labels = document.querySelectorAll(selector);
const otherRuleListIcon = document.querySelector('.rule-list-icon');
if (!otherRuleListIcon) {
return;
}
if (!otherRuleListIcon.getAttribute('check-logic-design-rule-list-icon-11ze')) {
// 点击后才有逻辑列表
otherRuleListIcon.click();
otherRuleListIcon.click();
}
otherRuleListIcon.setAttribute('check-logic-design-rule-list-icon-11ze', 'true');
function createButton(target, element, logicName) {
const newButton = document.createElement('button');
newButton.className =
buttonClass + ' modern-button el-button el-button--primary el-button--mini button-11ze';
newButton.innerHTML = '查看';
newButton.setAttribute('target-key', logicName);
newButton.onclick = function () {
element.click();
};
newButton.style.marginLeft = '10px';
// 将按钮直接添加到 label 元素中
target.appendChild(newButton);
}
const otherRuleList = document.querySelectorAll('.other-rule-list > .list-box > .list-item');
for (const label of labels) {
if (!label.innerText.includes('选择逻辑引擎')) {
continue;
}
const logicNameElement = document.querySelector(
'.el-scrollbar__view.el-select-dropdown__list > .el-select-dropdown__item.selected > span'
);
if (!logicNameElement) {
continue;
}
const logicName = logicNameElement.innerText.trim();
const existedButton = label.querySelector('.' + buttonClass);
if (existedButton) {
if (existedButton.getAttribute('target-key') === logicName) {
continue;
}
existedButton.remove();
}
for (const otherRule of otherRuleList) {
const inputName = otherRule.innerHTML.trim();
if (inputName === logicName) {
createButton(label, otherRule, logicName);
return;
}
if (inputName.includes(logicName)) {
createButton(label, otherRule, logicName);
return;
}
if (logicName.includes(inputName)) {
createButton(label, otherRule, logicName);
return;
}
}
addButtonToOpenNewLogicDesignForNestedLogicLater();
}
}
// 新版 JVS,在逻辑设计名称旁边添加复制按钮
function addButtonToCopyDesignName() {
// 逻辑设计
let designName = document.querySelector(
'#app > div > div > div.design-header-box > div.header-left > span'
);
if (!designName) {
// 表单设计
designName = document.querySelector(
'#app > div > div > div:nth-child(1) > div.design-header-box > div.header-left > span'
);
}
if (designName) {
const designNameText = designName.innerText.trim();
const existedButton = document.querySelector('#copy-design-name-button-11ze');
if (existedButton) {
if (existedButton.getAttribute('design-name-11ze') === designNameText) {
return;
}
existedButton.remove();
}
if (!designName.querySelector('use')) {
return;
}
const copyButton = document.createElement('button');
copyButton.innerHTML = '复制';
copyButton.className =
'modern-button el-button el-button--primary el-button--mini button-11ze';
copyButton.id = 'copy-design-name-button-11ze';
if (designNameText) {
copyButton.setAttribute('design-name-11ze', designNameText);
}
copyButton.onclick = function () {
copyToClipboard(designNameText, copyButton, '已复制');
};
designName.parentNode.insertBefore(copyButton, designName.nextSibling);
}
}
function copyToClipboard(text, button, successMessage) {
const copyTextToClipboard = (text) => {
if (navigator.clipboard && navigator.clipboard.writeText) {
return navigator.clipboard.writeText(text);
} else {
// 回退方法:创建一个临时的文本区域元素
const textArea = document.createElement('textarea');
textArea.value = text;
textArea.style.position = 'fixed'; // 避免滚动到底部
document.body.appendChild(textArea);
textArea.focus();
textArea.select();
try {
document.execCommand('copy');
return Promise.resolve();
} catch (err) {
return Promise.reject(err);
} finally {
document.body.removeChild(textArea);
}
}
};
copyTextToClipboard(text)
.then(() => {
const originalText = button.textContent;
button.textContent = successMessage;
button.disabled = true;
setTimeout(() => {
button.textContent = originalText;
button.disabled = false;
}, 1000);
})
.catch((err) => {
console.error('复制失败:', err);
alert('复制失败,请重试');
});
}
window.currentPageNotAddCopyComponentNameButton = false;
/**
* 新版 JVS,添加按钮复制组件名称
*/
function addButtonToCopyComponentName() {
const buttonClass = 'ze-copy-component-name-button';
const componentName = document.querySelector('#node_detailpannel > h4 > div > span');
if (!componentName) {
return;
}
// 不在旧版加按钮,如果父级有子元素 el-icon-document-copy,则 return
if (componentName.parentNode.querySelector('.el-icon-document-copy')) {
window.currentPageNotAddCopyComponentNameButton = true;
return;
}
if (window.currentPageNotAddCopyComponentNameButton) {
return;
}
const componentNameText = componentName.innerText.trim();
const existedButton = document.querySelector('#copy-component-name-button-11ze');
if (existedButton) {
if (existedButton.getAttribute('component-name-11ze') === componentNameText) {
return;
}
existedButton.remove();
}
const copyButton = document.createElement('button');
copyButton.innerHTML = '复制';
copyButton.className =
buttonClass + ' modern-button el-button el-button--primary el-button--mini button-11ze';
copyButton.onclick = function () {
copyToClipboard(componentNameText, copyButton, '已复制');
};
copyButton.id = 'copy-component-name-button-11ze';
copyButton.setAttribute('component-name-11ze', componentNameText);
componentName.parentNode.insertBefore(copyButton, componentName.nextSibling);
}
/**
* 新版 JVS,表单设计,自动展开表单设计的按钮设置
*/
function expandFormButton() {
const buttons = document.querySelectorAll('.item-body');
if (buttons) {
for (const button of buttons) {
if (button.getAttribute('item-body-checked-11ze') === 'true') {
continue;
}
if (button.style.display === 'none') {
button.style.display = 'block';
button.setAttribute('item-body-checked-11ze', 'true');
}
}
}
}
/**
* 逻辑设计,添加按钮一键清空所有字段
*/
function addButtonToClearAllFields() {
const boxes = document.querySelectorAll('.data-model-box');
for (let i = 0; i < boxes.length; i++) {
const box = boxes[i];
if (box.querySelector('#clear-all-fields-button-11ze' + i)) {
continue;
}
const button = document.createElement('button');
button.className = 'modern-button el-button el-button--primary el-button--mini button-11ze';
button.innerHTML = '清空';
button.id = 'clear-all-fields-button-11ze' + i;
button.onclick = function () {
const ps = box.querySelectorAll('p');
for (let i = ps.length - 1; i >= 0; i--) {
const el = ps[i];
if (el.querySelector('.delete-icon-button')) {
el.querySelector('.delete-icon-button > span').click();
}
if (el.querySelector('.el-icon-delete')) {
el.querySelector('.el-icon-delete').click();
}
}
};
box.insertBefore(button, box.firstChild);
}
}
/**
* 新版 JVS,逻辑设计,自动展开变量组件设置里的按钮
*/
function expandLogicVariableButton() {
const buttons = document.querySelectorAll('.bottom-body');
if (buttons) {
for (const button of buttons) {
if (button.getAttribute('bottom-body-checked-11ze') === 'true') {
continue;
}
if (button.style.display === 'none') {
button.style.display = 'block';
button.setAttribute('bottom-body-checked-11ze', 'true');
}
}
}
}
/**
* 在列表表单列表的 id 旁边添加查看按钮
*/
function addButtonToOpenNewFormOrListDesign() {
const selector =
'div.table-body-box > div > div.el-table__body-wrapper.is-scrolling-none > table > tbody > tr > td:nth-child(2) > div > span > span > div';
const elements = document.querySelectorAll(selector);
for (const element of elements) {
const designId = element.innerText.trim();
if (!designId) {
continue;
}
element.style.whiteSpace = 'normal';
if (element.getAttribute('form-added-button-11ze')) {
continue;
}
const targetUrl = getUrlFromLogs(designId, false);
if (!targetUrl) {
continue;
}
const copyButton = document.createElement('button');
copyButton.innerHTML = '查看';
copyButton.className =
'modern-button el-button el-button--primary el-button--mini button-11ze';
copyButton.id = 'open-new-form-or-list-design-button-11ze';
copyButton.onclick = function () {
window.open(targetUrl, '_blank');
};
const targetElement =
element.parentElement.parentElement.parentElement.parentElement.parentElement.querySelector(
'td:nth-child(5) > div > div'
);
if (targetElement) {
targetElement.appendChild(copyButton);
element.setAttribute('form-added-button-11ze', 'true');
}
}
}
/**
* 自动展开表单设计所有组件设置
*/
function expandFormDesignAllComponentSettings() {
// 表单设计
const tabType = window.getTabType();
if (tabType !== '表单设计') {
return;
}
// 文字标题元素跟展开内容元素同级
const buttons = document.querySelectorAll('.el-collapse-item > .el-collapse-item__wrap');
for (const button of buttons) {
if (button.getAttribute('bottom-body-checked-11ze')) {
continue;
}
const text = button.parentElement.innerText.trim();
const targetNames = ['设置', '扩展', '功能', '校验'];
for (const targetName of targetNames) {
if (text.includes(targetName)) {
button.style.display = 'block';
break;
}
}
button.setAttribute('bottom-body-checked-11ze', 'true');
}
}
/**
* 高亮应用中心的应用
*/
function highlightApps() {
const highlightAppsKey = '__11ze_HIGHLIGHT_APPS__';
const labelClass = 'ze-highlight-label';
const appList = JSON.parse(localStorage.getItem(highlightAppsKey) ?? '[]');
function getContentSelector() {
return 'div > div > div > p';
}
function handle(nodes, appList) {
// 如果在 appList 中,就高亮
nodes.forEach((n) => {
const text = getNodeText(n);
const label = n.querySelector(`.${labelClass}`);
if (!label) {
return;
}
if (appList.includes(text)) {
n.style.border = '2px solid blue';
label.style.backgroundColor = 'white';
} else {
n.style.border = '2px solid transparent';
label.style.backgroundColor = 'white';
}
});
}
function getNodeText(node) {
return node.querySelector(getContentSelector()).innerText.trim();
}
function handleClickNode(node) {
const text = getNodeText(node);
if (appList.includes(text)) {
appList.splice(appList.indexOf(text), 1);
} else {
appList.push(text);
}
localStorage.setItem(highlightAppsKey, JSON.stringify(appList));
}
function main() {
const containerSelector = '.application';
// 找到相关容器,不存在就结束
const application = document.querySelector(containerSelector);
if (!application) return;
// 找到相关数据项,不存在就结束
const nodes = [...document.querySelectorAll(containerSelector)];
if (!nodes) return; // 不存在,结束
// 设置点击事件
nodes.forEach((n) => {
if (n.querySelector(`.${labelClass}`)) {
return;
}
// 加一个按钮,点击后高亮
const button = document.createElement('button');
button.innerHTML = ' ';
button.className =
labelClass + ' modern-button el-button el-button--primary el-button--mini';
button.style.borderColor = '#c8f0c7';
button.style.borderRadius = '5px';
button.style.borderWidth = '1px';
button.style.borderStyle = 'solid';
button.style.borderColor = '#c8f0c7';
button.onclick = (event) => {
event.stopPropagation();
handleClickNode(n);
};
n.querySelector('div > div > div').appendChild(button);
});
// 渲染
handle(nodes, appList);
}
main();
}
/**
* 窗口聚焦时自动松开一次左 Ctrl 键
* 场景:按快捷键切换软件时,如果包含 Ctrl,回到逻辑设计时,Ctrl 会一直按住,导致鼠标拖拽变成画框
* 不用了,控制台有错误:Uncaught TypeError: Cannot read properties of undefined (reading 'removeEventListener')
at HTMLDocument.<anonymous> (page.f3111d50.js:34:1216471)
*/
function autoClickLeftCtrlKey() {
const container = document.querySelector('.container');
if (!container) {
return;
}
container.addEventListener('focus', function () {
// 创建一个模拟 Ctrl 键弹起的 KeyboardEvent (可选,如果需要模拟按下和弹起)
const ctrlUp = new KeyboardEvent('keyup', {
key: 'Control',
code: 'ControlLeft',
ctrlKey: false,
bubbles: true,
});
container.dispatchEvent(ctrlUp);
});
}
/**
* 逻辑设计,自动展开组件库里指定的分类
*/
function autoExpandComponentLibraryCategory() {
if (window.autoExpandComponentLibraryCategory11ze) {
return;
}
const ruleCategories = document.querySelectorAll('.left-tool-list-box > .rule-assembly-list');
if (ruleCategories.length === 0) {
return;
}
for (const ruleCategory of ruleCategories) {
if (ruleCategory.classList.contains('open')) {
continue;
}
if (ruleCategory.getAttribute('auto-expand-component-library-category-11ze')) {
continue;
}
const textDom = ruleCategory.querySelector('div > div > .label');
if (!textDom) {
continue;
}
const text = textDom.innerText.trim();
if (['模型插件', '服务插件'].includes(text)) {
const left = ruleCategory.querySelector('.t-left');
if (left) {
left.dispatchEvent(new MouseEvent('click', { bubbles: true }));
}
}
ruleCategory.setAttribute('auto-expand-component-library-category-11ze', 'true');
}
function simulateMouseClick(element) {
const mouseClickEvents = ['mousedown', 'click', 'mouseup'];
mouseClickEvents.forEach((mouseEventType) => {
const mouseEvent = new MouseEvent(mouseEventType, {
bubbles: true, // 允许事件冒泡
cancelable: true, // 允许事件被取消
clientX: element.clientWidth, // 设置点击位置
clientY: element.clientHeight, // 设置点击位置
});
element.dispatchEvent(mouseEvent);
});
}
// 鼠标左键点击一次,收起组件库
const container = document.querySelector('.butterfly-wrapper');
if (container) {
simulateMouseClick(container);
}
window.autoExpandComponentLibraryCategory11ze = true;
}
})();
/**
* 记录和查看开发日志
*/
window.onload = function () {
('use strict');
if (!isJVS()) {
return;
}
function log() {
function getTabType() {
return window.getTabType();
}
function getUrl() {
return window.getUrl();
}
function getJvsAppId() {
return window.getJvsAppId();
}
function getId() {
return getQueryParamMapping(getUrl())['id'];
}
function getCurrentTime() {
return new Date().getTime();
}
function getNewTabTitle() {
return window.getAppName();
}
const appIdSelectorList = window.appNameSelectorList;
function getAppName() {
for (let i = 0; i < appIdSelectorList.length; i++) {
const allTextElements = document.querySelectorAll(appIdSelectorList[i]);
for (let j = 0; j < allTextElements.length; j++) {
const text = allTextElements[j].textContent;
if (text) {
const textArray = text
.trim()
.split('\n')
.map((item) => item.trim());
const newTextArray = [];
textArray.forEach((item) => {
if (item.trim() === '') {
return;
}
newTextArray.push(item.trim());
});
if (document.querySelector('.list-item')) {
const name = newTextArray[newTextArray.length - 3];
if (name) {
return name;
}
}
return newTextArray[0];
}
}
}
return '';
}
const url = getUrl();
const id = getId();
const time = getCurrentTime();
const tabType = getTabType();
const designName = getNewTabTitle();
const appName = getAppName();
const jvsAppId = getJvsAppId();
// 如果有一个为空则返回 null
if (!designName || !appName || !id || !jvsAppId || !tabType) {
return null;
}
return {
tabType,
url,
time,
designName,
appName,
id,
jvsAppId,
};
}
function cutOverdueLogs(logs, currentTime) {
// 每个 log 有 time 字段,格式为时间戳
// 从数组删除时间戳跟当前时间相差 n 天的 log
for (let i = 0; i < logs.length; i++) {
const log = logs[i];
if (!log.time) {
logs.splice(i, 1);
continue;
}
if (Math.abs(currentTime - log.time) > logSaveDays * 24 * 60 * 60 * 1000) {
logs.splice(i, 1);
continue;
}
}
logs = uniqueLogs(logs);
localStorage.setItem(logsLocalStorageKey, JSON.stringify(logs));
return logs;
}
function getLogs() {
const logs = JSON.parse(localStorage.getItem(logsLocalStorageKey));
if (!logs) {
return [];
}
return cutOverdueLogs(logs, new Date().getTime());
}
window.getLogs = getLogs;
function saveLog(logObj, type) {
if (!logObj) {
return;
}
logObj.type = type;
const logList = getLogs();
logList.push(logObj);
localStorage.setItem(logsLocalStorageKey, JSON.stringify(logList));
}
function uniqueLogs(logs) {
const appIds = [];
const uniqueLogs = [];
for (let i = logs.length - 1; i >= 0; i--) {
const log = logs[i];
const urlParams = getQueryParamMapping(log.url);
const id = urlParams['id'] + log.type;
if (!appIds.includes(id)) {
appIds.push(id);
uniqueLogs.push(log);
}
}
return uniqueLogs.reverse();
}
function formatTime(time) {
const date = new Date(time);
const month = String(date.getMonth() + 1).padStart(2, '0');
const day = String(date.getDate()).padStart(2, '0');
const hours = String(date.getHours()).padStart(2, '0');
const minutes = String(date.getMinutes()).padStart(2, '0');
const seconds = String(date.getSeconds()).padStart(2, '0');
return `${month}-${day} ${hours}:${minutes}:${seconds}`;
}
function getQueryParamMapping(url) {
return window.getQueryParamMapping(url);
}
function getJvsAppIdsFromLogs(logs) {
const appIds = [];
for (let i = 0; i < logs.length; i++) {
const log = logs[i];
if (!appIds.includes(log.jvsAppId)) {
appIds.push(log.jvsAppId);
}
}
return appIds;
}
function getRandomColor() {
const letters = '0123456789ABCDEF';
let color = '#';
for (let i = 0; i < 6; i++) {
color += letters[Math.floor(Math.random() * 16)];
}
return color;
}
function getAppColorMapping(appNames) {
const appColorMapping = {};
for (let i = 0; i < appNames.length; i++) {
const appName = appNames[i];
appColorMapping[appName] = getRandomColor();
}
return appColorMapping;
}
function getOptions() {
const options = JSON.parse(localStorage.getItem(logOptionsLocalStorageKey));
if (!options) {
return logOptions;
}
if (options.length !== logOptions.length) {
return logOptions;
}
return options;
}
function saveOptions(selectedValue) {
if (!selectedValue) {
return;
}
const options = getOptions();
const selectedArray = selectedValue.split('-');
const value = selectedArray[0];
const selected = selectedArray[1];
for (let i = 0; i < options.length; i++) {
const option = options[i];
if (option.value === value) {
option.selected = selected;
}
}
localStorage.setItem(logOptionsLocalStorageKey, JSON.stringify(options));
}
function filterLogs(logs, options) {
if (!options) {
return logs;
}
const filteredLogs = [];
for (let i = 0; i < logs.length; i++) {
let log = logs[i];
let allSelected = true;
for (let j = 0; j < options.length; j++) {
const option = options[j];
if (option.selected === '全部') {
continue;
}
if (!log[option.value].includes(option.selected)) {
allSelected = false;
break;
}
}
if (allSelected) {
filteredLogs.push(log);
}
}
return filteredLogs;
}
function showPopup() {
const popupId = '11ze-jvs-log-popup';
const popup = document.createElement('div');
popup.className = 'popup';
popup.id = popupId;
const oldPopup = document.getElementById(popupId);
if (oldPopup) {
oldPopup.remove();
return;
}
const optionsDiv = document.createElement('div');
optionsDiv.style.textAlign = 'right';
const lastLogOptions = getOptions();
for (let i = 0; i < lastLogOptions.length; i++) {
const currentDiv = document.createElement('div');
currentDiv.className = 'log-11ze-select-container';
const selectDom = document.createElement('select');
selectDom.className = 'log-11ze-select';
for (let j = 0; j < lastLogOptions[i].options.length; j++) {
const option = document.createElement('option');
option.value = lastLogOptions[i].value + '-' + lastLogOptions[i].options[j];
option.textContent = lastLogOptions[i].options[j];
if (lastLogOptions[i].selected === lastLogOptions[i].options[j]) {
option.selected = true;
}
selectDom.appendChild(option);
}
const pDom = document.createElement('p');
pDom.className = 'log-11ze-select-label';
pDom.textContent = lastLogOptions[i].label;
currentDiv.appendChild(pDom);
currentDiv.appendChild(selectDom);
optionsDiv.appendChild(currentDiv);
selectDom.onchange = function () {
const selectedValue = selectDom.value;
saveOptions(selectedValue);
showPopup();
showPopup();
};
}
popup.appendChild(optionsDiv);
let logs = getLogs();
logs = filterLogs(logs, lastLogOptions);
const appColorMapping = getAppColorMapping(getJvsAppIdsFromLogs(logs));
const appModeMap = window.getAppModelMap();
const listContent = [];
for (let i = logs.length - 1; i >= 0; i--) {
const oneLog = logs[i];
const datetime = formatTime(oneLog.time);
const urlParams = getQueryParamMapping(oneLog.url);
const currentType = designSetting[oneLog.tabType] ?? {
color: 'red',
shortname: '未知',
};
const logFieldColor = oneLog.type === '打开' ? 'black' : 'red';
let appName = oneLog.appName;
if (appName.length > 16) {
appName = appName.substring(0, 16) + '…';
}
const designName = oneLog.designName;
const jvsAppId = oneLog.jvsAppId;
const appColor = appColorMapping[jvsAppId];
const mode = appModeMap[jvsAppId] ?? '';
const modeColor = window.getModeColor(mode);
listContent.push(`
<tr class="log-11ze-table-tr">
<td> ${urlParams.id} </td>
<td style="color: ${appColor}"> ${appName} </td>
<td style="color: ${modeColor}"> ${mode.replace('模式', '')} </td>
<td style="color: ${currentType.color}"> ${currentType.shortname} </td>
<td> ${designName} </td>
<td style="color: ${logFieldColor}"> ${oneLog.type} </td>
<td> ${datetime} </td>
<td> <a href="${oneLog.url}" target="_blank">打开</a> </td>
</tr>
`);
}
const logTable = document.createElement('table');
logTable.className = 'table';
logTable.innerHTML = `
<thead>
<tr style="background-color: #eef5fe" class="log-11ze-table-tr">
<th> 设计 id </th>
<th> 应用 </th>
<th> 模式 </th>
<th> 类型 </th>
<th> 名称 </th>
<th> 操作 </th>
<th> 时间 </th>
<th> 操作 </th>
</tr>
</thead>
<tbody>
${listContent.join('')}
</tbody>
`;
logTable.style.fontSize = '0.9em';
logTable.style.minWidth = '800px';
logTable.style.borderBottom = '1px solid #dddddd';
logTable.style.textAlign = 'left';
popup.appendChild(logTable);
popup.style.position = 'fixed';
popup.style.top = '50px';
popup.style.left = '20px';
popup.style.zIndex = '9999';
popup.style.backgroundColor = 'white';
popup.style.padding = '10px';
popup.style['max-height'] = '800px';
popup.style['overflow-y'] = 'auto';
document.body.appendChild(popup);
// 添加点击事件监听器
document.addEventListener('click', closePopupOnOutsideClick);
}
// 新增函数: 检查点击是否在popup外部并关闭popup
function closePopupOnOutsideClick(event) {
const popup = document.getElementById('11ze-jvs-log-popup');
// 这是打开popup的按钮
const button = document.querySelector('.modern-button');
if (popup && !popup.contains(event.target) && event.target !== button) {
popup.remove();
document.removeEventListener('click', closePopupOnOutsideClick);
}
}
window.savedLogDesignName = '';
window.savedLogAppName = '';
function main() {
const newLog = log();
if (newLog && newLog.tabType) {
const needToSave =
window.savedLogDesignName !== newLog.designName ||
window.savedLogAppName !== newLog.savedLogAppName;
if (needToSave) {
saveLog(newLog, '打开');
window.savedLogDesignName = newLog.designName;
window.savedLogAppName = newLog.appName;
}
}
// 用于显示当前在的模式
let buttonName = '日志';
let mode = getModeFromHistory();
if (!mode) {
mode = window.getMode();
}
if (mode) {
const modeSpan = document.createElement('span');
modeSpan.style.color = window.getModeColor(mode);
modeSpan.innerHTML = mode;
buttonName = modeSpan.outerHTML + '|' + buttonName;
}
createButton(buttonName);
}
function createButton(buttonName) {
const existButton = document.getElementById('ze-jvs-log-button');
if (existButton) {
if (existButton.innerHTML === buttonName) {
return;
}
existButton.remove();
}
// 在页面固定位置(绝对位置,悬浮)插入一个按钮
// 点击按钮打开一个弹窗显示内容(全局唯一),已有窗口则直接显示
const button = document.createElement('button');
button.innerHTML = buttonName;
button.className = 'modern-button el-button el-button--primary el-button--mini button-11ze';
button.style.position = 'fixed';
button.style.top = '10px';
button.style.right = '300px';
button.style.zIndex = '9998';
button.style.fontSize = '13px';
button.id = 'ze-jvs-log-button';
button.onclick = function (event) {
event.stopPropagation(); // 阻止事件冒泡到 document
showPopup();
};
document.body.appendChild(button);
}
// 逻辑设计的保存按钮
const saveButton = document.querySelector(
'#app > div > div > div.design-header-box > div.header-right > button'
);
if (saveButton) {
saveButton.addEventListener('click', function () {
const newLog = log();
if (newLog) {
saveLog(newLog, '保存');
}
});
}
setInterval(() => {
try {
main();
} catch (error) {
console.error('「改善 JVS 开发体验」日志功能运行错误:');
console.error(error);
}
}, 400);
};
window.iconMap = {
列: '',
表: '',
逻: '',
流: '',
};
const css = `
/* 修改模型信息的弹窗宽度 */
body > div > div.el-dialog[aria-label="修改模型"],
body > div > div.el-dialog[aria-label="数据集详情"],
body > div > div.el-dialog[aria-label="数据模型配置"] {
width: 75% !important;
}
/* 列表设计的按钮设置弹窗宽度 */
body > div > div.el-dialog[aria-label="导出"],
body > div > div.el-dialog[aria-label="下载模板"] {
width: 75% !important;
}
/* 逻辑设计,调整画布侧边栏宽度 */
.canvas-tool {
width: 260px !important;
}
/* 逻辑设计,画布列表的图标 */
.canvas-tool-item > svg {
display: none;
}
/* 调整画布右边的按钮位置 */
.itempannel-box {
left: 220px !important;
}
/* 新版 JVS,逻辑设计,把浮动操作栏的宽度减小到 45% */
.tool-bar {
width: 45% !important;
}
/* 新版 JVS,逻辑设计,组件详情底部的按钮,设置高度,原本 32px,改成 72px */
#node_detailpannel > div.block-container > div > form > div.el-row > div.form-item-btn.el-col.el-col-24 > div > div {
.el-button,
span {
height: 72px !important;
width: 90px !important;
}
}
/* 逻辑设计,展开组件库的组件名称(换行) */
.left-tool-list {
li.getItem > span {
white-space: normal !important;
}
}
/* 新版应用中心,移除每个分类末尾的透明方块(影响点击) */
#app > div > div > div.jvs-layout.jvs-layout-tempOpen > div.template-content-box > div > div.el-col.el-col-10.el-col-md-6.el-col-lg-18.el-col-xl-3 > div > div:nth-child(1) > div:nth-child(3) > div:nth-child(3) > img {
display: none !important;
}
/* 放大公式 icon,原本 16px */
.add-formula-svg {
width: 22px !important;
height: 22px !important;
}
/* 移除新版逻辑设计右下角遮挡按钮的透明条 */
.cont-box-right > .tool-bar {
display: none !important;
}
/* 逻辑设计,展开组件名称 */
.ef-node-text,
.canvas-tool-item {
white-space: normal !important;
width: 100% !important;
color: #363b4c !important;
}
/* 逻辑设计,调整页面设置和已使用逻辑的宽度 */
.content-box:has(.page-setting),
.content-box:has(.used-logic) {
width: 80% !important;
margin-left: 10% !important;
}
/* 逻辑设计,设计名称输入框 */
#app > div > div > div.design-header-box > div.header-left > div.el-input.el-input--mini > input.el-input__inner {
width: 300px !important;
}
/* 逻辑设计,设计名称编辑 icon */
#app > div > div > div.design-header-box > div.header-left > span > svg {
width: 22px !important;
height: 22px !important;
}
/* 自己加的组件 */
/* 日志弹窗的表格 */
.log-11ze-table-tr > td,
.log-11ze-table-tr > th {
}
.log-11ze-table-tr:hover {
background-color: #d5e3fb; /* 悬停时的背景颜色 */
}
.log-11ze-select-container {
display: inline-block;
margin-right: 10px;
}
.log-11ze-select-label {
display: inline-block;
margin-right: 5px;
}
.log-11ze-select {
display: inline-block;
}
/* 按钮统一样式 */
.button-11ze {
background-color: white !important;
border-color: #d4e3fc !important;
color: black !important;
border: 1px solid #e0e0e0 !important;
}
`;
GM_addStyle(css);