使用uniapp完成一個客服聊天功能


  起因

  最近在工作中,老大讓我重構一下商城的客服聊天頁面(因為之前是買來的代碼---php渲染惡心的一比,現在要用uniapp做多端應用,所以前端這塊的重構基本上就是重做)

  規划:

  1. 既然要做實時聊天呢,那自然少不了websocket的參與,當然了,在uniapp中,對websocket做了封裝和處理,我們直接使用它提供的api就行;
  2. 布局方面,其實只需要區分,自己發送的消息和別人發送的消息,自己發送的展示在右邊,別人發送的展示在左邊(這快比較簡單),然后你要展示什么樣的內容,那必須要有對應的標簽占位先(比如你要發送一個圖片,后端返回的是一個url,那你就需要留一個image標簽)
  3. 各種交互,需要考慮的是,每發送一個消息,都要滑動到最底部(剛進來之后同理)、還有輸入較多字符后,輸入框要增高一定的高度、點擊更多后要在底下展示發送圖片,拍照等按鈕、圖片預覽等(這塊很煩,h5,小程序,客戶端三端兼容很痛苦~)

  實際效果展示:

  

  代碼展示:

  嗯...,代碼好像有點多(采用的是uniapp + uview,所以組件、路由跳轉、網絡請求這些都是采用uview的方式)

 

