chrome.storage.local.get(["showChatBot"], function (result) { if (result.showChatBot === undefined || result.showChatBot) { const chatBtnId = 'fastgpt-chatbot-button'; const chatWindowId = 'fastgpt-chatbot-window'; const chatWindowWrapperId = 'fastgpt-chatbot-wrapper'; const defaultOpen = false; const canDrag = true; const MessageIcon = ``; const CloseIcon = ''; const FullscreenIcon = ''; const SwitchIcon = ''; const ChatBtn = document.createElement('div'); ChatBtn.id = chatBtnId; ChatBtn.style.cssText = 'position: fixed; bottom: 30px; right: 60px; width: 40px; height: 40px; cursor: pointer; z-index: 2147483647; transition: 0;'; const ChatBtnDiv = document.createElement('img'); ChatBtnDiv.src = defaultOpen ? CloseIcon : MessageIcon; ChatBtnDiv.setAttribute('width', '100%'); ChatBtnDiv.setAttribute('height', '100%'); ChatBtnDiv.draggable = false; const iframeWrapper = document.createElement('div'); iframeWrapper.id = chatWindowWrapperId; iframeWrapper.style.cssText = 'border: none; position: fixed; flex-direction: column; justify-content: space-between; box-shadow: rgba(150, 150, 150, 0.2) 0px 10px 30px 0px, rgba(150, 150, 150, 0.2) 0px 0px 0px 1px; bottom: 80px; right: 60px; max-width: 90vw; min-width: 10vw; max-height: 85vh; min-height: 15vh; border-radius: 0.75rem; display: flex; z-index: 2147483647; overflow: hidden; left: unset; background-color: #F3F4F6;'; iframeWrapper.style.visibility = defaultOpen ? 'unset' : 'hidden'; const iframe = document.createElement('iframe'); iframe.referrerPolicy = 'no-referrer'; iframe.allow = 'microphone'; iframe.title = 'FastGPT Chat Window'; iframe.id = chatWindowId; iframe.style.cssText = 'border: none; width: 100%; height: 100%;'; iframeWrapper.appendChild(iframe); const fullscreenBtn = document.createElement('img'); fullscreenBtn.src = FullscreenIcon; fullscreenBtn.style.position = 'absolute'; fullscreenBtn.style.background = 'none'; fullscreenBtn.style.border = 'none'; fullscreenBtn.style.cursor = 'pointer'; fullscreenBtn.id = 'fullscreenBtn'; fullscreenBtn.style.width = '35px'; fullscreenBtn.addEventListener('click', function () { const botSrc = iframe.src; if (botSrc) { window.open(botSrc, '_blank'); } }); const switchBtn = document.createElement('img'); switchBtn.src = SwitchIcon; switchBtn.style.position = 'absolute'; switchBtn.style.background = 'none'; switchBtn.style.border = 'none'; switchBtn.style.cursor = 'pointer'; switchBtn.id = 'switchBtn'; switchBtn.style.width = '35px'; switchBtn.addEventListener('click', function () { chrome.storage.local.get(["configs", "chatbotSrc",], function (result) { const configs = result.configs || []; // 创建或更新列表容器 let listWrapper = document.getElementById('configList'); if (listWrapper) { iframeWrapper.removeChild(listWrapper); return; } listWrapper = document.createElement('div'); listWrapper.id = 'configList'; listWrapper.className = 'ant-dropdown-menu'; listWrapper.style.position = 'absolute'; listWrapper.style.zIndex = '2147483647'; listWrapper.style.backgroundColor = '#fff'; listWrapper.style.border = '1px solid #ccc'; listWrapper.style.borderRadius = '4px'; listWrapper.classList.add('ant-dropdown', 'ant-dropdown-placement-bottomRight'); const switchBtnRect = switchBtn.getBoundingClientRect(); const iframeWrapperRect = iframeWrapper.getBoundingClientRect(); const switchBtnOffsetRight = iframeWrapperRect.right - switchBtnRect.right; const switchBtnOffsetTop = switchBtnRect.top - iframeWrapperRect.top; // 确保listWrapper存在并调整其位置 listWrapper.style.right = switchBtnOffsetRight + 'px'; listWrapper.style.top = (switchBtnOffsetTop + switchBtn.offsetHeight) + 'px'; listWrapper.style.padding = '5px'; // 显示所有chatbot名称 configs.forEach((config) => { const item = document.createElement('div'); item.textContent = config.name; item.className = 'ant-dropdown-menu-item'; // 使用 Ant Design 的类名 item.style.cursor = 'pointer'; item.style.padding = '5px 16px'; item.style.borderRadius = '4px'; // 设置默认样式 item.style.position = 'relative'; item.style.lineHeight = '22px'; item.style.color = '#606266'; item.style.fontSize = '14px'; item.style.whiteSpace = 'nowrap'; item.style.textAlign = 'left'; item.style.boxSizing = 'border-box'; item.style.background = '#fff'; item.style.borderBottomColor = '#e8eaec'; item.style.borderBottomStyle = 'solid'; item.style.borderBottomWidth = '1px'; // 设置选中样式 if (config.url === result.chatbotSrc) { item.style.color = '#1890ff'; item.style.fontWeight = 'bold'; item.style.background = '#e6f7ff'; } // 为每个列表项添加点击事件监听器 item.addEventListener('click', function () { // 更新样式,移除其他项的蓝色 const items = listWrapper.querySelectorAll('.ant-dropdown-menu-item'); items.forEach((i) => { i.style.color = '#606266'; i.style.fontWeight = 'normal'; i.style.background = '#fff'; }); // 设置当前项为蓝色 item.style.color = '#1890ff'; item.style.fontWeight = 'bold'; item.style.background = '#e6f7ff'; // 更新chatbotSrc的值 chrome.storage.local.set({chatbotSrc: config.url}, function () { console.log('Updated chatbotSrc:', config.url); }); // 更新iframe的src loadChatBotIframe(document.getElementById(chatWindowWrapperId)); }); listWrapper.appendChild(item); }); // 将列表容器添加到body中或确保它已经存在 if (!iframeWrapper.contains(listWrapper)) { iframeWrapper.appendChild(listWrapper); } }); }); document.body.appendChild(iframeWrapper); let chatBtnDragged = false; let chatBtnDown = false; let chatBtnMouseX; let chatBtnMouseY; ChatBtn.addEventListener('click', function () { if (chatBtnDragged) { chatBtnDragged = false; return; } const chatWindow = document.getElementById(chatWindowWrapperId); if (!chatWindow) return; const visibilityVal = chatWindow.style.visibility; if (visibilityVal === 'hidden') { ChatBtnDiv.src = CloseIcon; loadChatBotIframe(chatWindow); } else { chatWindow.style.visibility = 'hidden'; const tmpBtn = document.getElementById('fullscreenBtn'); const tmpBtn1 = document.getElementById('switchBtn'); const tmpEl = document.getElementById('configList'); if (tmpBtn) { chatWindow.removeChild(tmpBtn); } if (tmpBtn1) { chatWindow.removeChild(tmpBtn1); } if (tmpEl) { chatWindow.removeChild(tmpEl); } ChatBtnDiv.src = MessageIcon; } }); function loadChatBotIframe(chatWindow) { chrome.storage.local.get(["chatbotSrc", "shareId", "chatId", "fastUID", "chatBotWidth", "chatBotHeight"], function (result) { let botSrc = result.chatbotSrc; if (!botSrc || botSrc === 'about:blank' || botSrc === '') { console.log("Can't find botSrc"); iframe.src = 'data:text/html;charset=utf-8,没有配置机器人地址'; chatWindow.style.visibility = 'unset'; return; } let fastUID = result.fastUID; if (!fastUID || fastUID === '') { fastUID = generateUUID(); chrome.storage.local.set({ fastUID: fastUID }); } chrome.runtime.sendMessage({ action: "startRequestInterception", chatbotSrc: botSrc }); console.log('fastUID:', fastUID); botSrc = botSrc + "&authToken=" + fastUID; if (botSrc.includes(result.shareId)) { botSrc = botSrc + "&chatId=" + result.chatId; } if (result.chatBotWidth && result.chatBotHeight) { chatWindow.style.width = result.chatBotWidth + 'px'; chatWindow.style.height = result.chatBotHeight + 'px'; } else { chatWindow.style.width = '400px'; chatWindow.style.height = '700px'; } iframe.src = 'about:blank'; iframe.onload = function () { chatWindow.style.visibility = 'unset'; } adjustIframePosition(chatWindow); enableResize(iframeWrapper); setTimeout(() => { iframe.src = botSrc; iframe.onload = function () { if (parseInt(chatWindow.style.width, 10) >= 900) { fullscreenBtn.style.top = '13px'; fullscreenBtn.style.right = '60px'; switchBtn.style.top = '13px'; switchBtn.style.right = '90px'; } else { fullscreenBtn.style.top = '6px'; fullscreenBtn.style.right = '50px'; switchBtn.style.top = '6px'; switchBtn.style.right = '80px'; } chatWindow.appendChild(fullscreenBtn); chatWindow.appendChild(switchBtn); const tmpEl = document.getElementById('configList'); if (tmpEl) { chatWindow.removeChild(tmpEl); } }; }, 100); }); } ChatBtn.addEventListener('mousedown', (e) => { e.stopPropagation(); if (!chatBtnMouseX && !chatBtnMouseY) { chatBtnMouseX = e.clientX; chatBtnMouseY = e.clientY; } chatBtnDown = true; }); window.addEventListener('mousemove', throttle(handleMouseMove, 16)); // 60fps window.addEventListener('mouseup', handleMouseUp); function handleMouseMove(e) { e.stopPropagation(); if (!canDrag || !chatBtnDown) return; chatBtnDragged = true; const transformX = e.clientX - chatBtnMouseX; const transformY = e.clientY - chatBtnMouseY; ChatBtn.style.transform = `translate3d(${transformX}px, ${transformY}px, 0)`; adjustIframePosition(document.getElementById(chatWindowWrapperId)); } function handleMouseUp(e) { chatBtnDown = false; adjustIframePosition(document.getElementById(chatWindowWrapperId)); window.removeEventListener('mousemove', handleMouseMove); } ChatBtn.appendChild(ChatBtnDiv); document.body.appendChild(ChatBtn); function generateUUID() { const randomString = 'xxxxxxxxxxxxxxxx'.replace(/[x]/g, function () { const randomHex = (Math.random() * 16) | 0; return randomHex.toString(16); }); const timestamp = Date.now().toString(16); const extraRandom = (Math.random() * 1e16).toString(16); return `${randomString}-${timestamp}-${extraRandom}`; } function adjustIframePosition(chatWindow) { const chatBtnRect = ChatBtn.getBoundingClientRect(); const chatBtnWidth = chatBtnRect.width; const chatBtnHeight = chatBtnRect.height; const chatBtnLeft = chatBtnRect.left; const chatBtnTop = chatBtnRect.top; const screenWidth = window.innerWidth; const screenHeight = window.innerHeight; const iframeWidth = parseInt(chatWindow.style.width, 10); const iframeHeight = parseInt(chatWindow.style.height, 10); const iframeTopLeft = {x: chatBtnLeft - iframeWidth, y: chatBtnTop - iframeHeight}; const iframeTopRight = {x: chatBtnLeft + chatBtnWidth, y: chatBtnTop - iframeHeight}; const iframeBottomLeft = {x: chatBtnLeft - iframeWidth, y: chatBtnTop + chatBtnHeight}; const iframeBottomRight = {x: chatBtnLeft + chatBtnWidth, y: chatBtnTop + chatBtnHeight}; let bestPosition = iframeTopLeft; let bestDistance = Infinity; const positions = [iframeTopLeft, iframeTopRight, iframeBottomLeft, iframeBottomRight]; positions.forEach(position => { const distance = Math.sqrt(Math.pow(position.x, 2) + Math.pow(position.y, 2)); if (position.x + iframeWidth > screenWidth) { position.x = screenWidth - iframeWidth; } if (position.x < 0) { position.x = 0; } if (position.y + iframeHeight > screenHeight) { position.y = screenHeight - iframeHeight; } if (position.y < 0) { position.y = 0; } if (distance < bestDistance) { bestPosition = position; bestDistance = distance; } }); chatWindow.style.left = `${bestPosition.x}px`; chatWindow.style.top = `${bestPosition.y}px`; } function throttle(func, limit) { let lastFunc; let lastRan; return function () { const context = this; const args = arguments; if (!lastRan) { func.apply(context, args); lastRan = Date.now(); } else { clearTimeout(lastFunc); lastFunc = setTimeout(function () { if ((Date.now() - lastRan) >= limit) { func.apply(context, args); lastRan = Date.now(); } }, limit - (Date.now() - lastRan)); } } } function enableResize(iframeWrapper) { let isResizing = false; let lastDownX = 0; let lastDownY = 0; let resizeDirection = ''; // 创建八个调整大小的句柄 const handles = ['nw-resize', 'ne-resize', 'sw-resize', 'se-resize', 'n-resize', 's-resize', 'w-resize', 'e-resize']; const directions = ['tl', 'tr', 'bl', 'br', 't', 'b', 'l', 'r']; handles.forEach((cursorType, index) => { const handle = createResizeHandle(cursorType); iframeWrapper.appendChild(handle); positionHandle(handle, directions[index]); handle.addEventListener('mousedown', (e) => startResizing(e, directions[index])); }); function createResizeHandle(cursorType) { const handle = document.createElement('div'); handle.style.width = '15px'; handle.style.height = '15px'; handle.style.background = 'transparent'; handle.style.position = 'absolute'; handle.style.cursor = cursorType; return handle; } function positionHandle(handle, direction) { switch (direction) { case 'tl': handle.style.top = '0'; handle.style.left = '0'; break; case 'tr': handle.style.top = '0'; handle.style.right = '0'; break; case 'bl': handle.style.bottom = '0'; handle.style.left = '0'; break; case 'br': handle.style.bottom = '0'; handle.style.right = '0'; break; case 't': handle.style.top = '0'; handle.style.left = '50%'; handle.style.transform = 'translateX(-50%)'; handle.style.width = `${iframeWrapper.offsetWidth - 30}px`; handle.style.height = '3px'; break; case 'b': handle.style.bottom = '0'; handle.style.left = '50%'; handle.style.transform = 'translateX(-50%)'; handle.style.width = `${iframeWrapper.offsetWidth - 30}px`; handle.style.height = '3px'; break; case 'l': handle.style.top = '50%'; handle.style.left = '0'; handle.style.transform = 'translateY(-50%)'; handle.style.width = '3px'; handle.style.height = `${iframeWrapper.offsetHeight - 30}px`; break; case 'r': handle.style.top = '50%'; handle.style.right = '0'; handle.style.transform = 'translateY(-50%)'; handle.style.width = '3px'; handle.style.height = `${iframeWrapper.offsetHeight - 30}px`; break; } } function startResizing(e, direction) { isResizing = true; lastDownX = e.clientX; lastDownY = e.clientY; resizeDirection = direction; iframeWrapper.style.pointerEvents = 'none'; // 禁用 iframe 的鼠标事件 e.preventDefault(); } document.addEventListener('mousemove', throttle(handleResizeMouseMove, 50)); document.addEventListener('mouseup', handleResizeMouseUp); function handleResizeMouseMove(e) { if (!isResizing) return; const offsetX = e.clientX - lastDownX; const offsetY = e.clientY - lastDownY; requestAnimationFrame(() => { switch (resizeDirection) { case 'tl': iframeWrapper.style.width = `${iframeWrapper.offsetWidth - offsetX}px`; iframeWrapper.style.height = `${iframeWrapper.offsetHeight - offsetY}px`; iframeWrapper.style.left = `${iframeWrapper.offsetLeft + offsetX}px`; iframeWrapper.style.top = `${iframeWrapper.offsetTop + offsetY}px`; break; case 'tr': iframeWrapper.style.width = `${iframeWrapper.offsetWidth + offsetX}px`; iframeWrapper.style.height = `${iframeWrapper.offsetHeight - offsetY}px`; iframeWrapper.style.top = `${iframeWrapper.offsetTop + offsetY}px`; break; case 'bl': iframeWrapper.style.width = `${iframeWrapper.offsetWidth - offsetX}px`; iframeWrapper.style.height = `${iframeWrapper.offsetHeight + offsetY}px`; iframeWrapper.style.left = `${iframeWrapper.offsetLeft + offsetX}px`; break; case 'br': iframeWrapper.style.width = `${iframeWrapper.offsetWidth + offsetX}px`; iframeWrapper.style.height = `${iframeWrapper.offsetHeight + offsetY}px`; break; case 't': iframeWrapper.style.height = `${iframeWrapper.offsetHeight - offsetY}px`; iframeWrapper.style.top = `${iframeWrapper.offsetTop + offsetY}px`; break; case 'b': iframeWrapper.style.height = `${iframeWrapper.offsetHeight + offsetY}px`; break; case 'l': iframeWrapper.style.width = `${iframeWrapper.offsetWidth - offsetX}px`; iframeWrapper.style.left = `${iframeWrapper.offsetLeft + offsetX}px`; break; case 'r': iframeWrapper.style.width = `${iframeWrapper.offsetWidth + offsetX}px`; break; } lastDownX = e.clientX; lastDownY = e.clientY; if (parseInt(iframeWrapper.style.width, 10) >= 900) { fullscreenBtn.style.top = '13px'; fullscreenBtn.style.right = '60px'; switchBtn.style.top = '13px'; switchBtn.style.right = '90px'; } else { fullscreenBtn.style.top = '6px'; fullscreenBtn.style.right = '50px'; switchBtn.style.top = '6px'; switchBtn.style.right = '80px'; } }); } function handleResizeMouseUp() { console.log('handleResizeMouseUp'); if (isResizing) { isResizing = false; // 将 isResizing 重置为 false enableResize(iframeWrapper); iframeWrapper.style.pointerEvents = 'auto'; // 恢复 iframe 的鼠标事件 chrome.storage.local.set({ chatBotWidth: iframeWrapper.offsetWidth, chatBotHeight: iframeWrapper.offsetHeight }); } } } } });