改善 JVS 开发体验

2025-03-04

// ==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 = '&nbsp;&nbsp;&nbsp;&nbsp;';
        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} &nbsp; </td>
          <td style="color: ${appColor}"> ${appName} &nbsp; </td>
          <td style="color: ${modeColor}"> ${mode.replace('模式', '')} &nbsp; </td>
          <td style="color: ${currentType.color}"> ${currentType.shortname} &nbsp; </td>
          <td> ${designName} &nbsp; </td>
          <td style="color: ${logFieldColor}"> ${oneLog.type} &nbsp; </td>
          <td> ${datetime} &nbsp; </td>
          <td> <a href="${oneLog.url}" target="_blank">打开</a> &nbsp; </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 &nbsp;</th>
          <th> 应用 &nbsp;</th>
          <th> 模式 &nbsp;</th>
          <th> 类型 &nbsp;</th>
          <th> 名称 &nbsp;</th>
          <th> 操作 &nbsp;</th>
          <th> 时间 &nbsp;</th>
          <th> 操作 &nbsp;</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);