<template>
    <view class="service">
        <!-- 頭部 -->
        <u-navbar :is-back="false" title=" " title-width="0" :background="background" :border-bottom="false" class="header"
         height="44">
            <view class="serviceTitle">
                <u-image :src="fanhui" alt @click="goBack" width="24rpx" height="40rpx" />
                <view v-if="!isSysMessage">{{userInfo.shopName}}</view>
                <view v-else>通知消息</view>
            </view>
            <view class="goShopBtn" @click="goShopDetail" v-if="!isSysMessage">進店</view>
        </u-navbar>
        <!-- 分割線 -->
        <view class="hrline"></view>

        <view class="goodHref" v-if="!isSysMessage && goodsId" @click="goGoodsDetail">
            <view class="goodContent">
                <u-image :src="goodsInfo.goodsImg?url + goodsInfo.goodsImg:''" width="90rpx" height="90rpx" border-radius="12rpx"></u-image>
                <text>{{goodsInfo.goodsName}}</text>
            </view>
            <!-- <view class="goHref" @click="sendGoodsHref">發送鏈接</view> -->
        </view>

        <!-- 聊天窗口區域 -->
        <scroll-view :class="{'chatBox':true,'isGoods':Boolean(goodsId)}" :scroll-y="true" id="chatBoxScroll" :style="{height: style.contentViewHeight + 'px'}"
         :scroll-with-animation="true" :scroll-top="scrollTop" @click="noShowMore" refresher-enabled="true"
         :refresher-triggered="triggered" :refresher-threshold="100" refresher-background="#f2f2f2" @refresherrefresh="onRefresh"
         @refresherrestore="onRestore" @refresherpulling="isLoadmore=false">

            <!-- 加載中與沒有更多展示 -->
            <u-loadmore :status="status" margin-top="60" class="loadmore" v-show="isLoadmore" />
            <!-- 再包一層,方便計算高度 -->
            <view id="chatContent" :class="{isPadding:!isSysMessage}">
                <!-- 商品鏈接發送 -->


                <!-- 聊天 -->
                <view class="chat">
                    <!-- 這里再包一層殼子,方便整體循環 -->
                    <!-- 這里過濾,只展示圖片和文字,訂單和商品不展示 -->
                    <view class="chatMessage" v-for="(item,index) in messageList" :key="index" v-if="!isSysMessage">
                        <!-- 如果不是系統消息 -->
                        <view class="timeShow" v-if="((item.content&&(item.content.type=='image' || !item.content.type)))">{{item.createTime}}</view>
                        <!-- 別人發的信息展示 -->
                        <view v-if="item.type=='chat' && item.groupName=='店鋪客服'" class="isText">店鋪客服接待了你</view>
                        <view v-if="item.type=='message'" class="isText">當前無客服在線,您可以留言,我們會盡快回復您</view>
                        <view class="chatY" v-if="item.from && Number(item.from)!==userInfo.userId && ((item.content&&(item.content.type=='image' || !item.content.type)))">
                            <u-image :src="userInfo.shopImg" width="72rpx" height="72rpx" border-radius="50%"></u-image>
                            <view class="messagey" v-if="item.content && !item.content.type">
                                <text selectable="true">{{item.content}}</text>
                            </view>
                            <u-image class="messageyImg" :src="item.content?item.content.content:''" width="300rpx" height="300rpx"
                             border-radius="12rpx" v-if="item.content && item.content.type=='image'" @click="checkImg(item)"></u-image>
                        </view>
                        <!-- 自己發的信息展示 -->
                        <view class="chatM" v-if="(!item.from || (Number(item.from)==userInfo.userId)) && ((item.content&&(item.content.type=='image' || !item.content.type)))">
                            <view class="messagex" v-if="item.content && !item.content.type">
                                <text selectable="true">{{item.content}}</text>
                            </view>
                            <u-image class="messagexImg" :src="item.content?item.content.content:''" width="300rpx" height="300rpx"
                             border-radius="12rpx" v-if="item.content && item.content.type=='image'" @click="checkImg(item)"></u-image>
                            <u-image :src="userInfo.userPhoto" width="72rpx" height="72rpx" border-radius="50%"></u-image>
                        </view>
                    </view>

                    <!-- 如果是系統消息 -->
                    <view class="chatMessage" v-for="(item,index) in messageList" :key="index" v-if="isSysMessage && item.msgContent">
                        <!-- 別人發的信息展示 -->
                        <view class="timeShow">{{item.createTime}}</view>
                        <view class="chatY">
                            <u-image :src="sysImg" width="72rpx" height="72rpx" border-radius="50%"></u-image>
                            <view class="messagey">{{item.msgContent}}</view>
                        </view>
                    </view>
                </view>
            </view>
        </scroll-view>

        <!-- 輸入框 -->
        <view class="inputBox" v-if="!isSysMessage" id="inputBox">
            <u-field v-model="message" label=" " label-width="0" placeholder="在此輸入文字" type="textarea" maxlength="100" :clearable="false"
             clear-size="0" placeholder-style="color: #9A9A9A;font-size:28rpx" @click="isFocus" @blur="isBlur">
            </u-field>
            <u-image :src="jiahao" width="54rpx" height="54rpx" v-show="!message" @click="sendImage"></u-image>
            <view v-show="message" class="sendStn" @click="sendText">發送</view>
        </view>

        <!-- 更多 -->
        <!-- #ifdef H5 -->
        <view class="more" style="background: #f2f2f2; height: 200rpx;" v-if="!isSysMessage && isf">
            <!-- #endif -->
            <!-- #ifndef H5 -->
            <view class="more" style="background: #f2f2f2; height: 200rpx;" v-if="!isSysMessage">
                <!-- #endif -->
                <u-image :src="xiangji" width="90rpx" height="90rpx" @click="goCamera"></u-image>
                <u-image :src="tupian" width="90rpx" height="90rpx" @click="goAlbum"></u-image>
                <!-- <u-image :src="shangpin" width="90rpx" height="90rpx"></u-image> -->
            </view>
            <!-- toast彈窗 -->
            <u-toast ref="uToast" />

            <!-- 圖片預覽 -->
            <previewImage ref="previewImage" :imgs="messgaeImgList" :index="indexImg" @setIndexImg="indexImg=0"></previewImage>

            <!-- 無網絡提示 -->
            <u-no-network></u-no-network>
        </view>
</template>

