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
JavaScript
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