Files
FastGPT/projects/app/public/chrome_extension/fastgpt_agent/src/content.js
shilin f35ba8e5a7 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列表
2024-08-06 10:53:01 +08:00

534 lines
31 KiB
JavaScript

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
});
}
}
}
}
});