UNPKG

vue-chat-button-simple

Version:

Vue 2 & Vue 3 compatible chat button components with badge support, modal popup, fixed positioning, environment configuration, and customizable themes

857 lines 70.8 kB
import { toValue, ref, defineComponent, computed, watch, onMounted, onUnmounted, createBlock, openBlock, Teleport, createVNode, Transition, withCtx, createElementBlock, createCommentVNode, createElementVNode, withModifiers, normalizeStyle, normalizeClass, renderSlot, createTextVNode, toDisplayString, Fragment, withDirectives, vShow, unref } from "vue"; typeof WorkerGlobalScope !== "undefined" && globalThis instanceof WorkerGlobalScope; const noop = () => { }; function createFilterWrapper(filter, fn) { function wrapper(...args) { return new Promise((resolve, reject) => { Promise.resolve(filter(() => fn.apply(this, args), { fn, thisArg: this, args })).then(resolve).catch(reject); }); } return wrapper; } function debounceFilter(ms, options = {}) { let timer; let maxTimer; let lastRejector = noop; const _clearTimeout = (timer2) => { clearTimeout(timer2); lastRejector(); lastRejector = noop; }; let lastInvoker; const filter = (invoke) => { const duration = toValue(ms); const maxDuration = toValue(options.maxWait); if (timer) _clearTimeout(timer); if (duration <= 0 || maxDuration !== void 0 && maxDuration <= 0) { if (maxTimer) { _clearTimeout(maxTimer); maxTimer = void 0; } return Promise.resolve(invoke()); } return new Promise((resolve, reject) => { lastRejector = options.rejectOnCancel ? reject : resolve; lastInvoker = invoke; if (maxDuration && !maxTimer) { maxTimer = setTimeout(() => { if (timer) _clearTimeout(timer); maxTimer = void 0; resolve(lastInvoker()); }, maxDuration); } timer = setTimeout(() => { if (maxTimer) _clearTimeout(maxTimer); maxTimer = void 0; resolve(invoke()); }, duration); }); }; return filter; } // @__NO_SIDE_EFFECTS__ function useDebounceFn(fn, ms = 200, options = {}) { return createFilterWrapper( debounceFilter(ms, options), fn ); } const environmentMap = { development: "https://api.dev.szad.teehmoon.com/chat-service", test: "https://api.test.szad.teehmoon.com/chat-service", production: "https://api-x.westmonth.com/chat-service", uat: "https://api.uat.teehmoon.com/chat-service" }; const detectEnvironment = () => { if (typeof window !== "undefined") { const hostname = window.location.hostname; const protocol = window.location.protocol; if (hostname === "localhost" || hostname === "127.0.0.1" || hostname.includes(".local") || hostname.includes(".dev") || protocol === "file:") { return "development"; } const urlParams = new URLSearchParams(window.location.search); if (urlParams.get("env") === "dev") { return "development"; } return "production"; } return "development"; }; const modalUrlMap = { dev: "http://localhost:5666", development: "https://im.dev.szad.teehmoon.com", production: "https://im.westmonth.com", test: "https://im.test.szad.teehmoon.com", uat: "https://im.uat.teehmoon.com" }; const wsUrlMap = { development: "wss://api.test.szad.teehmoon.com/chat-service/init", test: "wss://api.test.szad.teehmoon.com/chat-service/init", uat: "wss://api.uat.teehmoon.com/chat-service/init", production: "wss://api-x.westmonth.com/chat-service/init" }; const getEnvironmentConfig = (environment, token) => { const isProduction = environment === "production"; const baseWsUrl = wsUrlMap[environment]; const wsUrl = token ? `${baseWsUrl}?token=${token}` : baseWsUrl; const defaultConfig = { isProduction, apiBaseUrl: environmentMap[environment], timeout: 1e4, retryCount: 3, modalUrl: modalUrlMap[environment], wsUrl }; return { ...defaultConfig }; }; const clearTokenCache = () => { localStorage.removeItem("im_token"); }; const globalModalInstance = ref(null); const globalModalState = ref({ visible: false, url: "", title: "西之月客服" }); const setGlobalModalInstance = (instance) => { globalModalInstance.value = instance; }; const openGlobalModal = (url, title = "西之月客服") => { if (globalModalInstance.value) { globalModalState.value.url = url; globalModalState.value.title = title; globalModalInstance.value.open(url, title); } else { console.warn("Global modal instance not found"); } }; const hasGlobalModalInstance = () => { return globalModalInstance.value !== null; }; class WebSocketClient { constructor(url, options = {}) { this.url = url; this.socket = null; this.reconnectAttempts = 0; this.maxReconnectAttempts = 5; this.reconnectInterval = 3e3; this.reconnectTimer = null; this.heartbeatTimer = null; this.heartbeatInterval = 3e3; this.pongTimeout = 5e3; this.callbacks = { onOpen: options.onOpen || (() => { }), onMessage: options.onMessage || (() => { }), onError: options.onError || (() => { }), onClose: options.onClose || (() => { }), onStatusChange: options.onStatusChange || (() => { }), onBadgeUpdate: options.onBadgeUpdate || (() => { }) }; this.status = "disconnected"; } // 设置事件回调 on(event, callback) { if (this.callbacks[event]) { this.callbacks[event] = callback; } } connect(token) { if (this.socket && this.socket.readyState === WebSocket.OPEN) { console.log("WebSocket已经连接"); return Promise.resolve(); } return new Promise((resolve, reject) => { this.updateStatus("connecting"); this.currentToken = token; try { this.socket = new WebSocket(this.url, [token]); this.socket.onopen = (event) => { this.handleOpen(event); resolve(event); }; this.socket.onmessage = (event) => { this.handleMessage(event); }; this.socket.onerror = (error) => { this.handleError(error); reject(error); }; this.socket.onclose = (event) => { this.handleClose(event); }; } catch (error) { console.error("WebSocket连接错误:", error); this.updateStatus("disconnected"); this.callbacks.onError(error); reject(error); } }); } handleOpen(event) { console.log("WebSocket连接已建立"); this.reconnectAttempts = 0; this.updateStatus("connected"); this.callbacks.onOpen(event); this.startHeartbeat(); } handleMessage(event) { try { const data = JSON.parse(event.data); this.callbacks.onBadgeUpdate(data.data.unread_data.default.total || 0); this.callbacks.onMessage(data, event); } catch (e) { this.callbacks.onMessage(event.data, event); } } handleError(error) { this.updateStatus("disconnected"); this.callbacks.onError(error); this.stopHeartbeat(); } handleClose(event) { this.updateStatus("disconnected"); this.stopHeartbeat(); this.callbacks.onClose(event); if (event.code !== 1e3 && this.reconnectAttempts < this.maxReconnectAttempts) { this.scheduleReconnect(); } else if (this.reconnectAttempts >= this.maxReconnectAttempts) { this.callbacks.onError(new Error("WebSocket重连次数已达上限")); } } scheduleReconnect() { if (this.reconnectTimer) { clearTimeout(this.reconnectTimer); } this.reconnectAttempts++; this.reconnectTimer = setTimeout(() => { if (this.currentToken && this.reconnectAttempts <= this.maxReconnectAttempts) { this.connect(this.currentToken).catch((error) => { console.error(`第${this.reconnectAttempts}次重连失败:`, error); }); } }, this.reconnectInterval); } startHeartbeat() { this.stopHeartbeat(); this.heartbeatTimer = setInterval(() => { if (this.socket && this.socket.readyState === WebSocket.OPEN) { this.send({ type: "ping", timestamp: Date.now() }); } else { this.stopHeartbeat(); } }, this.heartbeatInterval); } stopHeartbeat() { if (this.heartbeatTimer) { clearInterval(this.heartbeatTimer); this.heartbeatTimer = null; } } send(data) { if (this.socket && this.socket.readyState === WebSocket.OPEN) { try { const message = typeof data === "object" ? JSON.stringify(data) : data; this.socket.send(message); return true; } catch (error) { console.error("WebSocket发送失败:", error); this.callbacks.onError(error); return false; } } else { console.warn("WebSocket未连接,无法发送消息"); return false; } } disconnect() { this.reconnectAttempts = this.maxReconnectAttempts; if (this.reconnectTimer) { clearTimeout(this.reconnectTimer); this.reconnectTimer = null; } this.stopHeartbeat(); if (this.socket) { this.socket.close(1e3, "用户主动断开"); this.socket = null; } this.updateStatus("disconnected"); } updateStatus(status) { if (this.status !== status) { this.status = status; this.callbacks.onStatusChange(status); } } // 获取连接状态 getStatus() { return this.status; } // 检查是否已连接 isConnected() { return this.socket && this.socket.readyState === WebSocket.OPEN; } // 重置重连计数 resetReconnectCount() { this.reconnectAttempts = 0; } // 获取当前重连次数 getReconnectAttempts() { return this.reconnectAttempts; } // 手动重连(重置计数) reconnect() { this.resetReconnectCount(); if (this.currentToken) { return this.connect(this.currentToken); } return Promise.reject(new Error("没有可用的token")); } } const _imports_0 = ""; const _hoisted_1$2 = { class: "chat-modal-title center" }; const _hoisted_2$1 = { key: 0, t: "1756369066305", class: "icon", viewBox: "0 0 1024 1024", version: "1.1", xmlns: "http://www.w3.org/2000/svg", "p-id": "5356", width: "200", height: "200" }; const _hoisted_3$1 = { key: 1, t: "1756369303951", class: "icon", viewBox: "0 0 1024 1024", version: "1.1", xmlns: "http://www.w3.org/2000/svg", "p-id": "6339", width: "200", height: "200" }; const _hoisted_4$1 = { class: "chat-modal-content" }; const _hoisted_5$1 = ["src", "title"]; const _hoisted_6$1 = { class: "chat-modal-placeholder" }; const _sfc_main$2 = /* @__PURE__ */ defineComponent({ __name: "ChatModal", props: { visible: { type: Boolean, default: false }, position: { default: "center" }, width: { default: "400px" }, height: { default: "600px" }, closeOnOverlay: { type: Boolean, default: false }, closeOnEscape: { type: Boolean, default: false }, draggable: { type: Boolean, default: true }, resizable: { type: Boolean, default: true }, initialX: { default: void 0 }, initialY: { default: void 0 }, initialWidth: { default: void 0 }, initialHeight: { default: void 0 }, enableBoundary: { type: Boolean, default: true }, showFullscreen: { type: Boolean, default: false }, enableAdjustSize: { type: Boolean, default: false } }, emits: ["update:visible", "close", "open", "fullscreen", "move", "resize"], setup(__props, { expose: __expose, emit: __emit }) { const props = __props; const iframeUrl = ref(""); const modalTitle = ref("西之月客服"); const iframeKey = ref(0); const isFullscreen = ref(false); const isDragging = ref(false); const isResizing = ref(false); const hasBeenDragged = ref(false); const modalPosition = ref({ x: 0, y: 0 }); const modalSize = ref({ width: 400, height: 600 }); const dragStart = ref({ x: 0, y: 0 }); const resizeStart = ref({ x: 0, y: 0, width: 0, height: 0 }); const modalRef = ref(); const modalOverlayRef = ref(); const emit = __emit; const visible = ref(props.visible); const modalStyle = computed(() => { if (isFullscreen.value || props.position === "fullscreen") { return { width: "100vw", height: "100vh", left: "0", top: "0", right: "0", bottom: "0", transform: "none", position: "fixed", maxWidth: "none", maxHeight: "none", margin: "0", borderRadius: "0" }; } if (hasBeenDragged.value || props.position === "custom" || modalPosition.value.x !== 0 && modalPosition.value.y !== 0) { return { width: `${modalSize.value.width}px`, height: `${modalSize.value.height}px`, left: `${modalPosition.value.x}px`, top: `${modalPosition.value.y}px`, transform: "none", position: "fixed" }; } if (props.position === "center") { return { width: `${modalSize.value.width}px`, height: `${modalSize.value.height}px`, left: "50%", top: "50%", transform: "translate(-50%, -50%)", position: "fixed" }; } if (props.position === "right") { return { width: `${modalSize.value.width}px`, height: `${modalSize.value.height}px`, right: "20px", top: "50%", transform: "translateY(-50%)", position: "fixed" }; } return { width: `${modalSize.value.width}px`, height: `${modalSize.value.height}px`, left: `${modalPosition.value.x}px`, top: `${modalPosition.value.y}px`, transform: "none", position: "fixed" }; }); watch( () => props.visible, (newValue) => { visible.value = newValue; if (newValue) { initializePosition(); } } ); watch(visible, (newValue) => { emit("update:visible", newValue); if (newValue) { emit("open"); } else { emit("close"); } }); const initializePosition = () => { hasBeenDragged.value = false; if (props.initialWidth && props.initialHeight) { modalSize.value = { width: props.initialWidth, height: props.initialHeight }; } else { const width = parseInt(props.width); const height = parseInt(props.height); modalSize.value = { width, height }; } if (props.initialX !== void 0 && props.initialY !== void 0) { modalPosition.value = { x: props.initialX, y: props.initialY }; hasBeenDragged.value = true; } else if (props.position === "center") { modalPosition.value = { x: (window.innerWidth - modalSize.value.width) / 2, y: (window.innerHeight - modalSize.value.height) / 2 }; } else if (props.position === "right") { modalPosition.value = { x: window.innerWidth - modalSize.value.width - 20, y: (window.innerHeight - modalSize.value.height) / 2 }; } else { modalPosition.value = { x: 100, y: 100 }; } }; const resetPosition = () => { initializePosition(); }; const toggleFullscreen = () => { isFullscreen.value = !isFullscreen.value; emit("fullscreen", isFullscreen.value); if (isFullscreen.value) { if (modalRef.value) { const rect = modalRef.value.getBoundingClientRect(); modalPosition.value = { x: rect.left, y: rect.top }; modalSize.value = { width: rect.width, height: rect.height }; } } }; const startDrag = (event) => { var _a; if (isFullscreen.value || !props.draggable) return; event.preventDefault(); isDragging.value = true; hasBeenDragged.value = true; const clientX = "touches" in event ? event.touches[0].clientX : event.clientX; const clientY = "touches" in event ? event.touches[0].clientY : event.clientY; const rect = (_a = modalRef.value) == null ? void 0 : _a.getBoundingClientRect(); if (rect) { if (props.position === "center" || props.position === "right") { modalPosition.value = { x: rect.left, y: rect.top }; } else if (modalPosition.value.x === 0 && modalPosition.value.y === 0) { modalPosition.value = { x: rect.left, y: rect.top }; } } dragStart.value = { x: clientX - modalPosition.value.x, y: clientY - modalPosition.value.y }; document.addEventListener("mousemove", handleDrag); document.addEventListener("mouseup", stopDrag); document.addEventListener("touchmove", handleDrag); document.addEventListener("touchend", stopDrag); }; const handleDrag = (event) => { if (!isDragging.value) return; const clientX = "touches" in event ? event.touches[0].clientX : event.clientX; const clientY = "touches" in event ? event.touches[0].clientY : event.clientY; let newX = clientX - dragStart.value.x; let newY = clientY - dragStart.value.y; if (props.enableBoundary) { const minX = 0; const maxX = window.innerWidth - modalSize.value.width; const minY = 0; const maxY = window.innerHeight - modalSize.value.height; if (newX < minX) { newX = minX; } else if (newX > maxX) { newX = maxX; } if (newY < minY) { newY = minY; } else if (newY > maxY) { newY = maxY; } } modalPosition.value = { x: newX, y: newY }; emit("move", modalPosition.value); }; const stopDrag = () => { isDragging.value = false; document.removeEventListener("mousemove", handleDrag); document.removeEventListener("mouseup", stopDrag); document.removeEventListener("touchmove", handleDrag); document.removeEventListener("touchend", stopDrag); }; const startResize = (event) => { if (isFullscreen.value || !props.resizable) return; event.preventDefault(); isResizing.value = true; const clientX = "touches" in event ? event.touches[0].clientX : event.clientX; const clientY = "touches" in event ? event.touches[0].clientY : event.clientY; resizeStart.value = { x: clientX, y: clientY, width: modalSize.value.width, height: modalSize.value.height }; document.addEventListener("mousemove", handleResize); document.addEventListener("mouseup", stopResize); document.addEventListener("touchmove", handleResize); document.addEventListener("touchend", stopResize); }; const handleResize = (event) => { if (!isResizing.value) return; const clientX = "touches" in event ? event.touches[0].clientX : event.clientX; const clientY = "touches" in event ? event.touches[0].clientY : event.clientY; const deltaX = clientX - resizeStart.value.x; const deltaY = clientY - resizeStart.value.y; let newWidth = Math.max(300, resizeStart.value.width + deltaX); let newHeight = Math.max(200, resizeStart.value.height + deltaY); if (props.enableBoundary) { const maxWidth = window.innerWidth - modalPosition.value.x; const maxHeight = window.innerHeight - modalPosition.value.y; if (newWidth > maxWidth) { newWidth = maxWidth; } if (newHeight > maxHeight) { newHeight = maxHeight; } } modalSize.value = { width: newWidth, height: newHeight }; emit("resize", modalSize.value); }; const stopResize = () => { isResizing.value = false; document.removeEventListener("mousemove", handleResize); document.removeEventListener("mouseup", stopResize); document.removeEventListener("touchmove", handleResize); document.removeEventListener("touchend", stopResize); }; const closeModal = () => { visible.value = false; isFullscreen.value = false; emit("close"); }; const handleOverlayClick = () => { if (props.closeOnOverlay) { closeModal(); } }; const handleKeydown = (event) => { if (event.key === "Escape" && visible.value) { if (isFullscreen.value) { toggleFullscreen(); } else if (props.closeOnEscape) { event.preventDefault(); closeModal(); } } }; const open = async (url, title = "西之月客服") => { modalTitle.value = title; visible.value = true; if (!modalOverlayRef.value) { resetPosition(); } iframeKey.value = Date.now(); iframeUrl.value = `${url}?time=${iframeKey.value}`; }; const toggle = () => { visible.value = !visible.value; }; const setPosition = (x, y) => { modalPosition.value = { x, y }; emit("move", modalPosition.value); }; const setSize = (width, height) => { modalSize.value = { width, height }; emit("resize", modalSize.value); }; const instanceMethods = { open, close: closeModal, toggle, setPosition, setSize, toggleFullscreen, isFullscreen: () => isFullscreen.value }; __expose(instanceMethods); onMounted(() => { setGlobalModalInstance(instanceMethods); resetPosition(); document.addEventListener("keydown", handleKeydown, true); }); onUnmounted(() => { document.removeEventListener("keydown", handleKeydown, true); stopDrag(); stopResize(); }); return (_ctx, _cache) => { return openBlock(), createBlock(Teleport, { to: "body" }, [ createVNode(Transition, { name: "modal-fade" }, { default: withCtx(() => [ visible.value ? (openBlock(), createElementBlock("div", { key: 0, class: "chat-modal-overlay", ref_key: "modalOverlayRef", ref: modalOverlayRef, onClick: handleOverlayClick }, [ createElementVNode("div", { ref_key: "modalRef", ref: modalRef, class: normalizeClass(["chat-modal-container", [_ctx.position, { fullscreen: isFullscreen.value, dragging: isDragging.value }]]), style: normalizeStyle(modalStyle.value), onClick: _cache[2] || (_cache[2] = withModifiers(() => { }, ["stop"])) }, [ createElementVNode("div", { class: normalizeClass(["chat-modal-header", { draggable: !isFullscreen.value }]), onMousedown: startDrag, onTouchstart: startDrag }, [ createElementVNode("div", _hoisted_1$2, [ _cache[3] || (_cache[3] = createElementVNode("img", { src: _imports_0, alt: "logo", class: "chat-modal-logo" }, null, -1)), renderSlot(_ctx.$slots, "title", {}, () => [ createTextVNode(toDisplayString(modalTitle.value), 1) ]) ]), createElementVNode("div", { class: "chat-modal-controls", onMousedown: _cache[0] || (_cache[0] = withModifiers(() => { }, ["stop"])), onTouchstart: _cache[1] || (_cache[1] = withModifiers(() => { }, ["stop"])) }, [ _ctx.showFullscreen ? (openBlock(), createElementBlock(Fragment, { key: 0 }, [ !isFullscreen.value ? (openBlock(), createElementBlock("button", { key: 0, class: "chat-modal-close full-screen-icon", onClick: toggleFullscreen, title: "全屏" }, [ isFullscreen.value ? (openBlock(), createElementBlock("svg", _hoisted_2$1, _cache[4] || (_cache[4] = [ createElementVNode("path", { d: "M131.413333 85.333333h142.933334a32 32 0 1 0 0-64H54.186667a32 32 0 0 0-32 32v221.013334a32 32 0 1 0 64 0V131.413333l344.32 344.32a32 32 0 0 0 45.226666-45.226666zM969.813333 717.653333a32 32 0 0 0-32 32v142.506667l-344.32-344.32a32 32 0 0 0-45.226666 45.226667l344.32 345.6h-142.933334a32 32 0 0 0 0 64h220.16a32 32 0 0 0 32-32v-221.013334a32 32 0 0 0-32-32zM999.253333 42.666667a32 32 0 0 0-29.44-19.626667h-220.16a32 32 0 1 0 0 64h142.506667l-344.32 344.32a32 32 0 1 0 45.226667 45.226667L938.666667 131.413333v142.933334a32 32 0 1 0 64 0V54.186667a32 32 0 0 0-3.413334-11.52zM430.506667 548.266667L85.333333 892.586667v-142.933334a32 32 0 1 0-64 0v220.16a32 32 0 0 0 32 32h221.013334a32 32 0 1 0 0-64H131.413333l344.32-344.32a32 32 0 0 0-45.226666-45.226666z", fill: "", "p-id": "5357" }, null, -1) ]))) : (openBlock(), createElementBlock("svg", _hoisted_3$1, _cache[5] || (_cache[5] = [ createElementVNode("path", { d: "M999.908535 950.445791l-224.725864-224.789851h212.184216a36.473161 36.473161 0 1 0 0-73.138287H650.214103v334.849216a36.473161 36.473161 0 1 0 73.138286 0v-209.880648l224.725864 224.725864a36.601137 36.601137 0 0 0 51.766294-51.830282zM300.583659 36.697119v214.423796L74.706011 25.243267a36.601137 36.601137 0 0 0-51.766294 51.766294L247.729569 301.735425H36.569161a36.473161 36.473161 0 1 0 0 73.138286H373.785933V36.633131a36.473161 36.473161 0 1 0-73.138286 0zM998.756751 28.570643a36.601137 36.601137 0 0 0-51.766294 0l-223.57408 223.638068V36.633131a36.473161 36.473161 0 1 0-73.138286 0v338.24058h337.088796a36.473161 36.473161 0 1 0 0-73.138286h-209.880648l221.270512-221.3345a36.601137 36.601137 0 0 0 0-51.766294zM0.096 689.11879a36.473161 36.473161 0 0 0 36.537149 36.53715h213.336L21.851921 953.837156a36.601137 36.601137 0 0 0 51.830282 51.830281l226.965444-227.09342v208.792852a36.473161 36.473161 0 1 0 73.138286 0v-334.849216H36.633149a36.473161 36.473161 0 0 0-36.537149 36.601137z", fill: "#374866", "p-id": "6340" }, null, -1) ]))) ])) : createCommentVNode("", true) ], 64)) : createCommentVNode("", true), createElementVNode("button", { class: "chat-modal-close", onClick: closeModal }, _cache[6] || (_cache[6] = [ createElementVNode("svg", { t: "1755847270165", class: "icon", viewBox: "0 0 1024 1024", version: "1.1", xmlns: "http://www.w3.org/2000/svg", "p-id": "5200", width: "200", height: "200" }, [ createElementVNode("path", { d: "M213.333333 469.333333m21.333334 0l554.666666 0q21.333333 0 21.333334 21.333334l0 42.666666q0 21.333333-21.333334 21.333334l-554.666666 0q-21.333333 0-21.333334-21.333334l0-42.666666q0-21.333333 21.333334-21.333334Z", fill: "#4A4A4A", "p-id": "5201" }) ], -1) ])), createElementVNode("button", { class: "chat-modal-close", onClick: closeModal }, _cache[7] || (_cache[7] = [ createElementVNode("svg", { t: "1756369708796", class: "icon", viewBox: "0 0 1024 1024", fill: "red", version: "1.1", xmlns: "http://www.w3.org/2000/svg", "p-id": "7309", width: "200", height: "200" }, [ createElementVNode("path", { d: "M571.01312 523.776l311.3472-311.35232c15.7184-15.71328 15.7184-41.6256 0-57.344l-1.69472-1.69984c-15.7184-15.71328-41.6256-15.71328-57.34912 0l-311.3472 311.77728-311.35232-311.77728c-15.7184-15.71328-41.63072-15.71328-57.344 0l-1.69984 1.69984a40.0128 40.0128 0 0 0 0 57.344L452.92544 523.776l-311.35232 311.35744c-15.71328 15.71328-15.71328 41.63072 0 57.33888l1.69984 1.69984c15.71328 15.7184 41.6256 15.7184 57.344 0l311.35232-311.35232 311.3472 311.35232c15.72352 15.7184 41.63072 15.7184 57.34912 0l1.69472-1.69984c15.7184-15.70816 15.7184-41.6256 0-57.33888l-311.3472-311.35744z", "p-id": "7310" }) ], -1) ])) ], 32) ], 34), createElementVNode("div", _hoisted_4$1, [ withDirectives((openBlock(), createElementBlock("iframe", { key: iframeKey.value, src: iframeUrl.value, title: modalTitle.value, class: "chat-modal-iframe", frameborder: "0", allowfullscreen: "" }, null, 8, _hoisted_5$1)), [ [vShow, iframeUrl.value] ]), withDirectives(createElementVNode("div", _hoisted_6$1, [ renderSlot(_ctx.$slots, "content", {}, () => [ _cache[8] || (_cache[8] = createElementVNode("p", null, "请设置 iframe URL", -1)) ]) ], 512), [ [vShow, !iframeUrl.value] ]) ]), !isFullscreen.value && _ctx.position !== "fullscreen" && _ctx.enableAdjustSize ? (openBlock(), createElementBlock("div", { key: 0, class: "chat-modal-resize-handle", onMousedown: startResize, onTouchstart: startResize }, null, 32)) : createCommentVNode("", true) ], 6) ], 512)) : createCommentVNode("", true) ]), _: 3 }) ]); }; } }); const _hoisted_1$1 = { class: "error-modal-title" }; const _sfc_main$1 = /* @__PURE__ */ defineComponent({ __name: "ErrorModal", props: { visible: { type: Boolean, default: false }, message: { default: "操作失败" }, autoClose: { type: Boolean, default: true }, autoCloseDelay: { default: 3e3 }, onClose: { type: Function, default: () => { } }, onRetry: { type: Function, default: () => { } } }, emits: ["close", "retry"], setup(__props, { emit: __emit }) { const props = __props; const emit = __emit; const autoCloseTimer = ref(null); const handleClose = () => { emit("close"); props.onClose(); }; const handleOverlayClick = () => { handleClose(); }; const startAutoCloseTimer = () => { if (props.autoClose && props.autoCloseDelay > 0) { if (autoCloseTimer.value) { clearTimeout(autoCloseTimer.value); } autoCloseTimer.value = setTimeout(() => { handleClose(); }, props.autoCloseDelay); } }; const clearAutoCloseTimer = () => { if (autoCloseTimer.value) { clearTimeout(autoCloseTimer.value); autoCloseTimer.value = null; } }; watch(() => props.visible, (newVisible) => { if (newVisible) { document.body.style.overflow = "hidden"; startAutoCloseTimer(); } else { document.body.style.overflow = ""; clearAutoCloseTimer(); } }); onUnmounted(() => { clearAutoCloseTimer(); }); return (_ctx, _cache) => { return openBlock(), createBlock(Teleport, { to: "body" }, [ createVNode(Transition