<script>
    // 用戶圖片預覽
    import config from "common/config.js";
    import previewImage from '@/components/kxj-previewImage/kxj-previewImage.vue';
    // 引入,為解決小程序不支持eval和new Function
    import {
        Function,
        evaluate
    } from 'eval5';
    export default {
        data() {
            return {
                url: config.baseUrl,
                fanhui: '',
                jiahao: '',
                xiangji: '',
                tupian: '',
                shangpin: '',
                sysImg: '',
                background: {
                    backgroundColor: '#f2f2f2'
                },
                message: '',
                messageList: [],
                scrollTop: 0,
                // 聊天時頁面滾動樣式
                style: {
                    pageHeight: 0,
                    contentViewHeight: 0,
                    footViewHeight: 90,
                    mitemHeight: 0
                },
                messgaeImgList: [],
                indexImg: 0,
                // 這個狀態為了判斷更多是否顯示
                morenStyle: false,
                triggered: false,
                _freshing: false,
                isSysMessage: false,

                initHeight: 0,
                // 下拉加載更多相關
                lastPage: 1,
                page: 1,
                status: 'loadmore',
                isLoadmore: false,
                shopId: 0,
                // shopInfo: {},
                goodsId: 0,
                goodsInfo: {},
                userInfo: {},
                // 聊天分頁
                chatPage: 1,
                chatLastPage: 1,

                wssServer: '',
                tokenId: '',
                socketOpen: false,

                // 訂單相關
                orderNo: 0,
                orderId: 0,

                isKill: false,
                killId: 0,
                isf: true,
                windowSize: 0,

                // isNetWork: false
            }
        },
        components: {
            previewImage,
        },
        watch: {
            // 這個通過監聽輸入值的改變,改變聊天區域的高度
            message(val, oldVal) {
                // 小程序中,因為加了setTimeout,所以先觸發了監聽,導致that.initHeight!==res[0].height,最終高度計算錯誤
                // 所以,在這個做判斷,如果清空輸入值時,不觸發監聽
                if (val !== '') {
                    let that = this;
                    let query = uni.createSelectorQuery();
                    query.select('#inputBox').boundingClientRect();
                    setTimeout(() => {
                        query.exec((res) => {
                            if (that.initHeight !== res[0].height) {
                                that.style.contentViewHeight = that.style.contentViewHeight - (res[0].height - that.initHeight);
                                that.initHeight = res[0].height;
                            }
                        })
                    }, 100);
                }
            },
            windowSize(val, oldVal) {
                if (val > oldVal) {
                    this.isf = true;
                }
            },
            isNetWork(val, oldVal) {
                // 防止進入自己店鋪后也會觸發
                if (this.wssServer) {
                    if (val) {
                        this.connectWS();
                    } else {
                        uni.closeSocket();
                    }
                }
            }
        },
        onUnload() {
            uni.setStorageSync('isNet', false);
            uni.closeSocket();

            // 將未讀變為已讀
            if (this.isSysMessage) {
                this.updateMessages()
            } else {
                this.updataSysMessage()
            }
        },
        onLoad(data) {
            uni.setStorageSync('isNet', true);
            // this.isNetWork = uni.getStorageSync('isNetwork');
            if (data.sys) {
                this.isSysMessage = data.sys;
            }
            this.shopId = data.shopId;
            this.goodsId = data.goodsId;
            this.orderId = data.orderId;
            this.orderNo = data.orderNo;
            this.isKill = data.isKill;
            this.killId = data.id;

            this.tokenId = uni.getStorageSync('tokenId');

            // 獲取用戶信息
            this.queryUserDetails()


            // 進來先請求一次數據(歷史記錄)
            this.$nextTick(() => {
                this.onRefresh();
                this.scrollToBottom();
            });


            //監聽服務器打開
            uni.onSocketOpen(() => {
                this.socketOpen = true;
                let sendData = {
                    type: 'login',
                    uid: this.userInfo.userId,
                    userName: this.userInfo.loginName,
                    role: 'user',
                    platform: 2,
                    shopId: this.userInfo.shopId,
                    group: this.userInfo.receiveId
                }
                uni.sendSocketMessage({
                    data: JSON.stringify(sendData),
                })
                // 獲取帶出商品信息
                if (this.goodsId) {
                    this.getGoodsInfo(this.goodsId);

                }
                // 如果是訂單,則發送訂單
                if (this.orderId) {
                    let ordersData = {
                        content: `{"type":"orders","orderId":"${this.orderId}","content": "訂單號:${this.orderNo}"}`,
                        role: 'user',
                        type: 'say',
                        to: this.userInfo.receiveId
                    }
                    uni.sendSocketMessage({
                        data: JSON.stringify(ordersData)
                    })
                }
                // 接受服務器傳來的數據
                uni.onSocketMessage((res) => {
                    let data = JSON.parse(res.data);
                    // 如果是圖片相關,則轉換為json對象,重構數據
                    try {
                        // 這里不處理數字,否則回變為科學計數法
                        if (isNaN(data.content)) {
                            // let text = eval("(" + data.content + ")");
                            let text = (new Function("return" + data.content))();
                            data.content = text;
                        }
                        if (data.content && data.content.type == 'image') {
                            data.content.content = this.url + data.content.content;
                        }
                        if ((data.type == 'say' && data.role == 'user') || (data.type == 'chat' && data.groupName) || (data.type ==
                                'message')) {
                            this.messageList.push(data)

                            // 圖片預覽相關
                            this.messgaeImgList = [];
                            this.messageList.forEach((item, index) => {
                                if (item.content && item.content.type == 'image') {
                                    this.messgaeImgList.push(item.content.content);
                                }
                            })
                        };
                        // 如果是文字相關,則直接push
                    } catch (e) {
                        if ((data.type == 'say' && data.role == 'user') || (data.type == 'chat' && data.groupName) || (data.type ==
                                'message')) {
                            this.messageList.push(data)
                        }
                    }
                    this.$nextTick(function() {
                        this.scrollToBottom();
                    })
                });
            });

            //監聽服務器打開失敗重連
            uni.onSocketError((res) => {
                console.log('WebSocket連接打開失敗,請檢查!');
            });

            //監聽服務器關閉
            uni.onSocketClose((res) => {
                console.log("webSocket on 關閉連接");
            });
            
            // 獲取底部輸入框高度
            this.$nextTick(() => {
                let that = this;
                let query = uni.createSelectorQuery();
                query.select('#inputBox').boundingClientRect();
                query.exec((res) => {
                    that.initHeight = res[0].height;
                })
            })
            // 聊天區域高度獲取
            this.getchatHeight()

            if (this.goodsId) {
                this.style.contentViewHeight = this.style.contentViewHeight - 70
            }

            // 將未讀變為已讀
            if (this.isSysMessage) {
                this.updateMessages()
            } else {
                this.updataSysMessage()
            }

            uni.onWindowResize((res) => {
                this.windowSize = res.size.windowHeight;
                // alert('變化后的窗口高度=' + res.size.windowHeight)
            })
        },
        methods: {
            //創建wekSocket長連接
            connectWS() {
                // 創建Socket
                let that = this;
                this.$u.vuex('SocketTask', uni.connectSocket({
                    url: that.wssServer,
                    header: {
                        'content-type': 'application/json'
                    },
                    method: 'post',
                    success: function(res) {
                        console.log('WebSocket連接創建success', res);
                    },
                    fail: function(err) {
                        setTimeout(() => {
                            that.connectWS();
                        }, 500);
                        console.log('WebSocket連接創建fail', err)
                    },
                }));
            },


            goBack() {
                this.$u.route({
                    type: 'navigateBack'
                })
            },

            isJson(obj) {
                let isjson = typeof(obj) == "object" && Object.prototype.toString.call(obj).toLowerCase() == "[object object]" && !
                    obj.length;
                return isjson;
            },
            // 聊天區域高度獲取
            getchatHeight() {
                // 獲取頁面高度相關
                const res = uni.getSystemInfoSync(); //獲取手機可使用窗口高度
                this.style.pageHeight = res.windowHeight - res.statusBarHeight;
                if (this.isSysMessage) {
                    // 如果是系統消息
                    // #ifndef MP-WEIXIN
                    this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) + 10 //像素   因為給出的是像素高度 然后我們用的是rpx  所以換算一下 
                    // #endif

                    // #ifdef MP-WEIXIN
                    this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100);
                    // #endif
                } else {
                    // 否則
                    // #ifndef MP-WEIXIN
                    this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) - 40; //像素   因為給出的是像素高度 然后我們用的是rpx  所以換算一下 
                    // #endif

                    // #ifdef MP-WEIXIN
                    this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) - 40;
                    // #endif
                }
            },
            // 下拉相關
            onRefresh(pagesize, page) {
                // 下拉被觸發
                // 這里下拉刷新邏輯有點坑,參照:https://ask.dcloud.net.cn/article/37181
                this.isLoadmore = false;
                if (this._freshing) return;
                this._freshing = true;
                if (!this.triggered) { //界面下拉觸發,triggered可能不是true,要設為true
                    this.triggered = true;
                }

                if (this.isSysMessage) {
                    // 請求系統消息
                    this.queryMessages(4, this.page);
                } else {
                    // 這里加載更多聊天歷史記錄
                    this.getHistory(this.shopId, this.chatPage);
                }
            },
            onRestore() {
                // 下拉刷新被復位
                console.log("onRestore");
            },
            goShopDetail() {
                this.$u.route({
                    url: "pagesShopping/shop/index",
                    params: {
                        shopId: this.shopId,
                        isAgain: true
                    },
                    type: 'redirect'
                })
            },
            // 發送圖片
            sendImage() {
                this.isf = true;
                // #ifndef APP-PLUS
                if (!this.morenStyle) {
                    this.morenStyle = true;
                    /* 
                     這里是為了給下面的更多區域一個從下面滑上來的效果,
                     主要是通過改變回話區域的高度,來控制下方是否展示,下方隱藏展示同理
                     */
                    let i = 0;
                    const setTime = setInterval(() => {
                        i = i + 10;
                        this.style.contentViewHeight = this.style.contentViewHeight - 10;
                        if (i == 100) {
                            clearInterval(setTime)
                            this.scrollTop = this.scrollTop + 100;
                        }
                    }, 2)
                }
                // #endif

                // #ifdef APP-PLUS
                if (!this.morenStyle) {
                    this.morenStyle = true;
                    this.style.contentViewHeight = this.style.contentViewHeight - 100;
                    this.scrollTop = this.scrollTop + 100;
                }
                // #endif
            },
            // 點擊回話框其他區域,隱藏下方更多展示
            noShowMore() {
                // #ifndef APP-PLUS
                if (this.morenStyle) {
                    this.morenStyle = false;
                    let n = 0;
                    const setTime1 = setInterval(() => {
                        n = n + 10;
                        this.style.contentViewHeight = this.style.contentViewHeight + 10;
                        if (n == 100) {
                            clearInterval(setTime1)
                            this.scrollTop = this.scrollTop - 100;
                        }
                    }, 5)
                }
                // #endif

                // #ifdef APP-PLUS
                if (this.morenStyle) {
                    this.morenStyle = false;
                    this.style.contentViewHeight = this.style.contentViewHeight + 100;
                    this.scrollTop = this.scrollTop - 100;
                }
                // #endif
            },
            // 獲取焦點
            isFocus() {
                this.isf = false;
            },
            // 失去焦點
            isBlur() {},
            // 預覽圖片
            checkImg(item) {
                if (item.content && item.content.type == 'image') {
                    this.indexImg = this.messgaeImgList.indexOf(item.content.content);
                    this.$refs.previewImage.show = true;
                }
            },
            // 發送文字按鈕
            sendText() {
                if (this.message) {
                    let message = {
                        content: this.message,
                        role: 'user',
                        type: 'say',
                        to: this.userInfo.receiveId
                    };
                    //發送會話內容
                    if (this.socketOpen) {
                        uni.sendSocketMessage({
                            data: JSON.stringify(message),
                            success: () => {
                                this.$nextTick(function() {
                                    // 發送完之后清空輸入框
                                    this.message = '';
                                    // 重新獲取底部高度
                                    let that = this;
                                    let query = uni.createSelectorQuery();
                                    query.select('#inputBox').boundingClientRect();
                                    setTimeout(() => {
                                        query.exec((res) => {
                                            that.initHeight = res[0].height;
                                            // 重新獲取內容區高度
                                            this.getchatHeight();
                                            if (this.goodsId) {
                                                this.style.contentViewHeight = this.style.contentViewHeight - 70
                                            }
                                        })
                                    }, 150)

                                    this.morenStyle = false;
                                })
                            },
                            fail: (err) => {
                                console.log(err)
                            }
                        });
                    } else {
                        this.$refs.uToast.show({
                            title: '服務未開啟,暫不能發送消息'
                        })
                    }

                }
            },
            // 調起相機方法
            goCamera() {
                uni.chooseImage({
                    count: 1, //默認9
                    sizeType: ['compressed'],
                    sourceType: ['camera'],
                    success: (res) => {
                        const tempFilePaths = res.tempFilePaths;
                        uni.uploadFile({
                            url: this.url + '/weapp/user/uploadPicture',
                            filePath: tempFilePaths[0],
                            name: 'img',
                            formData: {
                                dir: 'users',
                                isTumb: 1,
                                isLocation: 1,
                                id: 'WU_FILE_0',
                            },
                            header: {
                                tokenid: this.tokenId
                            },
                            success: (ress) => {
                                let data = JSON.parse(ress.data);
                                let message = {
                                    content: `{"content":"${data.data}","type": "image"}`,
                                    role: 'user',
                                    type: 'say',
                                    to: this.userInfo.receiveId
                                };
                                uni.sendSocketMessage({
                                    data: JSON.stringify(message)
                                })
                            }
                        });
                    }
                });
            },
            // 選擇圖片方法
            goAlbum() {
                uni.chooseImage({
                    count: 1, //默認9
                    sizeType: ['original', 'compressed'],
                    sourceType: ['album'],
                    success: (res) => {
                        const tempFilePaths = res.tempFilePaths;
                        uni.uploadFile({
                            url: this.url + '/weapp/user/uploadPicture',
                            filePath: tempFilePaths[0],
                            name: 'img',
                            formData: {
                                dir: 'users',
                                isTumb: 1,
                                isLocation: 1,
                                id: 'WU_FILE_0',
                            },
                            header: {
                                tokenid: this.tokenId
                            },
                            success: (ress) => {
                                let data = JSON.parse(ress.data);
                                let message = {
                                    content: `{"content":"${data.data}","type": "image"}`,
                                    role: 'user',
                                    type: 'say',
                                    to: this.userInfo.receiveId
                                };
                                uni.sendSocketMessage({
                                    data: JSON.stringify(message)
                                })
                            }
                        });
                    }
                });
            },
            // 下滑到底部的方法 
            scrollToBottom() {
                let that = this;
                let query = uni.createSelectorQuery();
                query.select('#chatContent').boundingClientRect();
                query.select('#chatBoxScroll').boundingClientRect();
                query.exec((res) => {
                    that.style.mitemHeight = 0;
                    that.style.mitemHeight = res[0].height;
                    // res[0].forEach((rect) => that.style.mitemHeight = that.style.mitemHeight + rect.height + 40) //獲取所有內部子元素的高度
                    if (that.style.mitemHeight > (that.style.contentViewHeight)) { //判斷子元素高度是否大於顯示高度
                        that.scrollTop = that.style.mitemHeight - that.style.contentViewHeight + 100 //用子元素的高度減去顯示的高度就獲益獲得序言滾動的高度
                    }
                })
            },

            // 點擊進入商品詳情
            goGoodsDetail() {
                if (!this.orderNo && !this.isKill) {
                    this.$u.route({
                        url: 'pagesShopping/commodity/index',
                        params: {
                            goodsId: this.goodsId,
                            isAgain: true
                        },
                        type: 'redirect'
                    })
                } else if (this.isKill) {
                    this.$u.route({
                        url: 'pagesShopping/seckill/seckillDetail',
                        params: {
                            id: this.killId,
                            goodsId: this.goodsId,
                            isAgain: true
                        },
                        type: 'redirect'
                    })
                } else {
                    this.$u.route({
                        url: 'pageOrders/orderDetail/index',
                        params: {
                            orderId: this.orderId,
                            shopId: this.shopId,
                            isAgain: true
                        },
                        type: 'redirect'
                    })
                }
            },

            // 網絡請求相關
            queryMessages(pagesize, page) {
                // 獲取系統消息
                this.$u.post('/messages/queryMessages', {
                    pagesize,
                    page
                }).then(res => {
                    if (res.status == 200) {
                        this.lastPage = res.data.last_page;
                        if (this.page <= this.lastPage) {
                            this.lastPage == 1 ? this.status = 'nomore' : this.status = 'loading';
                            // 這里調轉一下數組之后再保存
                            let newList = [];
                            newList.push(...res.data.data);
                            newList.reverse();
                            this.messageList.unshift(...newList);
                            this.page++
                            this._freshing = false;
                            this.triggered = false
                        } else {
                            this.status = 'nomore';
                            this.isLoadmore = true;
                            this._freshing = false;
                            this.triggered = false
                        }
                    } else {
                        this._freshing = false;
                        this.triggered = false;
                    }
                })
            },

            // 獲取歷史聊天記錄
            getHistory(receiveId, page) {
                this.$u.post('/addon/wstim-chats-pagequery', {
                    receiveId,
                    page,
                }).then(res => {
                    if (res.status == 1 || res.status == 200) {
                        this.chatLastPage = res.data.last_page;
                        if (this.chatPage <= this.chatLastPage) {
                            this.chatLastPage == 1 ? this.status = 'nomore' : this.status = 'loading';
                            this.messageList.unshift(...res.data.data);
                            this.chatPage++
                            this._freshing = false;
                            this.triggered = false;

                            // 圖片預覽相關
                            res.data.data.forEach((item, index) => {
                                if (item.content && item.content.type == 'image') {
                                    this.messgaeImgList.push(item.content.content);
                                }
                            })
                        } else {
                            this.status = 'nomore';
                            this.isLoadmore = true;
                            this._freshing = false;
                            this.triggered = false
                        }
                    } else {
                        this._freshing = false;
                        this.triggered = false;
                    }
                })
            },
            updateMessages() {
                // 修改系統消息狀態(全部變為已讀)
                this.$u.post('/messages/updateMessages', {}).then(res => {
                    if (res.status == 200) {
                        return;
                    }
                })
            },

            updataSysMessage() {
                // 修改回話消息狀態(全部變為已讀)
                this.$u.post('/addon/wstim-chats-setRead', {
                    shopId: this.shopId
                }).then(res => {
                    if (res.status == 1 || res.status == 200) {
                        return;
                    }
                })
            },

            // 帶出商品信息獲取
            getGoodsInfo(goodsId) {
                this.$u.post('/goods_details/getProductDetails', {
                    goodsId
                }).then(res => {
                    if (res.status == 200) {
                        this.goodsInfo = res.data;
                        this.sendGoodsHref()
                    }
                })
            },

            // 發送商品信息
            sendGoodsHref() {
                let goodsData = {
                    content: `{"type":"goods","goodsId":"${this.goodsId}","content": "${this.goodsInfo.goodsName}"}`,
                    role: 'user',
                    type: 'say',
                    to: this.userInfo.receiveId
                }
                uni.sendSocketMessage({
                    data: JSON.stringify(goodsData),
                })
            },

            // 獲取基礎基礎數據(用戶,店鋪,聯系信息)
            queryUserDetails() {
                this.$u.get('/addon/wstim-chats-getBaseData', {
                    shopId: this.shopId,
                }).then(res => {
                    if (res.status == 1 || res.status == 200) {
                        this.userInfo = res.data;
                        this.wssServer = res.data.server;
                        //連接websoket
                        if (this.tokenId) {
                            this.connectWS();
                        }
                    }
                })
            }
        }
    }
