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
1 lines • 62.2 kB
JavaScript
!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).VueChatButtonGlobal={})}(this,(function(t){"use strict";const e={name:"ChatModal",props:{visible:{type:Boolean,default:!1},position:{type:String,default:"center"},width:{type:String,default:"400px"},height:{type:String,default:"600px"},closeOnOverlay:{type:Boolean,default:!1},closeOnEscape:{type:Boolean,default:!1},draggable:{type:Boolean,default:!0},resizable:{type:Boolean,default:!0},initialX:{type:Number,default:void 0},initialY:{type:Number,default:void 0},initialWidth:{type:Number,default:void 0},initialHeight:{type:Number,default:void 0},enableBoundary:{type:Boolean,default:!0},showFullscreen:{type:Boolean,default:!1},enableAdjustSize:{type:Boolean,default:!1}},data:()=>({iframeUrl:"",modalTitle:"西之月客服",visible:!1,modalInstance:null,modalElement:null,isFullscreen:!1,isDragging:!1,isResizing:!1,hasBeenDragged:!1,modalPosition:{x:0,y:0},modalSize:{width:400,height:600},dragStart:{x:0,y:0},resizeStart:{x:0,y:0,width:0,height:0}}),computed:{modalStyle(){return this.isFullscreen||"fullscreen"===this.position?{width:"100vw",height:"100vh",left:"0",top:"0",right:"0",bottom:"0",transform:"none",position:"fixed",maxWidth:"none",maxHeight:"none",margin:"0",borderRadius:"0"}:this.hasBeenDragged||"custom"===this.position||0!==this.modalPosition.x&&0!==this.modalPosition.y?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:this.modalPosition.x+"px",top:this.modalPosition.y+"px",transform:"none",position:"fixed"}:"center"===this.position?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:"50%",top:"50%",transform:"translate(-50%, -50%)",position:"fixed"}:"right"===this.position?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",right:"20px",top:"50%",transform:"translateY(-50%)",position:"fixed"}:{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:this.modalPosition.x+"px",top:this.modalPosition.y+"px",transform:"none",position:"fixed"}}},mounted(){this.closeOnEscape&&document.addEventListener("keydown",this.handleKeydown,!0),this.initializePosition()},beforeDestroy(){document.removeEventListener("keydown",this.handleKeydown,!0),this.destroyModal(),this.stopDrag(),this.stopResize()},methods:{createModal(){if(this.modalInstance)return;const t=document.createElement("div");t.id="chat-modal-container-"+Date.now(),document.body.appendChild(t);const e={template:'\n <div v-if="visible" class="chat-modal-overlay" @click="handleOverlayClick">\n <div \n ref="modalRef"\n class="chat-modal-container" \n :class="[position, { \'fullscreen\': isFullscreen, \'dragging\': isDragging }]" \n :style="modalStyle" \n @click.stop\n >\n \x3c!-- 弹窗头部 --\x3e\n <div \n class="chat-modal-header" \n :class="{ \'draggable\': !isFullscreen }"\n @mousedown="startDrag"\n @touchstart="startDrag"\n >\n <div class="chat-modal-title center">\n <img :src="logoUrl" alt="logo" class="chat-modal-logo" />\n <span>{{ modalTitle }}</span>\n </div>\n <div class="chat-modal-controls" @mousedown.stop @touchstart.stop>\n <template v-if="showFullscreen">\n \x3c!-- 全屏切换按钮 --\x3e\n <button\n v-if="!isFullscreen"\n class="chat-modal-close full-screen-icon" \n @click="toggleFullscreen"\n title="全屏"\n >\n <svg viewBox="0 0 1024 1024" width="20" height="20">\n <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"/>\n </svg>\n </button>\n </template>\n \x3c!-- 关闭按钮 --\x3e\n <button class="chat-modal-close" @click="closeModal">\n <svg viewBox="0 0 1024 1024" width="20" height="20">\n <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"/>\n </svg>\n </button>\n <button class="chat-modal-close" @click="closeModal">\n <svg viewBox="0 0 1024 1024" fill="red" width="20" height="20">\n <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"/>\n </svg>\n </button>\n </div>\n </div>\n\n \x3c!-- 弹窗内容 --\x3e\n <div class="chat-modal-content">\n <iframe v-if="iframeUrl" :src="iframeUrl" :title="modalTitle" class="chat-modal-iframe" frameborder="0" allowfullscreen></iframe>\n <div v-else class="chat-modal-placeholder">\n <p>请设置 iframe URL</p>\n </div>\n </div>\n\n \x3c!-- 调整大小手柄 --\x3e\n <div \n v-if="!isFullscreen && position !== \'fullscreen\' && enableAdjustSize"\n class="chat-modal-resize-handle"\n @mousedown="startResize"\n @touchstart="startResize"\n ></div>\n </div>\n </div>\n ',props:["visible","position","width","height","closeOnOverlay","closeOnEscape","draggable","resizable","initialX","initialY","initialWidth","initialHeight","enableBoundary","showFullscreen","enableAdjustSize"],data:()=>({iframeUrl:"",modalTitle:"西之月客服",logoUrl:"",isFullscreen:!1,isDragging:!1,isResizing:!1,hasBeenDragged:!1,modalPosition:{x:0,y:0},modalSize:{width:400,height:600},dragStart:{x:0,y:0},resizeStart:{x:0,y:0,width:0,height:0}}),computed:{modalStyle(){return this.isFullscreen||"fullscreen"===this.position?{width:"100vw",height:"100vh",left:"0",top:"0",right:"0",bottom:"0",transform:"none",position:"fixed",maxWidth:"none",maxHeight:"none",margin:"0",borderRadius:"0"}:this.hasBeenDragged||"custom"===this.position||0!==this.modalPosition.x&&0!==this.modalPosition.y?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:this.modalPosition.x+"px",top:this.modalPosition.y+"px",transform:"none",position:"fixed"}:"center"===this.position?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:"50%",top:"50%",transform:"translate(-50%, -50%)",position:"fixed"}:"right"===this.position?{width:this.modalSize.width+"px",height:this.modalSize.height+"px",right:"20px",top:"50%",transform:"translateY(-50%)",position:"fixed"}:{width:this.modalSize.width+"px",height:this.modalSize.height+"px",left:this.modalPosition.x+"px",top:this.modalPosition.y+"px",transform:"none",position:"fixed"}}},methods:{closeModal(){this.$emit("close")},handleOverlayClick(){this.closeOnOverlay&&this.closeModal()},initializePosition(){if(this.hasBeenDragged=!1,this.initialWidth&&this.initialHeight)this.modalSize={width:this.initialWidth,height:this.initialHeight};else{const t=parseInt(this.width),e=parseInt(this.height);this.modalSize={width:t,height:e}}void 0!==this.initialX&&void 0!==this.initialY?(this.modalPosition={x:this.initialX,y:this.initialY},this.hasBeenDragged=!0):"center"===this.position?this.modalPosition={x:(window.innerWidth-this.modalSize.width)/2,y:(window.innerHeight-this.modalSize.height)/2}:"right"===this.position?this.modalPosition={x:window.innerWidth-this.modalSize.width-20,y:(window.innerHeight-this.modalSize.height)/2}:this.modalPosition={x:100,y:100}},resetPosition(){this.initializePosition()},toggleFullscreen(){if(this.isFullscreen=!this.isFullscreen,this.$emit("fullscreen",this.isFullscreen),this.isFullscreen){const t=this.$refs.modalRef?this.$refs.modalRef.getBoundingClientRect():null;t&&(this.modalPosition={x:t.left,y:t.top},this.modalSize={width:t.width,height:t.height})}},startDrag(t){if(this.isFullscreen||!this.draggable)return;t.preventDefault(),this.isDragging=!0,this.hasBeenDragged=!0;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY,n=this.$refs.modalRef?this.$refs.modalRef.getBoundingClientRect():null;n&&("center"===this.position||"right"===this.position||0===this.modalPosition.x&&0===this.modalPosition.y)&&(this.modalPosition={x:n.left,y:n.top}),this.dragStart={x:e-this.modalPosition.x,y:i-this.modalPosition.y},document.addEventListener("mousemove",this.handleDrag),document.addEventListener("mouseup",this.stopDrag),document.addEventListener("touchmove",this.handleDrag),document.addEventListener("touchend",this.stopDrag)},handleDrag(t){if(!this.isDragging)return;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY;let n=e-this.dragStart.x,o=i-this.dragStart.y;if(this.enableBoundary){const t=0,e=window.innerWidth-this.modalSize.width,i=0,s=window.innerHeight-this.modalSize.height;n<t?n=t:n>e&&(n=e),o<i?o=i:o>s&&(o=s)}this.modalPosition={x:n,y:o},this.$emit("move",this.modalPosition)},stopDrag(){this.isDragging=!1,document.removeEventListener("mousemove",this.handleDrag),document.removeEventListener("mouseup",this.stopDrag),document.removeEventListener("touchmove",this.handleDrag),document.removeEventListener("touchend",this.stopDrag)},startResize(t){if(this.isFullscreen||!this.resizable)return;t.preventDefault(),this.isResizing=!0;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY;this.resizeStart={x:e,y:i,width:this.modalSize.width,height:this.modalSize.height},document.addEventListener("mousemove",this.handleResize),document.addEventListener("mouseup",this.stopResize),document.addEventListener("touchmove",this.handleResize),document.addEventListener("touchend",this.stopResize)},handleResize(t){if(!this.isResizing)return;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY,n=e-this.resizeStart.x,o=i-this.resizeStart.y;let s=Math.max(300,this.resizeStart.width+n),a=Math.max(200,this.resizeStart.height+o);if(this.enableBoundary){const t=window.innerWidth-this.modalPosition.x,e=window.innerHeight-this.modalPosition.y;s>t&&(s=t),a>e&&(a=e)}this.modalSize={width:s,height:a},this.$emit("resize",this.modalSize)},stopResize(){this.isResizing=!1,document.removeEventListener("mousemove",this.handleResize),document.removeEventListener("mouseup",this.stopResize),document.removeEventListener("touchmove",this.handleResize),document.removeEventListener("touchend",this.stopResize)}}};this.modalInstance=new Vue({render:h=>h(e,{props:{visible:this.visible,position:this.position,width:this.width,height:this.height,closeOnOverlay:this.closeOnOverlay,closeOnEscape:this.closeOnEscape,draggable:this.draggable,resizable:this.resizable,initialX:this.initialX,initialY:this.initialY,initialWidth:this.initialWidth,initialHeight:this.initialHeight,enableBoundary:this.enableBoundary,showFullscreen:this.showFullscreen,enableAdjustSize:this.enableAdjustSize},on:{close:()=>{this.closeModal(),this.$emit("close")},fullscreen:t=>this.$emit("fullscreen",t),move:t=>this.$emit("move",t),resize:t=>this.$emit("resize",t)}})}),this.modalInstance.$mount(t),this.modalElement=t,this.modalInstance.$children[0]&&this.modalInstance.$children[0].initializePosition&&this.modalInstance.$children[0].initializePosition()},destroyModal(){this.modalInstance&&(this.modalInstance.$destroy(),this.modalInstance=null),this.modalElement&&this.modalElement.parentNode&&(this.modalElement.parentNode.removeChild(this.modalElement),this.modalElement=null)},closeModal(){this.visible=!1,this.isFullscreen=!1,this.$emit("update:visible",!1),this.modalInstance&&(this.modalInstance.$children[0].visible=!1,this.modalInstance.$children[0].isFullscreen=!1)},handleOverlayClick(){this.closeOnOverlay&&this.closeModal()},handleKeydown(t){"Escape"===t.key&&this.visible&&(this.isFullscreen?this.toggleFullscreen():this.closeOnEscape&&(t.preventDefault(),this.closeModal()))},initializePosition(){if(this.hasBeenDragged=!1,this.initialWidth&&this.initialHeight)this.modalSize={width:this.initialWidth,height:this.initialHeight};else{const t=parseInt(this.width),e=parseInt(this.height);this.modalSize={width:t,height:e}}void 0!==this.initialX&&void 0!==this.initialY?(this.modalPosition={x:this.initialX,y:this.initialY},this.hasBeenDragged=!0):"center"===this.position?this.modalPosition={x:(window.innerWidth-this.modalSize.width)/2,y:(window.innerHeight-this.modalSize.height)/2}:"right"===this.position?this.modalPosition={x:window.innerWidth-this.modalSize.width-20,y:(window.innerHeight-this.modalSize.height)/2}:this.modalPosition={x:100,y:100}},resetPosition(){this.initializePosition()},startDrag(t){if(this.isFullscreen||!this.draggable)return;t.preventDefault(),this.isDragging=!0,this.hasBeenDragged=!0;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY,n=this.$refs.modalRef?this.$refs.modalRef.getBoundingClientRect():null;n&&("center"===this.position||"right"===this.position||0===this.modalPosition.x&&0===this.modalPosition.y)&&(this.modalPosition={x:n.left,y:n.top}),this.dragStart={x:e-this.modalPosition.x,y:i-this.modalPosition.y},document.addEventListener("mousemove",this.handleDrag),document.addEventListener("mouseup",this.stopDrag),document.addEventListener("touchmove",this.handleDrag),document.addEventListener("touchend",this.stopDrag)},handleDrag(t){if(!this.isDragging)return;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY;let n=e-this.dragStart.x,o=i-this.dragStart.y;if(this.enableBoundary){const t=0,e=window.innerWidth-this.modalSize.width,i=0,s=window.innerHeight-this.modalSize.height;n<t?n=t:n>e&&(n=e),o<i?o=i:o>s&&(o=s)}this.modalPosition={x:n,y:o},this.$emit("move",this.modalPosition)},stopDrag(){this.isDragging=!1,document.removeEventListener("mousemove",this.handleDrag),document.removeEventListener("mouseup",this.stopDrag),document.removeEventListener("touchmove",this.handleDrag),document.removeEventListener("touchend",this.stopDrag)},startResize(t){if(this.isFullscreen||!this.resizable)return;t.preventDefault(),this.isResizing=!0;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY;this.resizeStart={x:e,y:i,width:this.modalSize.width,height:this.modalSize.height},document.addEventListener("mousemove",this.handleResize),document.addEventListener("mouseup",this.stopResize),document.addEventListener("touchmove",this.handleResize),document.addEventListener("touchend",this.stopResize)},handleResize(t){if(!this.isResizing)return;const e="touches"in t?t.touches[0].clientX:t.clientX,i="touches"in t?t.touches[0].clientY:t.clientY,n=e-this.resizeStart.x,o=i-this.resizeStart.y;let s=Math.max(300,this.resizeStart.width+n),a=Math.max(200,this.resizeStart.height+o);if(this.enableBoundary){const t=window.innerWidth-this.modalPosition.x,e=window.innerHeight-this.modalPosition.y;s>t&&(s=t),a>e&&(a=e)}this.modalSize={width:s,height:a},this.$emit("resize",this.modalSize)},stopResize(){this.isResizing=!1,document.removeEventListener("mousemove",this.handleResize),document.removeEventListener("mouseup",this.stopResize),document.removeEventListener("touchmove",this.handleResize),document.removeEventListener("touchend",this.stopResize)},open(t,e="西之月客服"){this.iframeUrl=t,this.modalTitle=e,this.visible=!0,this.$nextTick((()=>{if(this.createModal(),this.modalInstance&&this.modalInstance.$children[0]){const i=this.modalInstance.$children[0];i.iframeUrl=t,i.modalTitle=e,i.visible=!0,i.initializePosition&&i.initializePosition()}})),this.$emit("update:visible",!0),console.log("ChatModal open 方法被调用,visible:",this.visible)},close(){this.closeModal()},toggle(){this.visible=!this.visible,this.$emit("update:visible",this.visible)},setPosition(t,e){if(this.modalInstance&&this.modalInstance.$children[0]){const i=this.modalInstance.$children[0];i.modalPosition={x:t,y:e},i.hasBeenDragged=!0,this.$emit("move",{x:t,y:e})}},setSize(t,e){if(this.modalInstance&&this.modalInstance.$children[0]){this.modalInstance.$children[0].modalSize={width:t,height:e},this.$emit("resize",{width:t,height:e})}},toggleFullscreen(){if(this.modalInstance&&this.modalInstance.$children[0]){this.modalInstance.$children[0].toggleFullscreen()}},isFullscreen(){if(this.modalInstance&&this.modalInstance.$children[0]){return this.modalInstance.$children[0].isFullscreen}return!1},handleLeftButtonClick(){this.$emit("left-button-click")}},render:createElement=>null},i={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"},n={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"},o={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"},s=(t,e)=>{const s=o[t];return{...{isProduction:"production"===t,apiBaseUrl:i[t],timeout:1e4,retryCount:3,modalUrl:n[t],wsUrl:e?`${s}?token=${e}`:s}}},a=()=>{localStorage.removeItem("im_token")},l=()=>localStorage.getItem("im_token"),d=t=>{localStorage.setItem("im_token",t)};class r{constructor(t,e={}){this.url=t,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:e.onOpen||(()=>{}),onMessage:e.onMessage||(()=>{}),onError:e.onError||(()=>{}),onClose:e.onClose||(()=>{}),onStatusChange:e.onStatusChange||(()=>{}),onBadgeUpdate:e.onBadgeUpdate||(()=>{})},this.status="disconnected"}on(t,e){this.callbacks[t]&&(this.callbacks[t]=e)}connect(t){return this.socket&&this.socket.readyState===WebSocket.OPEN?(console.log("WebSocket已经连接"),Promise.resolve()):new Promise(((e,i)=>{this.updateStatus("connecting"),this.currentToken=t;try{this.socket=new WebSocket(this.url,[t]),this.socket.onopen=t=>{this.handleOpen(t),e(t)},this.socket.onmessage=t=>{this.handleMessage(t)},this.socket.onerror=t=>{this.handleError(t),i(t)},this.socket.onclose=t=>{this.handleClose(t)}}catch(n){console.error("WebSocket连接错误:",n),this.updateStatus("disconnected"),this.callbacks.onError(n),i(n)}}))}handleOpen(t){console.log("WebSocket连接已建立"),this.reconnectAttempts=0,this.updateStatus("connected"),this.callbacks.onOpen(t),this.startHeartbeat()}handleMessage(t){try{const e=JSON.parse(t.data);this.callbacks.onBadgeUpdate(e.data.unread_data.default.total||0),this.callbacks.onMessage(e,t)}catch(e){this.callbacks.onMessage(t.data,t)}}handleError(t){this.updateStatus("disconnected"),this.callbacks.onError(t),this.stopHeartbeat()}handleClose(t){this.updateStatus("disconnected"),this.stopHeartbeat(),this.callbacks.onClose(t),1e3!==t.code&&this.reconnectAttempts<this.maxReconnectAttempts?this.scheduleReconnect():this.reconnectAttempts>=this.maxReconnectAttempts&&this.callbacks.onError(new Error("WebSocket重连次数已达上限"))}scheduleReconnect(){this.reconnectTimer&&clearTimeout(this.reconnectTimer),this.reconnectAttempts++,this.reconnectTimer=setTimeout((()=>{this.currentToken&&this.reconnectAttempts<=this.maxReconnectAttempts&&this.connect(this.currentToken).catch((t=>{console.error(`第${this.reconnectAttempts}次重连失败:`,t)}))}),this.reconnectInterval)}startHeartbeat(){this.stopHeartbeat(),this.heartbeatTimer=setInterval((()=>{this.socket&&this.socket.readyState===WebSocket.OPEN?this.send({type:"ping",timestamp:Date.now()}):this.stopHeartbeat()}),this.heartbeatInterval)}stopHeartbeat(){this.heartbeatTimer&&(clearInterval(this.heartbeatTimer),this.heartbeatTimer=null)}send(t){if(!this.socket||this.socket.readyState!==WebSocket.OPEN)return console.warn("WebSocket未连接,无法发送消息"),!1;try{const e="object"==typeof t?JSON.stringify(t):t;return this.socket.send(e),!0}catch(e){return console.error("WebSocket发送失败:",e),this.callbacks.onError(e),!1}}disconnect(){this.reconnectAttempts=this.maxReconnectAttempts,this.reconnectTimer&&(clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.stopHeartbeat(),this.socket&&(this.socket.close(1e3,"用户主动断开"),this.socket=null),this.updateStatus("disconnected")}updateStatus(t){this.status!==t&&(this.status=t,this.callbacks.onStatusChange(t))}getStatus(){return this.status}isConnected(){return this.socket&&this.socket.readyState===WebSocket.OPEN}resetReconnectCount(){this.reconnectAttempts=0}getReconnectAttempts(){return this.reconnectAttempts}reconnect(){return this.resetReconnectCount(),this.currentToken?this.connect(this.currentToken):Promise.reject(new Error("没有可用的token"))}}const c={name:"GlobalLoading",props:{visible:{type:Boolean,default:!1},text:{type:String,default:""},background:{type:String,default:"rgba(0, 0, 0, 0.5)"},zIndex:{type:Number,default:9999},size:{type:String,default:"40px"},color:{type:String,default:"#007BFF"},spinnerType:{type:String,default:"spinner",validator:t=>["spinner","dots","bars"].includes(t)},fullscreen:{type:Boolean,default:!0},target:{type:String,default:"body"}},data:()=>({loadingInstance:null,loadingElement:null,isVisible:!1}),watch:{visible:{handler(t){this.$nextTick((()=>{t?this.show():this.hide()}))},immediate:!0}},mounted(){this.visible&&this.show()},beforeDestroy(){this.hide()},methods:{show(){this.isVisible||(this.createLoading(),this.isVisible=!0)},hide(){this.isVisible&&(this.destroyLoading(),this.isVisible=!1)},createLoading(){if(this.loadingInstance)return;this.ensureStyles();const t=document.createElement("div");t.id=`global-loading-container-${Date.now()}`,t.className="global-loading-container";const e={position:"fixed",top:"0",left:"0",width:"100%",height:"100%",backgroundColor:this.background,display:"flex",flexDirection:"column",justifyContent:"center",alignItems:"center",zIndex:this.zIndex,fontFamily:'-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif',pointerEvents:"auto"};Object.assign(t.style,e);const i=document.createElement("div");i.className="global-loading-content",i.style.cssText="\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n border-radius: 8px;\n padding: 24px;\n min-width: 120px;\n text-align: center;\n pointer-events: none;\n ";const n=this.createSpinner();if(i.appendChild(n),this.text){const t=document.createElement("div");t.className="global-loading-text",t.textContent=this.text,t.style.cssText="\n margin-top: 16px;\n color: #333;\n font-size: 14px;\n font-weight: 500;\n line-height: 1.4;\n white-space: nowrap;\n ",i.appendChild(t)}t.appendChild(i);const o="body"===this.target?document.body:document.querySelector(this.target);o&&(o.appendChild(t),this.loadingElement=t),this.loadingInstance=new Vue({el:t,data:()=>({visible:!0}),methods:{hide(){this.visible=!1,this.$emit("hide")}}})},ensureStyles(){if(document.getElementById("global-loading-styles"))return;const t=document.createElement("style");t.id="global-loading-styles",t.textContent="\n @keyframes spin {\n 0% { transform: rotate(0deg); }\n 100% { transform: rotate(360deg); }\n }\n \n @keyframes bounce {\n 0%, 80%, 100% {\n transform: scale(0);\n opacity: 0.3;\n }\n 40% {\n transform: scale(1);\n opacity: 1;\n }\n }\n \n @keyframes bars {\n 0%, 40%, 100% {\n transform: scaleY(0.4);\n opacity: 0.3;\n }\n 20% {\n transform: scaleY(1);\n opacity: 1;\n }\n }\n \n // .global-loading-container {\n // animation: fadeIn 0.2s ease-in-out;\n // }\n \n @keyframes fadeIn {\n from { opacity: 0; }\n to { opacity: 1; }\n }\n \n // .global-loading-content {\n // animation: slideIn 0.3s ease-out;\n // }\n \n @keyframes slideIn {\n from {\n opacity: 0;\n transform: translateY(-20px) scale(0.9);\n }\n to {\n opacity: 1;\n transform: translateY(0) scale(1);\n }\n }\n ",document.head.appendChild(t)},createSpinner(){const t=document.createElement("div");switch(t.className="global-loading-spinner",t.style.cssText=`\n width: ${this.size};\n height: ${this.size};\n display: flex;\n align-items: center;\n justify-content: center;\n flex-shrink: 0;\n `,this.spinnerType){case"spinner":default:this.createSpinnerAnimation(t);break;case"dots":this.createDotsAnimation(t);break;case"bars":this.createBarsAnimation(t)}return t},createSpinnerAnimation(t){const e=document.createElement("div");e.style.cssText=`\n width: 100%;\n height: 100%;\n border: 3px solid #f3f3f3;\n border-top: 3px solid ${this.color};\n border-radius: 50%;\n animation: spin 1s linear infinite;\n box-sizing: border-box;\n `,t.appendChild(e)},createDotsAnimation(t){const e=document.createElement("div");e.style.cssText="\n display: flex;\n gap: 6px;\n align-items: center;\n justify-content: center;\n width: 100%;\n height: 100%;\n ";for(let i=0;i<3;i++){const t=document.createElement("div");t.style.cssText=`\n width: 8px;\n height: 8px;\n background-color: ${this.color};\n border-radius: 50%;\n animation: bounce 1.4s ease-in-out infinite both;\n animation-delay: ${.16*i}s;\n flex-shrink: 0;\n `,e.appendChild(t)}t.appendChild(e)},createBarsAnimation(t){const e=document.createElement("div");e.style.cssText="\n display: flex;\n gap: 4px;\n align-items: center;\n justify-content: center;\n width: 100%;\n height: 100%;\n ";for(let i=0;i<4;i++){const t=document.createElement("div");t.style.cssText=`\n width: 4px;\n height: 20px;\n background-color: ${this.color};\n animation: bars 1.2s ease-in-out infinite both;\n animation-delay: ${.1*i}s;\n flex-shrink: 0;\n border-radius: 2px;\n `,e.appendChild(t)}t.appendChild(e)},destroyLoading(){if(this.loadingInstance){try{this.loadingInstance.$destroy()}catch(t){console.error(t)}this.loadingInstance=null}if(this.loadingElement)try{this.loadingElement.parentNode&&this.loadingElement.parentNode.removeChild(this.loadingElement),this.loadingElement=null}catch(t){console.error(t),this.loadingElement&&(this.loadingElement.style.display="none",this.loadingElement.style.visibility="hidden")}this.cleanupAllLoadingElements()},cleanupAllLoadingElements(){try{document.querySelectorAll(".global-loading-container").forEach((t=>{t.parentNode&&t.parentNode.removeChild(t)}));const t=document.getElementById("global-loading-styles");t&&t.parentNode&&t.parentNode.removeChild(t)}catch(t){console.error("GlobalLoading cleanupAllLoadingElements() - error:",t)}},updateText(t){if(this.text=t,this.loadingElement){const e=this.loadingElement.querySelector(".global-loading-text");e&&(e.textContent=t)}},updateBackground(t){this.background=t,this.loadingElement&&(this.loadingElement.style.backgroundColor=t)},updateColor(t){if(this.color=t,this.loadingElement){const t=this.loadingElement.querySelector(".global-loading-spinner");t&&(t.innerHTML="",this.createSpinner())}}},render:()=>null},m={name:"ChatButtonWithBadge",components:{ChatModal:e,GlobalLoading:c},props:{autoFetch:{type:Boolean,default:!0},params:{type:Object,default:()=>({})},iconUrl:{type:String,default:""},iconColor:{type:String,default:"#fff"},themeColor:{type:String,default:"#1890ff"},badgeColor:{type:String,default:"#ff3860"},environment:{type:String,default:"development",validator:t=>["development","production","test"].includes(t)},showBadge:{type:Boolean,default:!0},modalUrl:{type:String,default:""},modalTitle:{type:String,default:"西之月客服"},modalPosition:{type:String,default:"center",validator:t=>["right","center","fullscreen"].includes(t)},modalWidth:{type:String,default:"600px"},modalHeight:{type:String,default:"600px"},closeOnOverlay:{type:Boolean,default:!1},closeOnEscape:{type:Boolean,default:!1},isFixed:{type:Boolean,default:!0},fixedPosition:{type:String,default:"bottom-right",validator:t=>["bottom-right","bottom-left","top-right","top-left"].includes(t)},fixedOffset:{type:Object,default:()=>({x:20,y:60})},autoPosition:{type:Boolean,default:!0},isOrder:{type:Boolean,default:!1},isOpenTab:{type:Boolean,default:!1},path:{type:String,default:"/chat"},token:{type:String,required:!0},showGlobalLoading:{type:Boolean,default:!0},globalLoadingText:{type:String,default:""},globalLoadingDelay:{type:Number,default:100},loadingSpinnerType:{type:String,default:"spinner",validator:t=>["spinner","dots","bars"].includes(t)},loadingColor:{type:String,default:"#1890ff"},loadingSize:{type:String,default:"40px"}},data:()=>({badgeCount:0,loading:!1,globalLoading:!1,globalLoadingTimer:null,errorModalVisible:!1,errorModalMessage:"",accessToken:"",startChat:!1,wsClient:null,wsStatus:"disconnected",wsConnecting:!1}),computed:{buttonStyle(){return{background:this.themeColor,boxShadow:`0 4px 12px ${this.themeColor}30`,fill:this.iconColor}},badgeStyle(){return{background:this.badgeColor,boxShadow:`0 2px 4px ${this.badgeColor}30`}},fixedStyle(){if(!this.isFixed)return{};const{x:t=20,y:e=20}=this.fixedOffset||{};const i={position:"fixed",zIndex:"9999"};switch(this.fixedPosition||"bottom-right"){case"bottom-right":default:i.right=`${t}px`,i.bottom=`${e}px`;break;case"bottom-left":i.left=`${t}px`,i.bottom=`${e}px`;break;case"top-r