网页定制(ChatGPT版)

ChatGPT助手,支持搜索增强、选中文本拓展、总结文章以及定制网页!网页版地址:https://chatgpt-echo.zeabur.app/ ,可关注微信公众号秋博士,获取ChatGPT访问密码

// ==UserScript==
// @name           网页定制(ChatGPT版)
// @namespace      http://tampermonkey.net/
// @version        v1.3.1
// @author         yiqiuzheng
// @description    ChatGPT助手,支持搜索增强、选中文本拓展、总结文章以及定制网页!网页版地址:https://chatgpt-echo.zeabur.app/ ,可关注微信公众号秋博士,获取ChatGPT访问密码
// @icon           
// @match          http*://*/*
// @run-at         document-end
// @require        https://cdn.staticfile.org/vue/2.7.0/vue.min.js
// @require        https://unpkg.com/element-ui/lib/index.js
// @require        https://cdn.jsdelivr.net/npm/[email protected]/dist/showdown.min.js
// @grant          GM_log
// @grant          GM_notification
// @grant          unsafeWindow
// @grant          GM_addStyle
// @grant          GM_setValue
// @grant          GM_getValue
// @grant          GM_setClipboard
// @grant          GM_xmlhttpRequest
// @grant          GM_addElement
// @grant          GM_openInTab
// @grant          GM_getResourceText
// @grant          GM_registerMenuCommand
// @grant          GM_info
// @license        MIT
// ==/UserScript==

