From 0b9a02afde56bb43c9c524be7e35c83905cfb948 Mon Sep 17 00:00:00 2001 From: xiaojunnuo Date: Wed, 14 May 2025 08:56:48 +0800 Subject: [PATCH] docs: docs --- packages/ui/certd-client/index.html | 11 ++ packages/ui/certd-client/public/iframe.js | 151 ++++++++++++++++++ .../plugin/upload-to-host/index.ts | 5 +- 3 files changed, 165 insertions(+), 2 deletions(-) create mode 100644 packages/ui/certd-client/public/iframe.js diff --git a/packages/ui/certd-client/index.html b/packages/ui/certd-client/index.html index 1740a304..f6398595 100644 --- a/packages/ui/certd-client/index.html +++ b/packages/ui/certd-client/index.html @@ -7,6 +7,17 @@ Loading +
diff --git a/packages/ui/certd-client/public/iframe.js b/packages/ui/certd-client/public/iframe.js new file mode 100644 index 00000000..22bfaf74 --- /dev/null +++ b/packages/ui/certd-client/public/iframe.js @@ -0,0 +1,151 @@ +function embedChatbot() { + const chatBtnId = "fastgpt-chatbot-button"; + const chatWindowId = "fastgpt-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;"; + + const ChatBtnDiv = document.createElement("img"); + ChatBtnDiv.src = defaultOpen ? CloseIcon : MessageIcon; + ChatBtnDiv.setAttribute("width", "100%"); + ChatBtnDiv.setAttribute("height", "100%"); + ChatBtnDiv.draggable = false; + + const parent = document.createElement("div"); + const iframe = document.createElement("iframe"); + iframe.allow = "*"; + iframe.referrerPolicy = "no-referrer"; + iframe.title = "FastGPT Chat Window"; + parent.id = chatWindowId; + iframe.src = botSrc; + parent.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; 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;"; + parent.style.visibility = defaultOpen ? "unset" : "hidden"; + iframe.style.cssText = "border: none;width:100%;height:100%"; + + document.body.appendChild(parent); + parent.appendChild(iframe); + + const tips = document.createElement("div"); + tips.innerText = "内容由AI生成,仅供参考"; + tips.style.cssText = "padding:5px;font-size:10px;display:flex; justify-content:center;border-top: 1px solid #ddd;"; + parent.appendChild(tips); + + const observer = new MutationObserver(mutations => { + mutations.forEach(mutation => { + if (mutation.type === "attributes" && mutation.attributeName === "data-bot-src") { + const newBotSrc = script.getAttribute("data-bot-src"); + if (newBotSrc) { + iframe.src = newBotSrc; + } + } + }); + }); + observer.observe(script, { + attributes: true, + attributeFilter: ["data-bot-src"], + }); + + let chatBtnDragged = false; + let chatBtnDown = false; + let chatBtnMouseX; + let chatBtnMouseY; + + const updateChatWindowPosition = () => { + const chatWindow = document.getElementById(chatWindowId); + const btn = ChatBtn.getBoundingClientRect(); + const [vw, vh, ww, wh] = [window.innerWidth, window.innerHeight, chatWindow.offsetWidth, chatWindow.offsetHeight]; + + let right = 0; + if (btn.left >= ww) { + right = vw - btn.left + 10; // 左侧有空间则放在左侧,间距 10 + } else if (vw - btn.right >= ww) { + right = vw - btn.right - ww - 10; // 右侧有空间则放在右侧 + } + + let bottom = Math.max(30, vh - btn.bottom); // 聊天窗口底部和按钮对齐,最少 30 + if (btn.top < wh) { + bottom = Math.min(bottom, vh - wh - 30); // 确保聊天窗口不超出顶部 + } + + chatWindow.style.right = `${right}px`; + chatWindow.style.bottom = `${bottom}px`; + }; + + 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(); + chatBtnMouseX = e.clientX; + chatBtnMouseY = e.clientY; + chatBtnDown = true; + + ChatBtn.initialRight = parseInt(ChatBtn.style.right) || 60; + ChatBtn.initialBottom = parseInt(ChatBtn.style.bottom) || 30; + }); + + window.addEventListener("mousemove", e => { + e.stopPropagation(); + if (!canDrag || !chatBtnDown) return; + + chatBtnDragged = true; + + const deltaX = e.clientX - chatBtnMouseX; + const deltaY = e.clientY - chatBtnMouseY; + + let newRight = ChatBtn.initialRight - deltaX; + let newBottom = ChatBtn.initialBottom - deltaY; + + newRight = Math.max(20, Math.min(window.innerWidth - 60, newRight)); + newBottom = Math.max(30, Math.min(window.innerHeight - 70, newBottom)); + + ChatBtn.style.right = `${newRight}px`; + ChatBtn.style.bottom = `${newBottom}px`; + + updateChatWindowPosition(); + }); + + window.addEventListener("mouseup", e => { + chatBtnDown = false; + }); + + ChatBtn.appendChild(ChatBtnDiv); + document.body.appendChild(ChatBtn); + updateChatWindowPosition(); +} +window.addEventListener("load", embedChatbot); diff --git a/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts b/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts index 4def0157..9a0c9f93 100644 --- a/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts +++ b/packages/ui/certd-server/src/plugins/plugin-host/plugin/upload-to-host/index.ts @@ -64,7 +64,7 @@ export class UploadCertToHostPlugin extends AbstractTaskPlugin { crtPath!: string; @TaskInput({ title: '私钥保存路径', - helper: '需要有写入权限,路径要包含私钥文件名,例如:/tmp/cert.key', + helper: '原本的私钥保存路径,需要有写入权限,路径要包含私钥文件名,例如:/tmp/cert.key', component: { placeholder: '/root/deploy/nginx/cert.key', }, @@ -212,8 +212,9 @@ export class UploadCertToHostPlugin extends AbstractTaskPlugin { name: 'a-textarea', vModel: 'value', rows: 6, + placeholder: 'systemctl restart nginx ', }, - helper: '上传后执行脚本命令,不填则不执行\n注意:如果目标主机是windows,且终端是cmd,系统会自动将多行命令通过“&&”连接成一行', + helper: '上传后执行脚本命令,让证书生效(比如重启nginx),不填则不执行\n注意:如果目标主机是windows,且终端是cmd,系统会自动将多行命令通过“&&”连接成一行', required: false, }) script!: string;