diff --git a/web/public/js/chat.js b/web/public/js/chat.js new file mode 100644 index 00000000..ca685d44 --- /dev/null +++ b/web/public/js/chat.js @@ -0,0 +1,99 @@ +function embedChatbot() { + const chatBtnId = 'fastwiki-chatbot-button'; + const chatWindowId = 'fastwiki-chatbot-window'; + const script = document.getElementById('chatbot-iframe'); + const botSrc = script?.getAttribute('data-bot-src'); + const defaultOpen = script?.getAttribute('data-default-open') === 'true'; + const canDrag = script?.getAttribute('data-drag') === 'true'; + const MessageIcon = + script?.getAttribute('data-open-icon') || + ``; + const CloseIcon = + script?.getAttribute('data-close-icon') || + ''; + + if (!botSrc) { + console.error(`Can't find appid`); + return; + } + if (document.getElementById(chatBtnId)) { + return; + } + + 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;'; + + // btn icon + const ChatBtnDiv = document.createElement('img'); + ChatBtnDiv.src = defaultOpen ? CloseIcon : MessageIcon; + ChatBtnDiv.setAttribute('width', '100%'); + ChatBtnDiv.setAttribute('height', '100%'); + ChatBtnDiv.draggable = false; + + const iframe = document.createElement('iframe'); + iframe.allow = '*'; + iframe.referrerPolicy = 'no-referrer'; + iframe.title = 'fastwiki Chat Window'; + iframe.id = chatWindowId; + iframe.src = botSrc; + iframe.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; width: 375px; height: 667px; max-width: 90vw; max-height: 85vh; border-radius: 0.75rem; display: flex; z-index: 2147483647; overflow: hidden; left: unset; background-color: #F3F4F6;'; + iframe.style.visibility = defaultOpen ? 'unset' : 'hidden'; + + document.body.appendChild(iframe); + + let chatBtnDragged = false; + let chatBtnDown = false; + let chatBtnMouseX; + let chatBtnMouseY; + ChatBtn.addEventListener('click', function () { + if (chatBtnDragged) { + chatBtnDragged = false; + return; + } + const chatWindow = document.getElementById(chatWindowId); + + if (!chatWindow) return; + const visibilityVal = chatWindow.style.visibility; + if (visibilityVal === 'hidden') { + chatWindow.style.visibility = 'unset'; + ChatBtnDiv.src = CloseIcon; + } else { + chatWindow.style.visibility = 'hidden'; + ChatBtnDiv.src = MessageIcon; + } + }); + + ChatBtn.addEventListener('mousedown', (e) => { + e.stopPropagation(); + + if (!chatBtnMouseX && !chatBtnMouseY) { + chatBtnMouseX = e.clientX; + chatBtnMouseY = e.clientY; + } + + chatBtnDown = true; + }); + + window.addEventListener('mousemove', (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)`; + }); + + window.addEventListener('mouseup', (e) => { + chatBtnDown = false; + }); + + ChatBtn.appendChild(ChatBtnDiv); + document.body.appendChild(ChatBtn); + } + window.addEventListener('load', embedChatbot); + \ No newline at end of file diff --git a/web/src/layouts/(desktop)/index.tsx b/web/src/layouts/(desktop)/index.tsx index 8ff72647..4cdcd18b 100644 --- a/web/src/layouts/(desktop)/index.tsx +++ b/web/src/layouts/(desktop)/index.tsx @@ -1,5 +1,5 @@ -import { ActionIcon, Avatar,SideNav, Tooltip } from "@lobehub/ui"; -import { Album, Settings2, Box, User, BotMessageSquare,SquareFunction } from 'lucide-react'; +import { ActionIcon, Avatar, SideNav, Tooltip } from "@lobehub/ui"; +import { Album, Settings2, Box, User, BotMessageSquare, SquareFunction } from 'lucide-react'; import { memo, useEffect, useState } from "react"; import { Flexbox } from 'react-layout-kit'; import { Outlet } from "react-router-dom"; @@ -31,7 +31,7 @@ const DesktopLayout = memo(() => { description: '知识库', path: '/wiki', role: 'admin, user' - },{ + }, { icon: SquareFunction, key: 'function-call', description: '函数管理', diff --git a/web/src/pages/app-detail/feautres/ReleaseApplication.tsx b/web/src/pages/app-detail/feautres/ReleaseApplication.tsx index 384141d7..0b5b38f1 100644 --- a/web/src/pages/app-detail/feautres/ReleaseApplication.tsx +++ b/web/src/pages/app-detail/feautres/ReleaseApplication.tsx @@ -87,11 +87,11 @@ export default memo((props: IReleaseApplicationProps) => { label: '复制飞书对接地址', onClick: async () => { let url = config.FAST_API_URL; - if(!url){ + if (!url) { url = location.origin; } // 删除最后的/ - if(url.endsWith('/')){ + if (url.endsWith('/')) { url = url.slice(0, url.length - 1); } copyToClipboard(url + "/v1/feishu/completions/" + item.id) @@ -101,6 +101,32 @@ export default memo((props: IReleaseApplicationProps) => { items.push({ key: '4', + label: '复制悬浮球接入地址', + onClick: async () => { + let url = config.FAST_API_URL; + if (!url) { + url = location.origin; + } + + let value = ` + ` + + copyToClipboard(value) + message.success('复制成功'); + } + }) + + items.push({ + key: '5', label: '删除', onClick: async () => { await RemoveChatShare(item.id);