</script>

<style lang="scss">
    .service {
        background: $bg_color;
        height: 100vh;
        width: 100vw;
        overflow: hidden;

        // 頭部
        .header {
            /deep/ .u-slot-content {
                display: flex;
                justify-content: space-between;
                padding-right: 24rpx;
            }

            .serviceTitle {
                display: flex;

                /* #ifdef MP-WEIXIN */
                .u-image {
                    margin-left: 24rpx;
                }

                /* #endif */
                &>view {
                    font-size: 32rpx;
                    font-family: PingFang SC;
                    font-weight: bold;
                    color: #343434;
                    margin-left: 24rpx;
                }
            }

            .goShopBtn {
                font-size: 28rpx;
                font-family: PingFang SC;
                font-weight: bold;
                color: $white_color;
                width: 87rpx;
                height: 48rpx;
                background: #CA3232;
                border-radius: 12rpx;
                text-align: center;
                line-height: 48rpx;
            }
        }

        // 分割線
        .hrline {
            height: 2rpx;
            width: 100vh;
            background: #E5E5E5;
            // margin-bottom: 69rpx;
        }

        .goodHref {
            position: fixed;
            left: 0;
            right: 0;
            z-index: 10;
            top: 120rpx;
            /* #ifdef MP-WEIXIN */
            top: 150rpx;
            /* #endif */

            /* #ifdef APP-PLUS */
            top: 180rpx;
            /* #endif */


            margin: 0 36rpx;
            // margin-top: 69rpx;
            background: $white_color;
            box-shadow: 1rpx 6rpx 24rpx 0rpx rgba(186, 186, 186, 0.11);
            border-radius: 30rpx;

            .goodContent {
                display: flex;
                padding: 36rpx;

                text {
                    margin-left: 24rpx;
                    flex: 1;
                    font-size: 28rpx;
                    font-family: PingFang SC;
                    color: #343434;
                    line-height: 42rpx;
                }
            }

            .goHref {
                text-align: center;
                font-size: 28rpx;
                font-family: PingFang SC;
                font-weight: bold;
                color: #343434;
                padding: 30rpx 0;
                border-top: 2px solid #F5F5F5;
            }
        }

        .isGoods {
            margin-top: 140rpx;
        }

        .loadmore {
            padding-bottom: 0rpx;
        }

        .chatBox {
            height: calc(100vh - 170rpx);

            #chatContent {
                .chat {
                    margin-top: 72rpx;
                    padding: 0 36rpx;

                    .chatMessage {
                        @mixin textShow {
                            margin: 0 auto;
                            background: #E3E3E3;
                            border-radius: 8rpx;
                            font-size: 24rpx;
                            font-family: PingFang SC;
                            font-weight: bold;
                            color: #939393;
                            width: 300rpx;
                            height: 42rpx;
                            line-height: 42rpx;
                            text-align: center;
                            margin-bottom: 52rpx;
                        }

                        .timeShow {
                            @include textShow;
                        }

                        .isText {
                            @include textShow;
                            background: transparent;
                        }

                        .chatY {
                            display: flex;
                            position: relative;
                            margin-bottom: 68rpx;

                            &>.messagey {
                                // flex: 1;
                                margin-left: 24rpx;
                                padding: 23rpx 26rpx;
                                background: $white_color;
                                max-width: 558rpx;
                                border-radius: 12rpx;
                                word-break: break-all;

                                &::before {
                                    content: '';
                                    display: inline-block;
                                    border: 12rpx solid;
                                    border-color: transparent $white_color transparent transparent;
                                    position: absolute;
                                    left: 75rpx;
                                    top: 30rpx;
                                }
                            }

                            &>.messageyImg {

                                /* #ifndef MP-WEIXIN */
                                &::before {
                                    content: '';
                                    display: inline-block;
                                    border: 12rpx solid;
                                    border-color: transparent $white_color transparent transparent;
                                    position: absolute;
                                    left: 4rpx;
                                    top: 30rpx;
                                }

                                /deep/ .u-image__image {
                                    padding: 20rpx;
                                    background: $white_color;
                                    margin-left: 24rpx;
                                }

                                /* #endif */
                                .u-image {
                                    &::before {
                                        content: '';
                                        display: inline-block;
                                        border: 12rpx solid;
                                        border-color: transparent $white_color transparent transparent;
                                        position: absolute;
                                        left: 4rpx;
                                        top: 30rpx;
                                    }
                                }

                                /* #ifdef MP-WEIXIN */

                                &::before {
                                    content: '';
                                    display: inline-block;
                                    border: 12rpx solid;
                                    border-color: transparent $white_color transparent transparent;
                                    position: absolute;
                                    left: 0rpx;
                                    top: 30rpx;
                                }

                                /deep/ .u-image__image {
                                    padding: 20rpx;
                                    background: $white_color;
                                    margin-left: 24rpx;
                                }

                                /* #endif */
                            }
                        }

                        .chatM {
                            display: flex;
                            position: relative;
                            justify-content: flex-end;
                            margin-bottom: 68rpx;

                            &>.messagex {
                                margin-right: 24rpx;
                                padding: 23rpx 26rpx;
                                max-width: 558rpx;
                                border-radius: 12rpx;
                                background: #CA3232;
                                color: $white_color;
                                word-break: break-all;

                                &::after {
                                    content: '';
                                    display: inline-block;
                                    border: 12rpx solid;
                                    border-color: transparent transparent transparent #CA3232;
                                    position: absolute;
                                    right: 75rpx;
                                    top: 30rpx;
                                }
                            }

                            &>.messagexImg {
                                display: flex;

                                // 微信小程序樣式
                                /* #ifdef MP-WEIXIN */
                                .u-image {
                                    margin-right: 64rpx;

                                    /deep/ .u-image__image {
                                        padding: 20rpx;
                                        background: #CA3232;
                                    }

                                    &::after {
                                        content: '';
                                        display: inline-block;
                                        border: 12rpx solid;
                                        border-color: transparent transparent transparent #CA3232;
                                        position: absolute;
                                        right: -62rpx;
                                        top: 30rpx;
                                    }
                                }

                                /* #endif */
                                /* #ifndef MP-WEIXIN */
                                /deep/ .u-image__image {
                                    padding: 20rpx;
                                    background: #CA3232;
                                    margin-left: -65rpx;
                                }

                                &::after {
                                    content: '';
                                    display: inline-block;
                                    border: 12rpx solid;
                                    border-color: transparent transparent transparent #CA3232;
                                    position: absolute;
                                    right: 4rpx;
                                    top: 30rpx;
                                }

                                /* #endif */
                            }
                        }
                    }
                }
            }

            .isPadding {
                padding-top: 69rpx;
            }
        }

        .inputBox {
            background: #FFFFFF;
            // position: fixed;
            // bottom: 0;
            // right: 0;
            // left: 0;
            display: flex;
            align-items: center;
            position: relative;

            /deep/ .u-field {
                padding: 0;
                width: 650rpx;
                margin-left: 24rpx;

                .u-flex {
                    margin: 0 !important;

                    .u-textarea-class {
                        // display: flex;
                        // align-items: center;
                        min-height: 30rpx !important;
                        max-height: 150rpx;
                        padding: 30rpx;

                        .uni-textarea-textarea {
                            overflow-y: auto !important;
                        }
                    }
                }
            }

            &>.u-image {
                position: absolute;
                right: 24rpx;
            }

            &>.sendStn {
                position: absolute;
                right: 28rpx;
                background: #CA3232;
                color: $white_color;
                font-weight: bold;
                padding: 10rpx 18rpx;
                border-radius: 12rpx;
                top: 50%;
                transform: translateY(-50%);
            }
        }

        .more {
            display: flex;

            .u-image {
                margin: 0 24rpx;
                margin-top: 36rpx;
            }
        }
    }
</style>


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM