// ==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.27 // @description 2025-04-28 // @downloadURL https://update.greasyfork.icu/scripts/527392/%E6%94%B9%E5%96%84%20JVS%20%E5%BC%80%E5%8F%91%E4%BD%93%E9%AA%8C.user.js // @updateURL https://update.greasyfork.icu/scripts/527392/%E6%94%B9%E5%96%84%20JVS%20%E5%BC%80%E5%8F%91%E4%BD%93%E9%AA%8C.meta.js // ==/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, applicationSetClick, ]; 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('')[1]; if (!splitText) { continue; } splitText = splitText.split('center;">')[1]; if (!splitText) { continue; } splitText = splitText.split('<')[0]; return splitText.trim(); } const splitText = text.split('')[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; } } 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) { // 从 otherRule 里拿到 span 标签的 title 属性,内容是逻辑设计的名称 const title = otherRule.querySelector('span').title.trim(); if (title === logicName) { 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 tabType = window.getTabType(); if (tabType === '列表设计') { return; } 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. (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.appNameMapKey = '__11ze_JVS_APP_NAME_MAP__'; function getAppNameMap() { return JSON.parse(localStorage.getItem(window.appNameMapKey) ?? '{}'); } window.getAppNameMap = getAppNameMap; function getAppIdName(jvsAppId) { const appNameMap = getAppNameMap(); return appNameMap[jvsAppId] ?? ''; } window.getAppIdName = getAppIdName; function saveAppIdName(jvsAppId, appName) { const appNameMap = getAppNameMap(); appNameMap[jvsAppId] = appName; localStorage.setItem(window.appNameMapKey, JSON.stringify(appNameMap)); } function applicationSetClick() { const applicationElements = document.querySelectorAll('div.application'); if (applicationElements.length === 0) { return; } applicationElements.forEach(function (appElement) { if (appElement.classList.contains('set-click-11ze')) { return; } appElement.addEventListener('click', function (event) { const clickedElement = event.currentTarget; const idElement = clickedElement.querySelector('label.el-checkbox span.el-checkbox__label'); const nameElement = clickedElement.querySelector('p'); let applicationId = null; let applicationName = null; if (idElement) { applicationId = idElement.textContent.trim(); } else { console.warn('11ze 未找到 ID 元素:', clickedElement); } if (nameElement) { applicationName = nameElement.textContent.trim(); } else { console.warn('11ze 未找到名称元素:', clickedElement); } if (applicationId !== null && applicationName !== null) { saveAppIdName(applicationId, applicationName); } else { console.warn('11ze 应用中心点击应用未能完整获取点击的应用信息:', clickedElement); } }); appElement.classList.add('set-click-11ze'); }); } })(); /** * 记录和查看开发日志 */ 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 []; } logs.forEach((log) => { if (!log.appName || log.appName.length > 100 || log.appName === log.designName) { log.appName = window.getAppIdName(log.jvsAppId); } }); 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(` ${urlParams.id}   ${appName}   ${mode.replace('模式', '')}   ${currentType.shortname}   ${designName}   ${oneLog.type}   ${datetime}   打开   `); } const logTable = document.createElement('table'); logTable.className = 'table'; logTable.innerHTML = ` 设计 id   应用   模式   类型   名称   操作   时间   操作   ${listContent.join('')} `; 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) { // 设计页面没有应用名称时会拿到逻辑设计列表 if (newLog.appName.length > 100 || newLog.appName === newLog.designName) { newLog.appName = window.getAppIdName(newLog.jvsAppId); } const needToSave = window.savedLogDesignName !== newLog.designName || window.savedLogAppName !== newLog.appName; 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);