mirror of
https://github.com/labring/FastGPT.git
synced 2025-07-23 21:13:50 +00:00
feat(chatbot-extension): a Chrome extension that can be using for chat with AI on any website (#2235)
* feat(chatbot-extension): a Chrome extension that can be using for chat with AI on any website * fix: 插件支持语音输入 feat:chatbot支持切换 * fix: 切换chatbot后,自动隐藏bot列表
This commit is contained in:
Binary file not shown.
After Width: | Height: | Size: 1.2 KiB |
Binary file not shown.
After Width: | Height: | Size: 1012 B |
Binary file not shown.
After Width: | Height: | Size: 1.9 KiB |
Binary file not shown.
After Width: | Height: | Size: 4.0 KiB |
@@ -0,0 +1,42 @@
|
||||
{
|
||||
"manifest_version": 3,
|
||||
"name": "ChatBot Extension",
|
||||
"version": "1.1",
|
||||
"description": "A ChatBot",
|
||||
"permissions": [
|
||||
"storage",
|
||||
"notifications",
|
||||
"tabs",
|
||||
"activeTab",
|
||||
"scripting",
|
||||
"webRequest"
|
||||
],
|
||||
"host_permissions": [
|
||||
"<all_urls>"
|
||||
],
|
||||
"background": {
|
||||
"service_worker": "src/background.js"
|
||||
},
|
||||
"action": {
|
||||
"default_popup": "src/popup.html",
|
||||
"default_icon": {
|
||||
"16": "img/favicon32.png",
|
||||
"48": "img/favicon32.png",
|
||||
"128": "img/favicon32.png"
|
||||
}
|
||||
},
|
||||
"icons": {
|
||||
"16": "img/favicon32.png",
|
||||
"32": "img/favicon32.png",
|
||||
"48": "img/favicon32.png",
|
||||
"128": "img/favicon32.png"
|
||||
},
|
||||
"content_scripts": [
|
||||
{
|
||||
"js": [
|
||||
"src/content.js"
|
||||
],
|
||||
"matches": ["<all_urls>"]
|
||||
}
|
||||
]
|
||||
}
|
@@ -0,0 +1,51 @@
|
||||
let requestInterceptor = null;
|
||||
|
||||
chrome.runtime.onMessage.addListener(function (message, sender, sendResponse) {
|
||||
if (message.action === "startRequestInterception") {
|
||||
const botSrc = message.chatbotSrc;
|
||||
console.log("src", botSrc);
|
||||
const urlObj = new URL(botSrc);
|
||||
const domain = urlObj.host;
|
||||
const searchParams = urlObj.searchParams;
|
||||
const frameShareId = searchParams.get('shareId') || '';
|
||||
let frameChatId = searchParams.get('chatId') || '';
|
||||
|
||||
// 移除已有的拦截器(如果存在)
|
||||
if (requestInterceptor) {
|
||||
chrome.webRequest.onBeforeRequest.removeListener(requestInterceptor);
|
||||
}
|
||||
|
||||
requestInterceptor = function (details) {
|
||||
if (details.frameId !== -1
|
||||
&& details.url.includes(domain)
|
||||
&& details.url.includes("chat/completions")) {
|
||||
console.log("Intercepted request from chatbot-iframe:", details);
|
||||
if (details.requestBody.raw) {
|
||||
let decoder = new TextDecoder("utf-8");
|
||||
let postData = decoder.decode(new Uint8Array(details.requestBody.raw[0].bytes));
|
||||
try {
|
||||
let postDataObj = JSON.parse(postData);
|
||||
let shareId = postDataObj.shareId;
|
||||
let chatId = postDataObj.chatId;
|
||||
|
||||
if (frameChatId !== chatId && frameShareId === shareId) {
|
||||
chrome.storage.local.set({["shareId"]: shareId});
|
||||
chrome.storage.local.set({["chatId"]: chatId});
|
||||
frameChatId = chatId;
|
||||
console.log(`Stored shareId: ${shareId} and chatId: ${chatId} to localStorage.`);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error("Error parsing postData:", error);
|
||||
}
|
||||
}
|
||||
}
|
||||
return {};
|
||||
};
|
||||
|
||||
chrome.webRequest.onBeforeRequest.addListener(
|
||||
requestInterceptor,
|
||||
{ urls: ["<all_urls>"] },
|
||||
["requestBody"]
|
||||
);
|
||||
}
|
||||
});
|
@@ -0,0 +1,533 @@
|
||||
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,<html><head><style>body { margin: 0; padding: 0; overflow: hidden; display: flex; justify-content: center; align-items: center; height: 100%; }</style></head><body>没有配置机器人地址</body></html>';
|
||||
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
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
@@ -0,0 +1,108 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
|
||||
<head>
|
||||
<title>Chatbot Extension</title>
|
||||
<meta charset="UTF-8">
|
||||
<style>
|
||||
body,
|
||||
html {
|
||||
height: 600px;
|
||||
width: 800px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
display: flex;
|
||||
}
|
||||
::-webkit-scrollbar {
|
||||
width: 0;
|
||||
}
|
||||
h1 {
|
||||
text-align: center;
|
||||
cursor: pointer;
|
||||
color: #007BFF;
|
||||
}
|
||||
#chatbot-container {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
margin: 0;
|
||||
flex-grow: 1;
|
||||
}
|
||||
|
||||
#chatbot-iframe {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
flex-grow: 1;
|
||||
border: none;
|
||||
box-shadow: 0px 0px 20px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.config-btn-icon {
|
||||
position: absolute;
|
||||
display: none;
|
||||
top: 8px;
|
||||
right: 50px;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
cursor: pointer;
|
||||
}
|
||||
.fullScreen-icon {
|
||||
position: absolute;
|
||||
display: none;
|
||||
top: 8px;
|
||||
right: 80px; /* Adjusted to position left of the config button */
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
cursor: pointer;
|
||||
}
|
||||
.overlay {
|
||||
position: absolute;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-color: rgba(255, 255, 255, 0.8);
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
transition: opacity 0.3s ease;
|
||||
border-radius: 10px;
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.overlay-content {
|
||||
background-color: #fff;
|
||||
padding: 20px;
|
||||
border-radius: 8px;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||
font-size: 18px;
|
||||
color: #333;
|
||||
font-weight: bold;
|
||||
transition: transform 0.3s ease;
|
||||
transform: scale(0.9);
|
||||
}
|
||||
|
||||
.overlay-content.show {
|
||||
transform: scale(1);
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div id="chatbot-container">
|
||||
<div id="configOverlay" class="overlay" style="display: none;">
|
||||
<div class="overlay-content">
|
||||
未配置机器人,请点击右上角【设置】
|
||||
</div>
|
||||
</div>
|
||||
<iframe id="chatbot-iframe" allowfullscreen allow="microphone"></iframe>
|
||||
<!-- <webview id="chatbot-iframe" allowfullscreen></webview>-->
|
||||
<img src="../img/fullScreen.png" class="fullScreen-icon" id="fullScreen">
|
||||
<img src="../img/setting.png" class="config-btn-icon" id="config-btn">
|
||||
</div>
|
||||
<script src="popup.js"></script>
|
||||
|
||||
</body>
|
||||
|
||||
</html>
|
@@ -0,0 +1,66 @@
|
||||
document.addEventListener('DOMContentLoaded', function () {
|
||||
const chatbotIframe = document.getElementById('chatbot-iframe');
|
||||
const fullScreenBtn = document.getElementById('fullScreen');
|
||||
const configBtn = document.getElementById('config-btn');
|
||||
const overlay = document.getElementById('configOverlay');
|
||||
configBtn.addEventListener('click', function () {
|
||||
window.location.href = 'setting.html'
|
||||
});
|
||||
|
||||
// 监听 chatbotIframe 加载完成事件
|
||||
chatbotIframe.addEventListener('load', function() {
|
||||
// 当 iframe 加载完成后显示按钮
|
||||
fullScreenBtn.style.display = 'inline-block';
|
||||
configBtn.style.display = 'inline-block';
|
||||
});
|
||||
chrome.storage.local.get(["chatbotSrc", "shareId", "chatId", "fastUID"]).then((result) => {
|
||||
const botSrc = result.chatbotSrc;
|
||||
let fastUID = result.fastUID;
|
||||
if (!fastUID || fastUID === '') {
|
||||
fastUID = generateUUID();
|
||||
chrome.storage.local.set({
|
||||
fastUID: fastUID
|
||||
});
|
||||
}
|
||||
console.log('fastUID is', fastUID);
|
||||
console.log("chatbotSrc is " + botSrc);
|
||||
console.log("shareId is " + result.shareId);
|
||||
console.log("chatId is " + result.chatId);
|
||||
chatbotIframe.src = result.chatbotSrc + "&authToken=" + fastUID;
|
||||
if (!botSrc || botSrc === 'about:blank' || botSrc === '') {
|
||||
overlay.style.display = 'flex';
|
||||
} else {
|
||||
if (botSrc.includes(result.shareId)) {
|
||||
chatbotIframe.src = chatbotIframe.src + "&chatId=" + result.chatId;
|
||||
console.log('chatbotIframe.src', chatbotIframe.src);
|
||||
}
|
||||
overlay.style.display = 'none';
|
||||
chrome.runtime.sendMessage({
|
||||
action: "startRequestInterception",
|
||||
chatbotSrc: botSrc
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
fullScreenBtn.addEventListener('click', function () {
|
||||
const iframe = document.getElementById('chatbot-iframe');
|
||||
if (iframe) {
|
||||
const iframeSrc = iframe.src;
|
||||
console.log('fullScreenSrc', iframeSrc)
|
||||
chrome.tabs.create({url: iframeSrc});
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
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}`;
|
||||
}
|
@@ -0,0 +1,382 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>配置项管理器</title>
|
||||
<style>
|
||||
body, html {
|
||||
height: 585px;
|
||||
width: 800px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
font-family: Arial, sans-serif;
|
||||
background-color: #f0f0f0;
|
||||
}
|
||||
|
||||
#startChatButton {
|
||||
position: absolute;
|
||||
top: 5px;
|
||||
left: 5px;
|
||||
width: 95px;
|
||||
height: 30px;
|
||||
background-color: #1890ff;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
color: white;
|
||||
font-size: 14px;
|
||||
font-weight: bold;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
clip-path: polygon(10% 0%, 100% 0%, 100% 100%, 10% 100%, 0% 50%);
|
||||
border-top-left-radius: 10px;
|
||||
border-bottom-left-radius: 10px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
table, form {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
h1, h2 {
|
||||
text-align: center;
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
#configTable th:nth-child(1),
|
||||
#configTable td:nth-child(1) {
|
||||
width: 150px; /* 设置第1列的宽度 */
|
||||
}
|
||||
|
||||
#configTable th:nth-child(3),
|
||||
#configTable td:nth-child(3) {
|
||||
width: 70px; /* 设置第3列的宽度 */
|
||||
}
|
||||
|
||||
#configTable th:nth-child(4),
|
||||
#configTable td:nth-child(4) {
|
||||
width: 50px; /* 设置第4列的宽度 */
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
table {
|
||||
width: 800px;
|
||||
border-collapse: collapse;
|
||||
background-color: #fff;
|
||||
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
th, td {
|
||||
padding: 10px;
|
||||
text-align: left;
|
||||
border-bottom: 1px solid #ddd;
|
||||
position: relative;
|
||||
word-break: break-word;
|
||||
}
|
||||
|
||||
th {
|
||||
background-color: #f2f2f2;
|
||||
}
|
||||
|
||||
form {
|
||||
max-width: 600px;
|
||||
margin: auto;
|
||||
background-color: #fff;
|
||||
padding: 20px;
|
||||
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
label {
|
||||
display: block;
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
|
||||
input[type="text"], input[type="number"] {
|
||||
width: calc(100% - 10px);
|
||||
padding: 8px;
|
||||
margin-bottom: 10px;
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 4px;
|
||||
box-sizing: border-box;
|
||||
vertical-align: middle;
|
||||
display: inline-block;
|
||||
transform: translateY(15%);
|
||||
}
|
||||
|
||||
button {
|
||||
background-color: #1890ff;
|
||||
color: #fff;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
button:hover {
|
||||
background-color: #0056b3;
|
||||
}
|
||||
|
||||
#addConfigButton {
|
||||
left: 10px;
|
||||
width: 60px;
|
||||
height: 30px;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
color: white;
|
||||
font-size: 15px;
|
||||
font-weight: bold;
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
.editName, .editUrl {
|
||||
width: 100%;
|
||||
padding: 8px;
|
||||
margin-bottom: 10px;
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 4px;
|
||||
box-sizing: border-box;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
.editButton[disabled] {
|
||||
color: #666; /* Gray text color */
|
||||
cursor: not-allowed; /* Change cursor to not-allowed */
|
||||
}
|
||||
|
||||
.editButton, .deleteButton {
|
||||
background-color: transparent; /* Remove background color */
|
||||
border: none;
|
||||
padding: 0;
|
||||
text-align: center;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
color: #1890ff;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.button-group {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.button-group button {
|
||||
margin-right: 5px; /* Add space between buttons */
|
||||
}
|
||||
|
||||
.editButton:hover {
|
||||
background-color: transparent;
|
||||
color: #4CAF50; /* Green color on hover */
|
||||
}
|
||||
|
||||
.deleteButton:hover {
|
||||
background-color: transparent;
|
||||
color: #f44336; /* Red color on hover */
|
||||
}
|
||||
|
||||
.editButton + div {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
position: absolute;
|
||||
left: -20px; /* 调整位置以适应左侧显示 */
|
||||
top: 0; /* 确保与编辑按钮对齐 */
|
||||
}
|
||||
|
||||
.editButton + div span {
|
||||
cursor: pointer;
|
||||
margin: 5px 0; /* 上下间距 */
|
||||
}
|
||||
|
||||
.icon-hover {
|
||||
transition: transform 0.3s ease; /* 添加过渡效果 */
|
||||
}
|
||||
|
||||
.icon-hover:hover {
|
||||
transform: scale(2); /* 鼠标悬停时放大1.2倍 */
|
||||
}
|
||||
|
||||
.custom-radio {
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.custom-radio input[type="radio"] {
|
||||
position: absolute;
|
||||
opacity: 0;
|
||||
cursor: pointer;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
z-index: 2;
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
height: 16px;
|
||||
width: 16px;
|
||||
background-color: #f1f1f1;
|
||||
border-radius: 50%;
|
||||
border: 1px solid #d9d9d9;
|
||||
z-index: 1;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.custom-radio:hover input ~ .radio-mark {
|
||||
background-color: #ccc;
|
||||
}
|
||||
|
||||
.custom-radio input:checked ~ .radio-mark {
|
||||
background-color: #1890ff;
|
||||
border-color: #1890ff;
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark:after {
|
||||
content: "";
|
||||
position: absolute;
|
||||
display: none;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
border-radius: 50%;
|
||||
background: white;
|
||||
transform: translate(-50%, -50%);
|
||||
}
|
||||
|
||||
.custom-radio input:checked ~ .radio-mark:after {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark:active {
|
||||
transform: scale(0.9);
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark:active:after {
|
||||
transform: scale(1.1);
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark:hover {
|
||||
transform: scale(1.1);
|
||||
}
|
||||
|
||||
.custom-radio .radio-mark:hover:after {
|
||||
transform: scale(1.1);
|
||||
}
|
||||
|
||||
#errorMsg {
|
||||
position: fixed;
|
||||
top: 20px;
|
||||
right: 20px;
|
||||
background-color: #f8d7da;
|
||||
color: #721c24;
|
||||
border: 1px solid #f5c6cb;
|
||||
padding: 10px 15px;
|
||||
border-radius: 5px;
|
||||
z-index: 1000;
|
||||
opacity: 0;
|
||||
transition: opacity 0.5s ease;
|
||||
}
|
||||
|
||||
/* 为确认和取消按钮添加伪元素 */
|
||||
.confirmButton::before,
|
||||
.cancelButton::before {
|
||||
content: ""; /* 初始时,伪元素的内容为空 */
|
||||
}
|
||||
|
||||
/* 当鼠标悬停在按钮上时,显示文字 */
|
||||
.confirmButton:hover::before {
|
||||
content: "确认";
|
||||
}
|
||||
|
||||
.cancelButton:hover::before {
|
||||
content: "取消";
|
||||
}
|
||||
|
||||
.confirmButton::before,
|
||||
.cancelButton::before {
|
||||
position: absolute;
|
||||
left: 20px;
|
||||
transform: translate(-50%, -50%);
|
||||
font-size: 8px;
|
||||
color: darkgray;
|
||||
white-space: nowrap;
|
||||
transition: content 0.3s;
|
||||
}
|
||||
|
||||
.confirmButton::before {
|
||||
top: 10%;
|
||||
}
|
||||
|
||||
.cancelButton::before {
|
||||
top: 80%;
|
||||
}
|
||||
|
||||
.config-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 10px;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.config-item label {
|
||||
margin-right: 10px;
|
||||
width: 120px;
|
||||
text-align: right;
|
||||
font-weight: bold;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.config-item input[type="number"] {
|
||||
width: 80px;
|
||||
padding: 6px;
|
||||
border: 1px solid #ccc;
|
||||
border-radius: 4px;
|
||||
box-sizing: border-box;
|
||||
text-align: center;
|
||||
}
|
||||
</style>
|
||||
<script src="setting.js" defer></script>
|
||||
</head>
|
||||
<body>
|
||||
<button id="startChatButton">开始聊天</button>
|
||||
<div id="errorMsg"></div>
|
||||
<h1>ChatBot配置</h1>
|
||||
<h3>页面机器人:</h3>
|
||||
<div style="display: flex">
|
||||
<div class="config-item">
|
||||
<label for="showChatBotSwitch">显示:</label>
|
||||
<input type="checkbox" id="showChatBotSwitch">
|
||||
</div>
|
||||
|
||||
<div class="config-item">
|
||||
<label for="chatBotWidthInput">宽度(px):</label>
|
||||
<input type="number" id="chatBotWidthInput" min="1">
|
||||
</div>
|
||||
|
||||
<div class="config-item">
|
||||
<label for="chatBotHeightInput">高度(px):</label>
|
||||
<input type="number" id="chatBotHeightInput" min="1">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<h3>机器人地址:</h3>
|
||||
<table id="configTable">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>名称</th>
|
||||
<th>地址</th>
|
||||
<th>操作</th>
|
||||
<th>选择BOT</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody id="configList"> <!-- Table rows will be dynamically added using JavaScript --> </tbody>
|
||||
</table>
|
||||
<button id="addConfigButton">添 加</button>
|
||||
</body>
|
||||
</html>
|
@@ -0,0 +1,283 @@
|
||||
let chatbotSrc = '';
|
||||
let configs = [];
|
||||
|
||||
document.addEventListener('DOMContentLoaded', async function () {
|
||||
const storageData = await chrome.storage.local.get(["chatbotSrc", "configs", "showChatBot", "chatBotWidth", "chatBotHeight"]);
|
||||
chatbotSrc = storageData.chatbotSrc || '';
|
||||
configs = storageData.configs || [];
|
||||
const showChatBot = storageData.showChatBot === undefined ? true : storageData.showChatBot;
|
||||
const chatBotWidth = storageData.chatBotWidth || 400;
|
||||
const chatBotHeight = storageData.chatBotHeight || 700;
|
||||
|
||||
await loadConfigs(configs, chatbotSrc);
|
||||
|
||||
document.getElementById('addConfigButton').addEventListener('click', handleAddButtonClick);
|
||||
document.getElementById('startChatButton').addEventListener('click', () => window.location.href = 'popup.html');
|
||||
|
||||
// 监听开关和输入框变化事件
|
||||
const showChatBotSwitch = document.getElementById('showChatBotSwitch');
|
||||
const chatBotWidthInput = document.getElementById('chatBotWidthInput');
|
||||
const chatBotHeightInput = document.getElementById('chatBotHeightInput');
|
||||
|
||||
showChatBotSwitch.addEventListener('change', () => handleShowChatBotChange(showChatBotSwitch.checked));
|
||||
chatBotWidthInput.addEventListener('change', () => handleChatBotWidthChange(chatBotWidthInput.value));
|
||||
chatBotHeightInput.addEventListener('change', () => handleChatBotHeightChange(chatBotHeightInput.value));
|
||||
|
||||
// 初始化开关和输入框的值
|
||||
showChatBotSwitch.checked = showChatBot;
|
||||
chatBotWidthInput.value = chatBotWidth;
|
||||
chatBotHeightInput.value = chatBotHeight;
|
||||
|
||||
});
|
||||
|
||||
async function loadConfigs(configs, chatbotSrc) {
|
||||
const configList = document.getElementById('configList');
|
||||
configList.innerHTML = '';
|
||||
configs.forEach(config => {
|
||||
const row = createConfigRow(config, chatbotSrc);
|
||||
configList.appendChild(row);
|
||||
});
|
||||
}
|
||||
|
||||
function createConfigRow(config, chatbotSrc) {
|
||||
const row = document.createElement('tr');
|
||||
row.innerHTML = `
|
||||
<td>${config.name}</td>
|
||||
<td>${config.url}</td>
|
||||
<td>
|
||||
<button type="button" class="editButton">编辑</button>
|
||||
<span>|</span>
|
||||
<button type="button" class="deleteButton">删除</button>
|
||||
</td>
|
||||
<td>
|
||||
<label class="custom-radio">
|
||||
<input type="radio" name="selectBot" class="selectBot" ${config.url === chatbotSrc ? 'checked' : ''}>
|
||||
<span class="radio-mark"></span>
|
||||
</label>
|
||||
</td>
|
||||
`;
|
||||
row.querySelector('.editButton').addEventListener('click', () => handleEditButtonClick(row, config));
|
||||
row.querySelector('.deleteButton').addEventListener('click', () => handleDeleteButtonClick(row, config));
|
||||
row.querySelector('.selectBot').addEventListener('change', () => handleSelectBotChange(config.url));
|
||||
return row;
|
||||
}
|
||||
|
||||
|
||||
async function handleDeleteButtonClick(row, config) {
|
||||
const configList = document.getElementById('configList');
|
||||
const index = Array.from(configList.children).indexOf(row);
|
||||
const selectedUrl = config.url;
|
||||
if (selectedUrl === chatbotSrc) {
|
||||
chatbotSrc = 'about:blank';
|
||||
await updateStorage('chatbotSrc', chatbotSrc);
|
||||
await updateStorage('chatId', '');
|
||||
await updateStorage('shareId', '');
|
||||
}
|
||||
configs.splice(index, 1);
|
||||
await updateStorage('configs', configs);
|
||||
row.remove();
|
||||
}
|
||||
|
||||
async function handleSelectBotChange(url) {
|
||||
chatbotSrc = url;
|
||||
await updateStorage('chatbotSrc', chatbotSrc);
|
||||
updateSelectedRadioButton();
|
||||
}
|
||||
|
||||
function updateSelectedRadioButton() {
|
||||
document.querySelectorAll('.selectBot').forEach(radio => {
|
||||
if (radio.closest('tr').querySelector('td:nth-child(2)').textContent === chatbotSrc) {
|
||||
radio.checked = true;
|
||||
} else {
|
||||
radio.checked = false;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
function showError(message) {
|
||||
const errorMsg = document.getElementById('errorMsg');
|
||||
errorMsg.textContent = message;
|
||||
errorMsg.style.opacity = 1; // 显示错误消息
|
||||
|
||||
// 设置一个定时器,在5秒后隐藏错误消息
|
||||
setTimeout(() => {
|
||||
errorMsg.style.opacity = 0; // 隐藏错误消息
|
||||
}, 3000);
|
||||
}
|
||||
|
||||
|
||||
async function updateStorage(key, value) {
|
||||
try {
|
||||
await chrome.storage.local.set({[key]: value});
|
||||
console.log(`${key} 已更新: ${value}`);
|
||||
} catch (error) {
|
||||
console.error(`更新 ${key} 出错:`, error);
|
||||
}
|
||||
}
|
||||
|
||||
function showEditControls(row, isNewConfig = false, config = null) {
|
||||
const nameCell = row.querySelector('td:nth-child(1)');
|
||||
const urlCell = row.querySelector('td:nth-child(2)');
|
||||
const name = isNewConfig ? '' : config.name;
|
||||
const url = isNewConfig ? '' : config.url;
|
||||
|
||||
nameCell.innerHTML = `<input type="text" class="editName" value="${name}">`;
|
||||
urlCell.innerHTML = `<input type="text" class="editUrl" value="${url}">`;
|
||||
|
||||
const iconContainer = document.createElement('div');
|
||||
iconContainer.style.display = 'flex';
|
||||
iconContainer.style.flexDirection = 'column';
|
||||
iconContainer.style.position = 'absolute';
|
||||
iconContainer.style.top = '15px';
|
||||
iconContainer.style.left = '-5px';
|
||||
|
||||
const confirmIcon = document.createElement('span');
|
||||
confirmIcon.textContent = '√';
|
||||
confirmIcon.style.color = 'green';
|
||||
confirmIcon.style.cursor = 'pointer';
|
||||
confirmIcon.style.margin = '0 0 10px 0';
|
||||
confirmIcon.classList.add('icon-hover');
|
||||
confirmIcon.classList.add('confirmButton');
|
||||
confirmIcon.addEventListener('click', () => {
|
||||
handleConfirmButtonClick(row, isNewConfig);
|
||||
});
|
||||
|
||||
const cancelIcon = document.createElement('span');
|
||||
cancelIcon.textContent = 'X';
|
||||
cancelIcon.style.color = 'red';
|
||||
cancelIcon.style.cursor = 'pointer';
|
||||
cancelIcon.classList.add('icon-hover');
|
||||
cancelIcon.classList.add('cancelButton')
|
||||
cancelIcon.addEventListener('click', () => {
|
||||
handleCancelButtonClick(row, isNewConfig);
|
||||
});
|
||||
|
||||
iconContainer.appendChild(confirmIcon);
|
||||
iconContainer.appendChild(cancelIcon);
|
||||
|
||||
const cell = row.querySelector('td:nth-child(3)');
|
||||
cell.insertBefore(iconContainer, cell.firstChild);
|
||||
}
|
||||
|
||||
function handleAddButtonClick() {
|
||||
const newRow = document.createElement('tr');
|
||||
newRow.innerHTML = `
|
||||
<td></td>
|
||||
<td></td>
|
||||
<td>
|
||||
</td>
|
||||
<td>
|
||||
</td>
|
||||
`;
|
||||
document.getElementById('configList').appendChild(newRow);
|
||||
showEditControls(newRow, true);
|
||||
}
|
||||
|
||||
function handleEditButtonClick(row, config) {
|
||||
disableAllEditButtons();
|
||||
showEditControls(row, false, config);
|
||||
}
|
||||
|
||||
function isConfigUnique(name, url, index) {
|
||||
return !configs.some((config, i) =>
|
||||
i !== index && (config.name === name || config.url === url)
|
||||
);
|
||||
}
|
||||
|
||||
function checkConfig(name, url, index) {
|
||||
let errorMsg = '';
|
||||
if (!name || !url) {
|
||||
errorMsg = '名称和地址都是必填项。';
|
||||
} else if (!isConfigUnique(name, url, index)) {
|
||||
errorMsg = '名称或地址不能重复。';
|
||||
}
|
||||
return errorMsg;
|
||||
}
|
||||
|
||||
function handleConfirmButtonClick(row, isNewConfig = false) {
|
||||
const name = row.querySelector('.editName').value;
|
||||
const url = row.querySelector('.editUrl').value;
|
||||
const index = isNewConfig ? -1 : Array.from(document.getElementById('configList').children).indexOf(row);
|
||||
const errorMsg = checkConfig(name, url, index);
|
||||
if (errorMsg) {
|
||||
showError(errorMsg);
|
||||
return;
|
||||
}
|
||||
|
||||
if (isNewConfig) {
|
||||
const newConfig = {name, url};
|
||||
configs.push(newConfig);
|
||||
updateStorage('configs', configs);
|
||||
|
||||
} else {
|
||||
const config = configs[index];
|
||||
config.name = name;
|
||||
if (config.url === chatbotSrc) {
|
||||
chatbotSrc = url;
|
||||
updateStorage('chatId', '');
|
||||
updateStorage('shareId', '');
|
||||
chrome.runtime.sendMessage({
|
||||
action: "startRequestInterception",
|
||||
chatbotSrc: chatbotSrc
|
||||
});
|
||||
}
|
||||
config.url = url;
|
||||
updateStorage('configs', configs);
|
||||
updateStorage('chatbotSrc', chatbotSrc);
|
||||
}
|
||||
loadConfigs(configs, chatbotSrc);
|
||||
row.remove();
|
||||
enableAllEditButtons();
|
||||
}
|
||||
|
||||
|
||||
function handleCancelButtonClick(row, isNewConfig = false) {
|
||||
if (isNewConfig) {
|
||||
row.remove();
|
||||
} else {
|
||||
const index = Array.from(document.getElementById('configList').children).indexOf(row);
|
||||
const config = configs[index];
|
||||
row.querySelector('td:nth-child(1)').textContent = config.name;
|
||||
row.querySelector('td:nth-child(2)').textContent = config.url;
|
||||
row.querySelector('.editButton').disabled = false;
|
||||
// 移除编辑模式下的确认和取消按钮
|
||||
const iconContainer = row.querySelector('td:nth-child(3) > div');
|
||||
if (iconContainer) {
|
||||
iconContainer.remove();
|
||||
}
|
||||
}
|
||||
enableAllEditButtons();
|
||||
}
|
||||
|
||||
async function handleShowChatBotChange(showChatBot) {
|
||||
await updateStorage('showChatBot', showChatBot);
|
||||
}
|
||||
|
||||
async function handleChatBotWidthChange(width) {
|
||||
const parsedWidth = parseInt(width);
|
||||
if (!isNaN(parsedWidth)) {
|
||||
await updateStorage('chatBotWidth', parsedWidth);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleChatBotHeightChange(height) {
|
||||
const parsedHeight = parseInt(height);
|
||||
if (!isNaN(parsedHeight)) {
|
||||
await updateStorage('chatBotHeight', parsedHeight);
|
||||
}
|
||||
}
|
||||
|
||||
function disableAllEditButtons() {
|
||||
const allEditButtons = document.querySelectorAll('.editButton');
|
||||
allEditButtons.forEach(button => {
|
||||
button.disabled = true;
|
||||
});
|
||||
}
|
||||
|
||||
function enableAllEditButtons() {
|
||||
const allEditButtons = document.querySelectorAll('.editButton');
|
||||
allEditButtons.forEach(button => {
|
||||
button.disabled = false;
|
||||
});
|
||||
}
|
Reference in New Issue
Block a user