(function () {
  'use strict';

  const $ = (selector) => {
    return document.querySelector(selector);
  };

  const genLink = (url) => {
    const linkEl = document.createElement('link');
    linkEl.rel = 'stylesheet';
    linkEl.href = url;
    $('head').insertBefore(linkEl, $('head').firstChild);
  }
  genLink('https://cdnjs.cloudflare.com/ajax/libs/element-ui/2.15.13/theme-chalk/index.min.css')
  genLink('<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/github-markdown-css/5.1.0/github-markdown.css">')

  // 聊天按钮样式
  GM_addStyle(`
    #chatBtn.continue-chat {
        height: 32px;
        border-radius: 16px;
        font-weight: 400;
    }
    #chatBtn.primary-outline-button.btn {
        background: #ffffff;
        border: 1px solid rgba(56,114,224,.48);
        border-radius: 6px;
        align-items: center;
        height: 36px;
        display: flex;
        justify-content: center;
        font-weight: 700;
        font-size: 14px;
        color: #3872e0;
        line-height: 24px;
    }
    #chatBtn.primary-outline-button.btn:hover {
        background: #eff3ff;
    }
    .monica-btn {
        box-sizing: border-box;
        padding: 0 15px;
        cursor: pointer;
        user-select: none;
        -webkit-user-drag: none;
        white-space: nowrap;
    }
    .continue-chat .chatgpt-text {
        margin-left: 8px;
    }
  `);

  GM_addStyle(`
      .btn-area .btn-box {
          display: flex;
          flex-direction: row;
          align-items: center;
          justify-content: center;
          font-size: 13px;
          box-sizing: border-box;
          border: 1px solid #611fec;
          box-shadow: 0 2px #611fec;
          border-radius: 6px;
          background: rgba(255,255,255,.45);
          transition: background-color .2s ease,top .1s ease,box-shadow .1s ease;
          color: #212b36;
          cursor: pointer;
          user-select: none;
          -webkit-user-drag: none;
          position: relative;
          top: -1px;
      }
      .btn-area .btn-box .title {
          flex: 1;
          height: 100%;
          line-height: 24px;
          font-weight: 700;
          box-sizing: border-box;
          padding: 0 4px 0 8px;
          min-width: 0;
          max-width: 150px;
          white-space: nowrap;
          overflow: hidden;
          text-overflow: ellipsis;
          text-align: center;
          border-radius: 4px 0 0 4px;
          transition: all .2s ease-in-out;
      }
      .btn-box .title:hover {
          background: rgba(235,202,254,.24);
          text-decoration: none!important;
      }
    `);

  // 公用样式类
  GM_addStyle(`
    .flex {
      display: flex;
    }
    .flex-1 {
      flex: 1;
    }
    .flex-0.2 {
      flex: 0.2
    }
    .flex-8 {
      flex: 8
    }
    .items-center {
      align-items: center;
    }
    .justify-center {
      justify-content: center;
    }
    .chatgpt-mb-1 {
      margin-bottom: 10px;
    }
    .cursor-pointer {
      cursor: pointer;
    }
  `);

  let vm = null;
  let dialogVm = null;

  // 本地调试ChatGPT的iframe
  //const domain = 'http://localhost:3000/#/';
  // 线上ChatGPT的iframe地址
  const domain = 'https://chatgpt-echo.zeabur.app/';

  const isDomain = (platform) => {
    return window.location.host.includes(platform);
  };

  // 设置油猴API,方便油猴脚本助手直接运行代码
  const setGMAPI = () => {
    unsafeWindow.GM_log = GM_log;
    unsafeWindow.GM_notification = GM_notification;
    unsafeWindow.unsafeWindow = unsafeWindow;
    unsafeWindow.GM_addStyle = GM_addStyle;
    unsafeWindow.GM_setValue = GM_setValue;
    unsafeWindow.GM_getValue = GM_getValue;
    unsafeWindow.GM_setClipboard = GM_setClipboard;
    unsafeWindow.GM_xmlhttpRequest = GM_xmlhttpRequest;
    // unsafeWindow.GM_addElement = GM_addElement;
    unsafeWindow.GM_openInTab = GM_openInTab;
    unsafeWindow.GM_getResourceText = GM_getResourceText;
    unsafeWindow.GM_registerMenuCommand = GM_registerMenuCommand;
    unsafeWindow.GM_info = GM_info;
  };
  setGMAPI();

  const getSearchContent = () => {
    if (isDomain('baidu')) {
      return $('#kw').value;
    } else if (isDomain('google')) {
      return $('#APjFqb').innerHTML;
    } else {
      return '';
    }
  };

  // 默认选项
  var defaultOptions = {
    apiKey: '',
    model: 'microsoft/DialoGPT-large',
    customization: [],
  };

  // 获取配置选项,并使用默认选项填充未定义的选项
  var getConfig = function () {
    var config = JSON.parse(localStorage.getItem('pageCustomizationConfig'));
    if (config) {
      config = Object.assign({}, defaultOptions, config);
    } else {
      config = defaultOptions;
    }
    return config;
  };

  // 保存新的配置选项
  var saveConfig = function (config) {
    localStorage.setItem('pageCustomizationConfig', JSON.stringify(config));
  };

  // 初始化配置
  var initConfig = function () {
    var config = getConfig();
    var apiKeyInput = document.querySelector('#api-key');
    if (apiKeyInput) {
      // “API密钥”输入框
      apiKeyInput.value = config.apiKey;
      apiKeyInput.addEventListener('change', function () {
        config.apiKey = apiKeyInput.value;
        saveConfig(config);
      });
    }
    var modelSelect = document.querySelector('#model-select');
    if (modelSelect) {
      // “交互模型”下拉框
      modelSelect.value = config.model;
      modelSelect.addEventListener('change', function () {
        config.model = modelSelect.value;
        saveConfig(config);
      });
    }
    var customizationInput = document.querySelector(
      '#customization-user-input'
    );
    if (customizationInput) {
      // “对网页的定制要求”输入框
      customizationInput.value = config.customization.join('\n');
      customizationInput.addEventListener('change', function () {
        config.customization = customizationInput.value
          .split('\n')
          .map(function (item) {
            return item.trim();
          });
        saveConfig(config);
      });
    }

    var saveButton = document.querySelector('#save-button');
    if (saveButton) {
      // “保存”按钮
      saveButton.addEventListener('click', function () {
        const userInput = document.querySelector('#customization-user-input');
        const chatHistory = document.querySelector(
          '#customization-chat-history'
        );
        var message = userInput.value.trim();
        if (message === '') {
          alert('请先输入定制要求!');
          return;
        }
        // 清空用户输入框
        userInput.value = '';
        // 在历史记录区域中显示用户发送的消息
        addUserMessage(message);
        var config = getConfig();
        getCode(config)
          .then(function (code) {
            const res = addCustomization(code);
            addAiMessage(code);
          })
          .catch(function (error) {
            console.log(error);
          });

        // 提示保存成功
        var successMessage = document.createElement('div');
        successMessage.innerHTML = '保存成功';
        successMessage.style.position = 'fixed';
        successMessage.style.top = '50%';
        successMessage.style.left = '50%';
        successMessage.style.transform = 'translate(-50%, -50%)';
        successMessage.style.backgroundColor = '#fff';
        successMessage.style.border = '2px solid #ddd';
        successMessage.style.padding = '10px';
        document.body.appendChild(successMessage);

        // 3 秒后移除提示信息
        setTimeout(function () {
          document.body.removeChild(successMessage);
        }, 3000);
      });
    }
  };

  // 获取chatgpt生成的代码
  var getCode = function (config) {
    // 调用chatgpt接口,获取生成的代码
    var url = '';
    var headers = {
      'Content-Type': 'application/json',
    };
    var promise = fetch(url, {
      method: 'POST',
      headers: headers,
      body: JSON.stringify({
        messages: Array.isArray(config.customization.join(''))
          ? config.customization.join('')
          : [
              {
                role: 'user',
                content: config.customization.join('')
                  ? config.customization.join('') +
                    '生成对应js代码,直接返回代码即可'
                  : config.customization.join(''),
              },
            ],
      }),
    })
      .then(function (response) {
        if (!response.ok) {
          throw new Error('chatgpt接口调用出错');
        }
        return response.text();
      })
      .then(function (data) {
        var code = data;
        // 替换特殊字符,以便在脚本里使用
        const regex = /(?<=\`).*?(?=\`)/g;
        const matches = code.match(regex);

        return matches ? matches[0] : code;
      });
    return promise;
  };
  // 创建用户界面
  var createUI = function () {
    // 创建悬浮按钮
    var button = document.createElement('div');
    button.id = 'customization-button';
    button.style.position = 'fixed';
    button.style.bottom = '20px';
    button.style.right = '20px';
    button.style.width = '50px';
    button.style.height = '50px';
    button.style.borderRadius = '50%';
    button.style.backgroundColor = '#409EFF';
    button.style.color = '#fff';
    button.style.fontSize = '24px';
    button.style.textAlign = 'center';
    button.style.lineHeight = '50px';
    button.style.cursor = 'pointer';
    button.textContent = '⚙';

    // 创建侧边栏抽屉
    var drawer = document.createElement('div');
    drawer.id = 'customization-drawer';
    drawer.style.position = 'fixed';
    drawer.style.top = '0';
    drawer.style.right = '-500px';
    drawer.style.width = '500px';
    drawer.style.height = '100%';
    drawer.style.backgroundColor = '#f0f0f0';
    drawer.style.transition = 'right 0.3s ease';
    drawer.style.overflowY = 'auto';
    drawer.style.zIndex = '2000';

    // 创建关闭按钮
    var closeButton = document.createElement('div');
    closeButton.id = 'customization-close-button';
    closeButton.style.position = 'absolute';
    closeButton.style.top = '10px';
    closeButton.style.right = '10px';
    closeButton.style.width = '20px';
    closeButton.style.height = '20px';
    closeButton.style.borderRadius = '50%';
    closeButton.style.backgroundColor = '#aaa';
    closeButton.style.color = '#fff';
    closeButton.style.fontSize = '12px';
    closeButton.style.textAlign = 'center';
    closeButton.style.lineHeight = '20px';
    closeButton.style.cursor = 'pointer';
    closeButton.style.zIndex = '10000';
    closeButton.textContent = '×';
    closeButton.addEventListener('click', function () {
      drawer.style.right = '-500px';
    });

    // 创建聊天室
    var chatRoom = document.createElement('div');
    chatRoom.id = 'customization-chat-room';
    chatRoom.style.height = '100%';

    chatRoom.innerHTML = `
      <el-tabs v-model="activeName"  @tab-click="handleClick">
        <el-tab-pane label="ChatGPT" name="chatgpt">
          <div id="chatgpt-box" class="tab-item">
            <iframe id="chatgpt-iframe" src="${domain}"></iframe>
          </div>
        </el-tab-pane>
        <el-tab-pane label="更新日志" name="log">
          <div class="tab-item">
          <el-result subTitle="关注微信公众号:秋博士,获取访问密码">
            <template slot="icon">
              <div class="flex">
                <el-image src="https://i.postimg.cc/dtWjSGDN/me.jpg"></el-image>
              </div>
            </template>
            <template slot="extra">
              <el-button @click="window.open('https://greasyfork.org/zh-CN/scripts/467425-%E7%BD%91%E9%A1%B5%E5%AE%9A%E5%88%B6-chatgpt%E7%89%88/feedback', '_blank')" type="primary" size="medium">觉得好用?请给个五星好评⭐️~</el-button>
            </template>
          </el-result>
            <el-collapse v-model="activeCollapse">
              <el-collapse-item :key="version.id" v-for="(version, index) in logs" :title="version.title" :name="version.id">
                <div :key="detail" v-for="detail in version.content">{{detail}}</div>
              </el-collapse-item>
            </el-collapse>
          </div>
        </el-tab-pane>
        <el-tab-pane name="plugins">
            <el-badge label="插件管理"  slot="label" :is-dot="isGetNew" class="tab-item-badge">
                插件管理
            </el-badge>

            <div v-if="localCode.length > 0" class="tab-item plugins-box">
              <el-alert
                class="chatgpt-mb-1"
                title="勾选即下次进入网页自动生效,单击按钮即单次执行生效"
                type="success"
                show-icon>
              </el-alert>
              <el-collapse>
                <el-collapse-item :key="item.key" v-for="item in localCode">
                <template slot="title">
                    <el-checkbox @change="(isChecked) => handleCheckboxChange(isChecked, item)" v-model="item.isChecked">
                      <el-button size="mini" @click.stop="evalCode(item.content)">{{item.name}}</el-button>
                    </el-checkbox>
                  </template>
                  <el-input
                    class="chatgpt-mb-1"
                    type="textarea"
                    autosize
                    v-model="item.content"
                  ></el-input>
                  <div class="flex justify-center chatgpt-mb-1">
                    <el-button class="flex-1" size="mini" @click.stop="updateStorageItem(item)">更新</el-button>
                    <el-button class="flex-1" size="mini" @click.stop="deleteStorageItem(item.key)">删除</el-button>
                  </div>
                </el-collapse-item>
              </el-collapse>
            </div>
            <div v-else class="tab-item plugins-box">
              暂无插件,请在聊天中生成油猴脚本点击运行代码即可生成。
            </div>
          </el-tab-pane>
          <el-tab-pane label="跳转网页版" name="web">
            <div class="tab-item">
              跳转网页版
            </div>
          </el-tab-pane>
      </el-tabs>
    `;
    // 将聊天室添加到侧边栏抽屉中
    drawer.appendChild(closeButton);
    drawer.appendChild(chatRoom);

    // 将悬浮按钮和侧边栏抽屉添加到页面上
    document.body.appendChild(button);
    document.body.appendChild(drawer);

    // 点击悬浮按钮显示侧边栏抽屉
    button.addEventListener('click', function () {
      if (drawer.style.right === '-500px') {
        drawer.style.right = '0';
      } else {
        drawer.style.right = '-500px';
      }
    });

    // element
    GM_addStyle(`
      #customization-chat-room .el-tabs {
        height: inherit;
        background: #e7f8ff;
      }
      #customization-chat-room .el-tabs__active-bar {
        width: 58px!important;
      }

      #customization-chat-room .el-tabs__content {
        height: inherit;
        box-sizing: border-box;
        overflow: auto;
      }

      #customization-chat-room .el-tab-pane {
        height: inherit;
        box-sizing: border-box;
        width: 100%;
      }
      #customization-chat-room .el-tabs__header {
        margin: 0 0 10px;
        padding: 0 10px;
      }
      #chatgpt-iframe {
        width: 100%;
        height: inherit;
      }
      #customization-chat-room .el-collapse-item__header {
        padding: 0 10px;
      }
      #customization-chat-room .el-collapse-item__wrap {
        padding: 0 10px;
      }
      #customization-chat-room .tab-item-badge .el-badge__content.is-fixed {
        top: 5px;
      }
    `);

    // 美化页面
    GM_addStyle(`
            #customization-chat-room .tab-item {
              padding: 10px;
              height: 100%;
            }
            #customization-chat-room .match-item {
              width: 100%;
            }
            #customization-chat-room {
                border-left: 1px solid #e5e8eb;
            }
            iframe {
                border: medium none;
            }
            #customization-button:hover {
                background-color: #66b1ff;
            }

            #customization-close-button:hover {
                background-color: #ff7875;
            }

            #customization-drawer::-webkit-scrollbar {
                width: 8px;
                height: 8px;
            }

            #customization-drawer::-webkit-scrollbar-thumb {
                background: #aaa;
                border-radius: 6px;
            }

            #customization-drawer::-webkit-scrollbar-track {
                background: #f0f0f0;
                border-radius: 6px;
            }

            #customization-user-input:focus {
                outline: none;
                border-color: #409eff;
            }
            #save-button:hover {
                background-color: #66b1ff;
            }
            #customization-chat-history {
                display: flex;
                flex-direction: column;
            }
            .customization-message-wrapper {
                margin: 10px;
                display: flex;
                justify-content: flex-end;
            }
            .customization-message-wrapper:hover {
                background-color: #eee;
                cursor: pointer;
            }
            .customization-message-bubble {
                max-width: 80%;
                padding: 10px;
                border-radius: 10px;
                margin-top: 5px;
                margin-bottom: 5px;
                font-size: 16px;
            }
            .customization-user-message {
                align-self: flex-end;
            }
            .customization-ai-message {
                align-self: flex-start;
            }
            .customization-user-message, .customization-ai-message {
                background-color: #409eff;
                color: white;
                padding: 8px;
                border-radius: 10px;
            }
            .customization-save-icon-wrapper {
                position: absolute;
                bottom: -10px;
            }
        `);
  };

  const evalCode = (content) => {
    const myFunc = new Function(content);
    myFunc();
  };

  const showToast = (tips) => {
    // 提示保存成功
    var successMessage = document.createElement('div');
    successMessage.innerHTML = tips;
    successMessage.style.position = 'fixed';
    successMessage.style.top = '50%';
    successMessage.style.left = '50%';
    successMessage.style.transform = 'translate(-50%, -50%)';
    successMessage.style.backgroundColor = '#fff';
    successMessage.style.border = '1px solid #ddd';
    successMessage.style.borderRadius = '10px';
    successMessage.style.padding = '10px';
    successMessage.style.zIndex = '10000';
    document.body.appendChild(successMessage);

    // 3 秒后移除提示信息
    setTimeout(function () {
      document.body.removeChild(successMessage);
    }, 3000);
  };

  // 在历史记录区域中显示用户发送的消息
  var addUserMessage = function (message) {
    const chatHistory = document.querySelector('#customization-chat-history');
    var messageWrapper = document.createElement('div');
    messageWrapper.className = 'customization-message-wrapper';
    messageWrapper.id = 'customization-message-wrapper';
    var messageBubble = document.createElement('div');
    messageBubble.className =
      'customization-message-bubble customization-user-message';
    messageBubble.textContent = message;
    messageWrapper.appendChild(messageBubble);
    chatHistory.appendChild(messageWrapper);
  };

  // 在历史记录区域中显示 AI 回复的消息
  var addAiMessage = function (message) {
    const chatHistory = document.querySelector('#customization-chat-history');
    var messageWrapper = document.createElement('div');
    messageWrapper.className = 'customization-message-wrapper-left';
    var messageBubble = document.createElement('div');
    messageBubble.className =
      'customization-message-bubble customization-ai-message';
    messageBubble.textContent = message;
    messageWrapper.appendChild(messageBubble);
    chatHistory.appendChild(messageWrapper);
  };
  // 添加定制功能
  var addCustomization = function (code) {
    try {
      eval(code);
    } catch (err) {
      console.log(err);
    }
  };

  function copyToClipboard(text) {
    try {
      navigator.clipboard.writeText(text);
      showToast('复制成功');
    } catch (error) {
      const textArea = document.createElement('textarea');
      textArea.value = text;
      document.body.appendChild(textArea);
      textArea.focus();
      textArea.select();
      try {
        document.execCommand('copy');
        showToast('复制成功');
      } catch (error) {
        showToast('复制失败,请手动复制');
      }
      document.body.removeChild(textArea);
    }
  }

  const handleTextAutoHeight = () => {
    var textarea = document.querySelector('textarea');

    textarea.addEventListener('input', (e) => {
      textarea.style.height = '100px';
      textarea.style.height = e.target.scrollHeight + 'px';
    });
  };

  const creatChatBtn = (parentNode) => {
    const btnBox = document.createElement('div');
    btnBox.id = 'chatBtn';
    btnBox.className = 'monica-btn btn continue-chat primary-outline-button';
    btnBox.innerHTML = `
      <svg aria-hidden="true" focusable="false" role="img" class="octicon octicon-comment" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" style="display: inline-block; user-select: none; vertical-align: text-bottom; overflow: visible;"><path d="M1 2.75C1 1.784 1.784 1 2.75 1h10.5c.966 0 1.75.784 1.75 1.75v7.5A1.75 1.75 0 0 1 13.25 12H9.06l-2.573 2.573A1.458 1.458 0 0 1 4 13.543V12H2.75A1.75 1.75 0 0 1 1 10.25Zm1.75-.25a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h2a.75.75 0 0 1 .75.75v2.19l2.72-2.72a.749.749 0 0 1 .53-.22h4.5a.25.25 0 0 0 .25-.25v-7.5a.25.25 0 0 0-.25-.25Z"></path>
      </svg>
      <span class="text">在聊天中继续</span>
    `;

    parentNode && parentNode.appendChild(btnBox);
    $('#chatBtn').addEventListener('click', () => {
      $('#customization-drawer').style.right = 0;
    });
  };

  const creatBtn = ({ parentNode, btnText, clickFn }) => {
    const btnBox = document.createElement('div');
    btnBox.className = 'btn-area';
    btnBox.innerHTML = `
      <div class="btn-box">
        <div class="title">${btnText}</div>
      </div>
    `;
    parentNode && parentNode.appendChild(btnBox);
    btnBox.addEventListener('click', () => {
      clickFn && clickFn();
    });
  };

  // 谷歌搜索ChatGPT应答
  const generateSearchEnhance = () => {
    const messageWrapper = document.createElement('div');
    messageWrapper.className = 'chatgpt-search-enhance';
    messageWrapper.id = 'chatgpt-search-enhance';
    messageWrapper.innerHTML = `
        <div class="header-7QHGYk">
          <div class="lt-znd2I9">
            <a class="title" href="${domain}" target="_blank" rel="noreferrer">
              <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="30" height="30" fill="none" class="user-avatar"><defs><path id="bot_svg__a" d="M0 0h30v30H0z"></path><path id="bot_svg__c" d="M0 0h20.455v20.455H0z"></path></defs><g><rect fill="#E7F8FF" width="30" height="30" rx="10"></rect><mask id="bot_svg__b" fill="#fff"><use xlink:href="#bot_svg__a"></use></mask><g mask="url(#bot_svg__b)"><g transform="translate(4.773 4.773)"><mask id="bot_svg__d" fill="#fff"><use xlink:href="#bot_svg__c"></use></mask><g mask="url(#bot_svg__d)"><path fill-rule="evenodd" d="M19.11 8.37c.17-.52.26-1.06.26-1.61 0-.9-.24-1.79-.71-2.57a5.24 5.24 0 0 0-4.53-2.59c-.37 0-.73.04-1.09.11A5.201 5.201 0 0 0 9.17 0h-.04C6.86 0 4.86 1.44 4.16 3.57A5.11 5.11 0 0 0 .71 6.04C.24 6.83 0 7.72 0 8.63c0 1.27.48 2.51 1.35 3.45-.18.52-.27 1.07-.27 1.61 0 .91.25 1.8.71 2.58 1.13 1.94 3.41 2.94 5.63 2.47a5.18 5.18 0 0 0 3.86 1.71h.05c2.26 0 4.27-1.44 4.97-3.57a5.132 5.132 0 0 0 3.45-2.47c.46-.78.7-1.67.7-2.58 0-1.28-.48-2.51-1.34-3.46ZM8.947 18.158c-.04.03-.08.05-.12.07.7.58 1.57.89 2.48.89h.01c2.14 0 3.88-1.72 3.88-3.83v-4.76c0-.02-.02-.04-.04-.05l-1.74-.99v5.75c0 .23-.13.45-.34.57l-4.13 2.35Zm-.67-1.153 4.17-2.38c.02-.01.03-.03.03-.05v-1.99l-5.04 2.87c-.21.12-.47.12-.68 0l-4.13-2.35c-.04-.02-.09-.06-.12-.07-.04.21-.06.43-.06.65 0 .67.18 1.33.52 1.92v-.01c.7 1.19 1.98 1.92 3.37 1.92.68 0 1.35-.18 1.94-.51ZM3.903 5.168v-.14c-.85.31-1.57.9-2.02 1.68a3.78 3.78 0 0 0-.52 1.91c0 1.37.74 2.64 1.94 3.33l4.17 2.37c.02.01.04.01.06 0l1.75-1-5.04-2.87a.64.64 0 0 1-.34-.57v-4.71Zm13.253 3.337-4.18-2.38c-.02 0-.04 0-.06.01l-1.74.99 5.04 2.87c.21.12.34.34.34.58v4.85c1.52-.56 2.54-1.99 2.54-3.6 0-1.37-.74-2.63-1.94-3.32ZM8.014 5.83c-.02.01-.03.03-.03.05v1.99L13.024 5a.692.692 0 0 1 .68 0l4.13 2.35c.04.02.08.05.12.07.03-.21.05-.43.05-.65 0-2.11-1.74-3.83-3.88-3.83-.68 0-1.35.18-1.94.51l-4.17 2.38Zm1.133-4.492c-2.15 0-3.89 1.72-3.89 3.83v4.76c0 .02.02.03.03.04l1.75 1v-5.75c0-.23.13-.45.34-.57l4.13-2.35c.04-.03.09-.06.12-.07-.7-.58-1.58-.89-2.48-.89ZM7.983 11.51l2.24 1.27 2.25-1.27V8.95l-2.25-1.28-2.24 1.28v2.56Z" style="fill: rgb(31, 148, 140);"></path></g></g></g></g></svg>
              <span>ChatGPT</span>
            </a>
          </div>
          <div id="header-right" class="header-right">
            <div id="toolbar" class="toolbar">
              <el-tooltip class="item" effect="dark" content="Top Left 提示文字" placement="top-start">
                <span title="复制" class="toolbar-item-EhZYV5">
                  <svg id="copy" width="16" height="16" fill="none" viewBox="0 0 16 16" style="min-width: 16px; min-height: 16px;"><g><path data-follow-stroke="currentColor" d="M4.334 4.145v-1.54c0-.517.42-.937.937-.937h8.125c.518 0 .938.42.938.937v8.125c0 .518-.42.938-.938.938H11.84" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round"></path><path data-follow-stroke="currentColor" d="M10.729 4.332H2.604a.937.937 0 0 0-.938.938v8.125c0 .517.42.937.938.937h8.125c.517 0 .937-.42.937-.938V5.27a.937.937 0 0 0-.938-.938Z" stroke="currentColor" stroke-width="1.2" stroke-linejoin="round"></path></g>
                  </svg>
                </span>
              </el-tooltip>
            </div>

          </div>
        </div>
        <div id="answer" class="answer">
        </div>
        <div id="footer" class="footer">
        </div>
    `;
    GM_addStyle(`
      .chatgpt-search-enhance {
        border-radius: 8px;
        border: 1px solid #dadce0;
        padding: 0 16px;
        overflow: auto;
      }
      .chatgpt-search-enhance .header-7QHGYk {
          height: 56px;
          display: flex;
          box-sizing: border-box;
      }
      .header-7QHGYk .lt-znd2I9 .title span {
        font-size: 15px;
        color: #000;
        font-weight: 700;
    }
      .chatgpt-search-enhance .header-7QHGYk .lt-znd2I9 {
          display: inline-flex;
          align-items: center;
          gap: 6px;
      }
      .chatgpt-search-enhance .header-7QHGYk .lt-znd2I9 .title {
          display: inline-flex;
          align-items: center;
          cursor: pointer;
          gap: 6px;
      }
      .chatgpt-search-enhance .header-7QHGYk .header-right {
          flex: 1;
          display: flex;
          justify-content: flex-end;
          align-items: center;
          gap: 8px;
      }
      .chatgpt-search-enhance .header-7QHGYk .header-right .btn-area {
          height: 24px;
      }
      .chatgpt-search-enhance .toolbar {
          display: none;
          align-items: center;
          gap: 8px;
          color: #595959;
      }
      .toolbar-item-EhZYV5 svg:hover {
        cursor: pointer;
      }
      .chatgpt-search-enhance .title:hover {
        cursor: pointer;
        user-select: none;
        -webkit-user-drag: none;
        text-decoration: none;
        color: #3872e0!important;
        background-color: transparent;
      }
      .chatgpt-search-enhance .footer {
        box-sizing: border-box;
        display: none;
        align-items: center;
        margin: 16px 0;
      }
    `);
    const searchBox = document.getElementById(
      isDomain('google') ? 'rhs' : 'content_right'
    );
    searchBox?.appendChild(messageWrapper);
    creatChatBtn($('#footer'));
    creatBtn({
      parentNode: $('#header-right'),
      btnText: '询问ChatGPT',
      clickFn: () => {
        postMsg({
          type: 'search',
          content: getSearchContent(),
        });
        $('#answer').innerHTML = '加载中...';
        $('#copy').addEventListener('click', () => {
          const text = $('#answer').innerText;
          copyToClipboard(text);
        });
        $('#footer').style.display = 'flex';
      },
    });
  };


  //md转换
  function mdConverter(rawData) {
    var converter = new showdown.Converter(); //增加拓展table
    converter.setOption('tables', true); //启用表格选项。从showdown 1.2.0版开始,表支持已作为可选功能移入核心拓展,showdown.table.min.js扩展已被弃用
    var view = converter.makeHtml(rawData);
    return view;
  }

  // 流式输出递归更新节点
  function updateNode(element, target) {
    //如果是标签节点
    if (element.nodeType == 1 && element.outerHTML != target.outerHTML) {
      target.parentNode.replaceChild(element.cloneNode(true), target)
      //如果是文本节点
    } else if (element.nodeType == 3 && element.nodeValue != target.nodeValue) {
      target.parentNode.replaceChild(element.cloneNode(true), target)
    }
  }
  function diffAndUpdate(target, newContent) {
    var array = newContent.childNodes
    var org = target.childNodes
    for (let index = 0; index < org.length; index++) {
      if (!array[index]) {
        org[index].parentNode.removeChild(org[index])
      }

      if (array[index] && array[index].cloneNode().outerHTML != org[index].cloneNode().outerHTML) {
        org[index].parentNode.replaceChild(array[index].cloneNode(), org[index])
      }

    }
    for (let index = 0; index < array.length; index++) {
      const element = array[index];

      if (org[index]) {
        //如果有子节点,递归
        if (element.childNodes && element.childNodes.length > 0) {
          //首先要把当前节点更改成空节点
          //  if (org[index].nodeType == 1 && org[index].outerHTML != element.outerHTML) {
          //   org[index].parentNode.replaceChild(element.cloneNode(), org[index])
          //  }
          diffAndUpdate(org[index], element)
        } else
          updateNode(element, org[index])

      } else {
        //新增
        if (target.nodeType == 1)
          target.appendChild(element.cloneNode(true))

      }
    }

  }
  function demandInnerHTML(target, html) {
    var copy = document.createElement('div')
    copy.insertAdjacentHTML('beforeend', html)
    diffAndUpdate(target, copy)
  }
  
  // 谷歌搜索ChatGPT应答
  const handleGoogleSearch = () => {
    window.addEventListener('load', () => {
      // generateSearchEnhance();
    });
  };

  const postMsg = ({ type, ...data }) => {
    const chatgptIframe = $('#chatgpt-iframe');
    chatgptIframe.contentWindow.postMessage(
      { data, type, origin: 'parent' },
      domain
    );
  };
  const extractArticle = () => {
    function isText(node) {
      return node.nodeType === Node.TEXT_NODE;
    }

    function getTextContent(node) {
      return node.textContent.trim();
    }

    function isArticleNode(node) {
      if (node.nodeName === 'P' || /^H[1-6]$/.test(node.nodeName)) {
        return getTextContent(node) !== '';
      }

      // if (node.nodeName === 'UL' || node.nodeName === 'OL') {
      //   return node.querySelector('li') !== null;
      // }

      if (node.nodeName === 'DIV') {
        return node.classList.contains('page-main_content') || node.classList.contains('content-main');
      }

      if (node.nodeName === 'SECTION' || node.nodeName === 'ARTICLE') {
        return node.querySelector('p,h1,h2,h3,h4,h5,h6,ul,ol,div') !== null;
      }

      return false;
    }

    function getArticleNodes() {
      const articleNodes = [];

      function traverse(node) {
        if (isArticleNode(node)) {
          articleNodes.push(node);
        }

        node.childNodes.forEach(traverse);
      }

      traverse(document.body);

      return articleNodes;
    }

    function getArticleText() {
      const articleNodes = getArticleNodes();

      const articleText = articleNodes.map(node => getTextContent(node)).join('\n');

      return articleText.replace(/\s+/g, ' ').trim();
    }

    const result = getArticleText();

    const splitStr = (str) => {
      const maxLength = 4400;
      const strArr = [];
      let temp = '';
      for (let i = 0; i < str.length; i++) {
        temp += str[i];
        if ((i + 1) % maxLength === 0 || i === str.length - 1) {
          strArr.push(temp);
          temp = '';
        }
      }
      return strArr;
    };

    const formatResult = splitStr(result);

    return formatResult;
  };


  // 计算输入框大小
  // input:(必填)输入框
  // rowh:(必填)当没有内容时默认高度,如果不传当没有内容时会没高度
  // colw:(可选)当没有内容时默认宽度,如果不传宽度会很窄
  function inputSizeChange(textarea) {
    let text = textarea.value;
    let lines = text.split('\n');
    let lineHeight = 32;
    let height = lines.length * lineHeight;
    textarea.style.height = height + 'px';
  }

  // 获取油猴脚本信息
  const getCodeInfo = (content) => {
    const match = content.match(/@name\s*([^\n;]+)/);
    const name = match ? match[1].trim() : '未命名脚本';
    return { name };
  };

  // 匹配油猴脚本中生效和不生效的网址
  function getUrls(script) {
    const matchRegex = /(@match|@include)\s*([^]*?)\n/g;
    const blockRegex = /(\S+)\s*>=\s*0\s*\)/g;
    let matches, blocks;
    const urls = {
      matches: [],
      blocks: [],
    };
    while ((matches = matchRegex.exec(script)) !== null) {
      if (matches[1] === '@match') {
        urls.matches.push(matches[2]);
      } else if (matches[1] === '@include') {
        urls.matches.push(matches[2]);
      }
    }
    while ((blocks = blockRegex.exec(script)) !== null) {
      urls.blocks.push(blocks[1]);
    }
    return urls;
  }
  // 处理监听ChatGPT-web的消息
  const handlePostMessage = () => {
    // 父级,在frame处抛出接收事件
    window.addEventListener(
      'message',
      (event) => {
        if (event.data.origin && event.data.origin === 'chatgpt-web') {
          // console.log('chatgpt-web', event.data, GM_addStyle);
          const { content, key } = event.data.data;
          if (event.data.type === 'code') {
            evalCode(content);
            const chatgptKey = `chatgpt-${key}`;
            const item = localStorage.getItem(chatgptKey);
            const { name } = getCodeInfo(content);
            const storageItem = {
              content,
              key: chatgptKey,
              isChecked: true,
              name,
              ...getUrls(content),
            };
            localStorage.setItem(chatgptKey, JSON.stringify(storageItem));
            vm.updateStorageItem(storageItem);
            vm.isGetNew = true;
          }
          if (event.data.type === 'read') {
            const article = extractArticle();
            postMsg({ type: 'read', content: article });
          }
          if (event.data.type === 'ready' && (isDomain('google') || isDomain('baidu'))) {
            const rId = isDomain('google') ? 'rhs' : 'content_right'

            const searchBox = document.getElementById(rId);
            const contentBox = document.getElementById(
              isDomain('google') ? 'rcnt' : 'container'
            );
            if(!searchBox) {
              const rIdBox = document.createElement('rId');
              rIdBox.id = rId;
              contentBox?.appendChild(rIdBox);
            }
            !$('#chatgpt-search-enhance') && generateSearchEnhance();
          }
          if (event.data.type === 'search') {
            // $('#answer').innerHTML = mdConverter(event.data.data.content);
            demandInnerHTML($('#answer'), mdConverter(event.data.data.content))
            $('#toolbar').style.display = 'inline-flex';
          }
          if (event.data.type === 'selectText') {
            // $('#selectResult').value = event.data.data.content;
            // $('#selectResult').addec;
            // inputSizeChange($('#selectResult'));
            dialogVm.result =  event.data.data.content;
          }
        }
      },
      false
    );
  };


  // 选中文本的弹出框
  function createDialogBox() {
    const dialogBox = document.createElement('div');
    // dialogBox.id="floatDialog"
    dialogBox.innerHTML = `
      <div id="floatDialog" v-show="isShowDialog">
        <div v-show="isShowBrief" id="briefContent" class="brief-content">
          <div class="brief-title lt-znd2I9">
            <a class="title"  target="_blank" rel="noreferrer">
              <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="30" height="30" fill="none" class="user-avatar"><defs><path id="bot_svg__a" d="M0 0h30v30H0z"></path><path id="bot_svg__c" d="M0 0h20.455v20.455H0z"></path></defs><g><rect fill="#E7F8FF" width="30" height="30" rx="10"></rect><mask id="bot_svg__b" fill="#fff"><use xlink:href="#bot_svg__a"></use></mask><g mask="url(#bot_svg__b)"><g transform="translate(4.773 4.773)"><mask id="bot_svg__d" fill="#fff"><use xlink:href="#bot_svg__c"></use></mask><g mask="url(#bot_svg__d)"><path fill-rule="evenodd" d="M19.11 8.37c.17-.52.26-1.06.26-1.61 0-.9-.24-1.79-.71-2.57a5.24 5.24 0 0 0-4.53-2.59c-.37 0-.73.04-1.09.11A5.201 5.201 0 0 0 9.17 0h-.04C6.86 0 4.86 1.44 4.16 3.57A5.11 5.11 0 0 0 .71 6.04C.24 6.83 0 7.72 0 8.63c0 1.27.48 2.51 1.35 3.45-.18.52-.27 1.07-.27 1.61 0 .91.25 1.8.71 2.58 1.13 1.94 3.41 2.94 5.63 2.47a5.18 5.18 0 0 0 3.86 1.71h.05c2.26 0 4.27-1.44 4.97-3.57a5.132 5.132 0 0 0 3.45-2.47c.46-.78.7-1.67.7-2.58 0-1.28-.48-2.51-1.34-3.46ZM8.947 18.158c-.04.03-.08.05-.12.07.7.58 1.57.89 2.48.89h.01c2.14 0 3.88-1.72 3.88-3.83v-4.76c0-.02-.02-.04-.04-.05l-1.74-.99v5.75c0 .23-.13.45-.34.57l-4.13 2.35Zm-.67-1.153 4.17-2.38c.02-.01.03-.03.03-.05v-1.99l-5.04 2.87c-.21.12-.47.12-.68 0l-4.13-2.35c-.04-.02-.09-.06-.12-.07-.04.21-.06.43-.06.65 0 .67.18 1.33.52 1.92v-.01c.7 1.19 1.98 1.92 3.37 1.92.68 0 1.35-.18 1.94-.51ZM3.903 5.168v-.14c-.85.31-1.57.9-2.02 1.68a3.78 3.78 0 0 0-.52 1.91c0 1.37.74 2.64 1.94 3.33l4.17 2.37c.02.01.04.01.06 0l1.75-1-5.04-2.87a.64.64 0 0 1-.34-.57v-4.71Zm13.253 3.337-4.18-2.38c-.02 0-.04 0-.06.01l-1.74.99 5.04 2.87c.21.12.34.34.34.58v4.85c1.52-.56 2.54-1.99 2.54-3.6 0-1.37-.74-2.63-1.94-3.32ZM8.014 5.83c-.02.01-.03.03-.03.05v1.99L13.024 5a.692.692 0 0 1 .68 0l4.13 2.35c.04.02.08.05.12.07.03-.21.05-.43.05-.65 0-2.11-1.74-3.83-3.88-3.83-.68 0-1.35.18-1.94.51l-4.17 2.38Zm1.133-4.492c-2.15 0-3.89 1.72-3.89 3.83v4.76c0 .02.02.03.03.04l1.75 1v-5.75c0-.23.13-.45.34-.57l4.13-2.35c.04-.03.09-.06.12-.07-.7-.58-1.58-.89-2.48-.89ZM7.983 11.51l2.24 1.27 2.25-1.27V8.95l-2.25-1.28-2.24 1.28v2.56Z" style="fill: rgb(31, 148, 140);"></path></g></g></g></g></svg>
            </a>
          </div>
          <div class="action-bar-lite-window-vuGdUx action-bar-SA76od" style="opacity: 1;">
            <div class="box-NzPGeV">
              <span @click="handleClick('translate')" id="translate" data-id="751059" class="tag-button tag-button-active-y5uMAc" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M16.5 3v9l-3-2.25-3 2.25V3M4.5 20.25V21H18" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M4.5 20.25A2.25 2.25 0 0 1 6.75 18H19.5V3H6.75A2.25 2.25 0 0 0 4.5 5.25v15Z" data-follow-stroke="#000"></path></g></svg>
                <span class="text-wrapper-92ojRl" data-text="翻译">
                  <span class="text-0Pw8ng">翻译</span>
                </span>
              </span>
              <span @click="handleClick('summary')" id="quickSummary" data-id="751059" class="tag-button tag-button-active-y5uMAc" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M16.5 3v9l-3-2.25-3 2.25V3M4.5 20.25V21H18" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M4.5 20.25A2.25 2.25 0 0 1 6.75 18H19.5V3H6.75A2.25 2.25 0 0 0 4.5 5.25v15Z" data-follow-stroke="#000"></path></g></svg>
                <span class="text-wrapper-92ojRl" data-text="总结">
                  <span class="text-0Pw8ng">总结</span>
                </span>
              </span>
            </div>
          </div>
          <el-dropdown id="selectTextDropDown" @command="handleClickDropDown">
            <span class="el-dropdown-link">
              <i class="el-icon-arrow-down el-icon--right"></i>
            </span>
            <el-dropdown-menu slot="dropdown">
              <el-dropdown-item command="translate">翻译</el-dropdown-item>
              <el-dropdown-item command="summary">总结</el-dropdown-item>
              <el-dropdown-item command="explain">解释</el-dropdown-item>
              <el-dropdown-item command="explainCode">解释代码</el-dropdown-item>
              <el-dropdown-item command="grammar">语法</el-dropdown-item>
              <el-dropdown-item command="rewrite">重写</el-dropdown-item>
              <el-dropdown-item command="qa">问答</el-dropdown-item>
              <el-dropdown-item command="expansion">扩写</el-dropdown-item>
            </el-dropdown-menu>
          </el-dropdown>
        </div>

        <div v-show="isShowComplete" id="complateContent">
          <div id="dialogBoxSelectBox">
            <h5 id="dialogBoxSelectTitle">选中文本<h5/>
            <el-input
              id="selectVal"
              class="text-area" 
              type="textarea"
              :autosize="{ minRows: 2, maxRows: 5 }"
              placeholder="请输入内容"
              v-model="selectVal">
            </el-input>
          <div>
          <div id="selectActions">
            <div class="action-bar-lite-window-vuGdUx action-bar-SA76od" style="opacity: 1;">
              <div class="box-NzPGeV">
                <div @click="handleClick('summary')" id="summary" data-id="751061" class="tag-button" style="opacity: 1; pointer-events: auto;">
                  <svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M10.5 10.5h6M10.5 13.5h6M19.5 3.75h-15a.75.75 0 0 0-.75.75v15c0 .414.336.75.75.75h15a.75.75 0 0 0 .75-.75v-15a.75.75 0 0 0-.75-.75ZM7.5 3.75v16.5" data-follow-stroke="#000"></path></g></svg>
                  <div class="text-wrapper-92ojRl" data-text="总结"><div class="text-0Pw8ng">摘要</div></div></div>
                <span @click="handleClick('grammar')"  id="grammar" data-id="1924664" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 32 32" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="m27 26-5-10-5 10M18.428 23.143h7.143M5 18h9M5 23h9M5.006 8h21M5.006 13h21" data-follow-stroke="#000"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="语法"><span class="text-0Pw8ng">语法</span></span></span>
                <span @click="handleClick('explainCode')" id="explainCode" data-id="1924663" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path fill="currentColor" d="M7.8 19.2H7A1.2 1.2 0 0 1 5.8 18v-4.1a1.8 1.8 0 0 0-1.236-1.71L3.987 12l.577-.19A1.8 1.8 0 0 0 5.8 10.1V6A1.2 1.2 0 0 1 7 4.8h.8V3.2H7A2.8 2.8 0 0 0 4.2 6v3.7a1.7 1.7 0 0 1-1.7 1.7h-.3v1.2h.3a1.7 1.7 0 0 1 1.7 1.7V18A2.8 2.8 0 0 0 7 20.8h.8v-1.6ZM16.2 19.2h.8a1.2 1.2 0 0 0 1.2-1.2v-4.1a1.8 1.8 0 0 1 1.236-1.71l.577-.19-.577-.19A1.8 1.8 0 0 1 18.2 10.1V6A1.2 1.2 0 0 0 17 4.8h-.8V3.2h.8A2.8 2.8 0 0 1 19.8 6v3.7a1.701 1.701 0 0 0 1.7 1.7h.3v1.2h-.3a1.7 1.7 0 0 0-1.7 1.7V18a2.8 2.8 0 0 1-2.8 2.8h-.8v-1.6Z" clip-rule="evenodd" fill-rule="evenodd" data-follow-fill="#09121F"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="解释代码"><span class="text-0Pw8ng">解释代码</span></span></span>
                <span @click="handleClick('rewrite')" id="rewrite" data-id="1924662" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M12.75 6 18 11.25M20.25 20.25H9l-5.202-5.202" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M9 20.25H4.5a.75.75 0 0 1-.75-.75v-4.19a.75.75 0 0 1 .22-.53L15.22 3.53a.75.75 0 0 1 1.06 0l4.19 4.19a.75.75 0 0 1 0 1.06L9 20.25Z" data-follow-stroke="#000"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="重写"><span class="text-0Pw8ng">重写</span></span></span>
                <span @click="handleClick('explain')" id="explain" data-id="751059" class="tag-button tag-button-active-y5uMAc" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M16.5 3v9l-3-2.25-3 2.25V3M4.5 20.25V21H18" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M4.5 20.25A2.25 2.25 0 0 1 6.75 18H19.5V3H6.75A2.25 2.25 0 0 0 4.5 5.25v15Z" data-follow-stroke="#000"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="解释"><span class="text-0Pw8ng">解释</span></span></span>
                <span @click="handleClick('translate')" id="translate" data-id="751060" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M21.75 20.25 16.5 9.75l-5.25 10.5M12.75 17.25h7.5M8.25 3v2.25M2.25 5.25h12M11.25 5.25a9 9 0 0 1-9 9" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M5.763 8.25a9.004 9.004 0 0 0 8.486 5.997" data-follow-stroke="#000"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="翻译"><span class="text-0Pw8ng">翻译</span></span></span>
                <span @click="handleClick('qa')"  id="qa" data-id="1924661" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 24 24" style="min-width: 16px; min-height: 16px;"><g><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M20.95 11.625a8.625 8.625 0 0 1-8.625 8.625H3.7v-8.625a8.625 8.625 0 0 1 17.25 0Z" data-follow-stroke="#000"></path><path stroke-linejoin="round" stroke-linecap="round" stroke-width="1.5" stroke="currentColor" d="M12.325 13.62v-1.725a2.587 2.587 0 1 0-2.588-2.588" data-follow-stroke="#000"></path><path fill="currentColor" d="M12.325 17.501a1.078 1.078 0 1 0 0-2.156 1.078 1.078 0 0 0 0 2.156Z" clip-rule="evenodd" fill-rule="evenodd" data-follow-fill="#000"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="问答"><span class="text-0Pw8ng">问答</span></span></span>
                <span @click="handleClick('expansion')" id="expansion" data-id="1924660" class="tag-button" style="opacity: 1; pointer-events: auto;"><svg width="16" height="16" fill="none" viewBox="0 0 16 16" style="min-width: 16px; min-height: 16px;"><g><path data-follow-stroke="#637381" d="M10 3h3v3m-3.5.5L13 3" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"></path><path data-follow-stroke="#37404A" d="M6 13H3v-3m3.5-.5L3 13" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"></path></g></svg>
                  <span class="text-wrapper-92ojRl" data-text="扩写"><span class="text-0Pw8ng">扩写</span></span></span></div>
            </div>
          </div>
          <div>
            <h5 id="dialogBoxResultTitle">结果<h5/>
            <el-input
              class="text-area" 
              id="selectResult"
              type="textarea"
              :autosize="{ minRows: 2, maxRows: 20  }"
              placeholder="请输入内容"
              v-model="result">
            </el-input>
          <div>
          <div class="monica-btn btn continue-chat primary-outline-button" id="chatBtn">
            <svg aria-hidden="true" focusable="false" role="img" class="octicon octicon-comment" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" style="display: inline-block; user-select: none; vertical-align: text-bottom; overflow: visible;"><path d="M1 2.75C1 1.784 1.784 1 2.75 1h10.5c.966 0 1.75.784 1.75 1.75v7.5A1.75 1.75 0 0 1 13.25 12H9.06l-2.573 2.573A1.458 1.458 0 0 1 4 13.543V12H2.75A1.75 1.75 0 0 1 1 10.25Zm1.75-.25a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h2a.75.75 0 0 1 .75.75v2.19l2.72-2.72a.749.749 0 0 1 .53-.22h4.5a.25.25 0 0 0 .25-.25v-7.5a.25.25 0 0 0-.25-.25Z"></path>
            </svg>
            <span class="chatgpt-text">在聊天中继续</span>
          </div>
        </div>
      </div>
    `
    GM_addStyle(`
      #floatDialog {
        position: absolute;
        background-color: white;
        padding: 10px;
        z-index: 999;
        display: block;
        border-radius: 8px;
        border: 1px solid #dadce0;
        left: -1000px;
      }
      #briefContent {
        align-items: center;
        display: flex;
      }
      #briefContent .brief-title {
        margin-right: 10px;
      }
      #complateContent {
        display: flex;
      }
      #complateContent .brief-title {
        margin-right: 10px;
      }
      #complateContent .text-area {
        width: 99%;
        margin-bottom: 10px;
      }
      #selectActions {
        margin-bottom: 10px;
      }
      .action-bar-SA76od.action-bar-lite-window-vuGdUx .box-NzPGeV {
          justify-content: left;
      }
      .action-bar-SA76od .box-NzPGeV {
          min-width: 0;
          display: inline-flex;
          flex-flow: wrap;
          height: 24px;
          gap: 10px;
          align-content: flex-start;
          justify-content: right;
      }
      .action-bar-SA76od .box-NzPGeV {
          min-width: 0;
          display: inline-flex;
          flex-flow: wrap;
          height: 24px;
          gap: 10px;
          align-content: flex-start;
          justify-content: right;
      }
      #floatDialog .tag-button  {
          position: relative;
          cursor: pointer;
          user-select: none;
          -webkit-user-drag: none;
          height: 24px;
          min-height: 24px;
          padding: 0 8px;
          display: flex;
          align-items: center;
          justify-content: center;
          border: 1px solid rgba(56,114,224,.48);
          border-radius: 4px;
          font-size: 13px;
          color: #3872e0;
          gap: 4px;
          max-width: 150px;
          background: #fff;
      }
      #floatDialog .tag-button .text-wrapper-92ojRl {
          flex: 1;
          min-width: 0px;
          display: flex;
          flex-direction: column;
      }
      #floatDialog .tag-button .text-wrapper-92ojRl {
          flex: 1;
          min-width: 0px;
          display: flex;
          flex-direction: column;
      }
      #floatDialog .tag-button:hover {
          background: rgba(235,202,254,.24);
          text-decoration: none!important;
      }
    `);
    // dialogBox.appendChild(complateContent);
    document.body.appendChild(dialogBox);
    // creatChatBtn(complateContent);

    return dialogBox;
  }

  // 主函数
  var main = function () {
    // 创建用户界面
    createUI();
    // 初始化配置
    initConfig();

    // 处理监听ChatGPT-web的消息
    handlePostMessage();
    // 插件管理
    vm = new Vue({
      el: '#customization-chat-room',
      data: {
        activeName: 'chatgpt',
        localCode: [],
        isGetNew: false,
        activeCollapse: [3],
        logs: [
          {
            title: '版本 1.3.1',
            id: 3,
            content: [
              '1. 支持流式输出,响应更快啦!',
              '2. 支持选中文本快捷操作',
              '3. 增加请求频率控制',
              '4. 优化兼容',
            ]
          },
          {
            title: '版本 1.1.7',
            id: 2,
            content: [
              '1. 支持 Midjourney 出图,聊天框输入/mj即可体验,(如/mj iron man)'
            ]
          },
          {
            title: '版本 1.1.6',
            id: 1,
            content: [
              '1. 支持百度、谷歌搜索增强',
              '2. 支持选中文本总结、翻译、扩写、代码解释、解释、语法、问答',
              '3. 支持ChatGPT聊天的完整交互,可自定义prompt、定义角色等等',
              '4. 支持一句话生成油猴脚本,插件管理',
            ]
          }
        ]
      },
      mounted() {
        this.setGMAPI();
        window.addEventListener('storage', this.handleStorageChange);
        this.handleStorageChange();
        $('.el-tabs__content').style.height = window.innerHeight - 50 + 'px';
        $('.el-tabs__content').style.overflowY = 'hidden';
      },
      beforeUnmount() {
        window.removeEventListener('storage', this.handleStorageChange);
      },
      methods: {
        handleClick(tab, event) {
          if (tab.name === 'web') {
            window.open(domain, '_blank');
          } else if (tab.name === 'plugins') {
            this.isGetNew = false;
            $('.plugins-box') &&
              ($('.plugins-box').style.height = window.innerHeight - 80 + 'px');
            $('.el-tabs__content').style.overflowY = 'auto';
          } else {
            $('.el-tabs__content').style.overflowY = 'hidden';
          }
        },
        setGMAPI() {
          try {
            unsafeWindow.GM_log = GM_log;
            unsafeWindow.GM_notification = GM_notification;
            unsafeWindow.unsafeWindow = unsafeWindow;
            unsafeWindow.GM_addStyle = GM_addStyle;
            unsafeWindow.GM_setValue = GM_setValue;
            unsafeWindow.GM_getValue = GM_getValue;
            unsafeWindow.GM_setClipboard = GM_setClipboard;
            unsafeWindow.GM_xmlhttpRequest = GM_xmlhttpRequest;
            // unsafeWindow.GM_addElement = GM_addElement;
            unsafeWindow.GM_openInTab = GM_openInTab;
            unsafeWindow.GM_getResourceText = GM_getResourceText;
            unsafeWindow.GM_registerMenuCommand = GM_registerMenuCommand;
            unsafeWindow.GM_info = GM_info;
          } catch (error) {
            console.error('setGMAPI', error);            
          }
        },
        evalCode(content) {
          const myFunc = () => content;
          myFunc();
        },
        handleCheckboxChange(isChecked, item) {
          item.isChecked = isChecked;
          this.updateStorageItem(item);
        },
        deleteStorageItem(key) {
          this.localCode = this.localCode.filter((item) => item.key !== key);
          localStorage.removeItem(key);
        },
        updateStorageItem(item) {
          localStorage.setItem(
            item.key,
            JSON.stringify({
              ...item,
              ...getUrls(item.content),
              ...getCodeInfo(item.content),
            })
          );
          this.handleStorageChange();
        },
        isScriptMatched(matches = []) {
          const currentURL = window.location.href;
          // 将通配符转换成正则表达式语法
          const patterns = matches.map((pattern) => {
            if (pattern.includes("*")) {
              // 将 * 转换成 .*
              pattern = pattern.replace(/\*/g, ".*");
            }
            if (pattern.includes("?")) {
              // 将 ? 转换成 .
              pattern = pattern.replace(/\?/g, ".");
            }
            return `^${pattern}$`;
          });
          // 检查油猴脚本匹配模式是否包含当前网页的网址
          return patterns.some((pattern) => {
            const regex = new RegExp(pattern);
            return regex.test(currentURL);
          });
        },
        handleStorageChange() {
          this.localCode = [];
          for (let key in localStorage) {
            if (key.split('-')[0] === 'chatgpt') {
              const storageItem = JSON.parse(localStorage.getItem(key));
              this.localCode.push(storageItem);
              if (
                storageItem.isChecked &&
                this.isScriptMatched(storageItem.matches) &&
                !this.isScriptMatched(storageItem.blocks)
              ) {
                evalCode(storageItem.content);
              }
            }
          }
        },
      },
    });

    // 选中文本的弹出框
    createDialogBox();
    // handleTextAutoHeight();
    dialogVm = new Vue({
      el: '#floatDialog',
      data: {
        dialogBox: $('#floatDialog'),
        isShowDialog: false,
        isShowBrief: true,
        isShowComplete: false,
        result: '',
        mapDict: {
          summary: '给一个二年级的学生总结一下',
          explain: '解释{your content here},并说明其中使用的任何技术术语',
          grammar: '校对并纠正这段文字',
          explainCode: '解释以下代码',
          rewrite: '重新表述这段文字',
          translate: '给出以下文字的中文以及英文翻译',
          qa: '回答这个问题',
          expansion: '详细说明这段文字'
        },
        selectVal: ''
      },
      mounted() {
        // 处理选中文本
        this.handleSelect();
        $('#briefContent').addEventListener('click', () => {
          // $('#briefContent').style.display = 'none';
          // complateContent.style.display = 'block';
          this.isShowBrief = false;
          this.isShowComplete = true;
        });
        this.registerClick();
      },
      beforeUnmount() {},
      methods: {
        handleClickDropDown(type) {
          this.handleClick(type)
          this.isShowComplete = true;
          this.isShowBrief = false;
        },
        handleClick(type) {
          this.result = '加载中...';
          postMsg({
            type: 'selectText',
            content: `${this.mapDict[type]}: ${this.selectVal}`,
          });
        },
        addClickAndPostMsg({ targetId, type, content, resultId, clickFn }) {
          $(`#${targetId}`).addEventListener('click', () => {
            postMsg({ type, content });
            $(`#${resultId}`).value = '加载中...';
            clickFn && clickFn();
          });
        },
        registerClick() {
          $('#chatBtn').addEventListener('click', () => {
            $('#customization-drawer').style.right = 0;
          });
        },
        isInDialogBox() {
          const element = $('#floatDialog'); // 获取需要检查的元素
          const rect = element?.getBoundingClientRect(); // 获取元素位置和尺寸信息
          if(!rect) return false;
          const x = event.clientX; // 获取鼠标点击的X坐标
          const y = event.clientY; // 获取鼠标点击的Y坐标
          // let selectedElement = document.getSelection()?.focusNode?.parentNode;
          // console.log('选中的元素是:', document.getSelection(), selectedElement, {
          //   rect,
          //   x,
          //   y,
          // });

          // 判断鼠标点击的坐标是否在元素位置之内
          // if (
          //   rect &&
          //   x >= rect.left &&
          //   x <= rect.right &&
          //   y >= rect.top &&
          //   y <= rect.bottom
          // ) {
          //   // console.log('点击的位置处于该元素之中!');
          // } else {
          //   // console.log('点击的位置不在该元素之中。');
          // }
          return (
            x >= rect.left && x <= rect.right && y >= rect.top && y <= rect.bottom
          );
        },
        setDialogBoxPos() {
          let sel = window.getSelection();
          let range = sel.getRangeAt(0);
          let rect = range?.getBoundingClientRect();

          $('#floatDialog').style.top = rect.bottom + window.pageYOffset + 'px';
          $('#floatDialog').style.left =
            (rect.left + rect.right) / 2 + window.pageXOffset + 'px';
        },
        // 选中文本逻辑处理
        handleSelect() {
          const that = this;
          let selectionText = document.addEventListener('mouseup', (e) => {
            selectionText = window.getSelection().toString();
            if (selectionText && !that.isInDialogBox()) {
              // 如果选中的是文本,而不是DOM元素
              that.selectVal = selectionText;
              that.isShowDialog = true;
              that.isShowBrief = true;
              that.isShowComplete = false;
              that.setDialogBoxPos();
            } else {
              let selectedElement = document.getSelection()?.focusNode?.parentNode;
              const classList = selectedElement?.classList?.value
              if(classList && !classList.includes('chat') && !classList.includes('tab-item-badge')) {
                // postMsg({type: 'clickElement', content: classList})
              }
              if(!that.isInDialogBox() && classList !== 'el-dropdown-menu__item') {
                that.isShowDialog = false;
              } 
            }
          });
        },
      }
    });

    const iframe = $('#chatgpt-iframe');

    iframe.onerror = function () {
      // iframe 加载失败
      const chatRoom = $('#customization-chat-room');
      chatRoom.innerHTML = `
            <p>抱歉,由于某些原因,无法加载此内容。</p>
            <p>请跳转网页版体验:${domain}。</p>
      `;
    };

    // 网页内容读取
    if (isDomain('google') || isDomain('baidu')) {
      handleGoogleSearch();
    }
  };

  // 执行主函数
  const isQqMails = () => isDomain('mail.qq.com');
  try {
    if (window.self !== window.top && isQqMails()) {
      main();
    } else if(window.self === window.top && !isQqMails()) {
      main();
    }
  } catch (error) {
    console.error(error);
  }
})();