main.vue 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. <!--
  2. - Copyright (C) 2018-2019
  3. - All rights reserved, Designed By www.joolun.com
  4. 非繁源码:
  5. ① 移除暂时用不到的 websocket
  6. ② 代码优化,补充注释,提升阅读性
  7. -->
  8. <template>
  9. <ContentWrap>
  10. <div class="msg-div" ref="msgDivRef">
  11. <!-- 加载更多 -->
  12. <div v-loading="loading"></div>
  13. <div v-if="!loading">
  14. <div class="el-table__empty-block" v-if="hasMore" @click="loadMore"
  15. ><span class="el-table__empty-text">点击加载更多</span></div
  16. >
  17. <div class="el-table__empty-block" v-if="!hasMore"
  18. ><span class="el-table__empty-text">没有更多了</span></div
  19. >
  20. </div>
  21. <!-- 消息列表 -->
  22. <MsgList :list="list" :account-id="accountId" :user="user" />
  23. </div>
  24. <div class="msg-send" v-loading="sendLoading">
  25. <WxReplySelect ref="replySelectRef" v-model="reply" />
  26. <el-button type="success" class="send-but" @click="sendMsg">发送(S)</el-button>
  27. </div>
  28. </ContentWrap>
  29. </template>
  30. <script lang="ts" setup>
  31. import WxReplySelect, { Reply, ReplyType } from '@/views/mp/components/wx-reply'
  32. import MsgList from './components/MsgList.vue'
  33. import { getMessagePage, sendMessage } from '@/api/mp/message'
  34. import { getUser } from '@/api/mp/user'
  35. import profile from '@/assets/imgs/profile.jpg'
  36. import { User } from './types'
  37. defineOptions({ name: 'WxMsg' })
  38. const message = useMessage() // 消息弹窗
  39. const props = defineProps({
  40. userId: {
  41. type: Number,
  42. required: true
  43. }
  44. })
  45. const accountId = ref(-1) // 公众号ID,需要通过userId初始化
  46. const loading = ref(false) // 消息列表是否正在加载中
  47. const hasMore = ref(true) // 是否可以加载更多
  48. const list = ref<any[]>([]) // 消息列表
  49. const queryParams = reactive({
  50. pageNo: 1, // 当前页数
  51. pageSize: 14, // 每页显示多少条
  52. accountId: accountId
  53. })
  54. // 由于微信不再提供昵称,直接使用“用户”展示
  55. const user: User = reactive({
  56. nickname: '用户',
  57. avatar: profile,
  58. accountId: accountId // 公众号账号编号
  59. })
  60. // ========= 消息发送 =========
  61. const sendLoading = ref(false) // 发送消息是否加载中
  62. // 微信发送消息
  63. const reply = ref<Reply>({
  64. type: ReplyType.Text,
  65. accountId: -1,
  66. articles: []
  67. })
  68. const replySelectRef = ref<InstanceType<typeof WxReplySelect> | null>(null) // WxReplySelect组件ref,用于消息发送成功后清除内容
  69. const msgDivRef = ref<HTMLDivElement | null>(null) // 消息显示窗口ref,用于滚动到底部
  70. /** 完成加载 */
  71. onMounted(async () => {
  72. const data = await getUser(props.userId)
  73. user.nickname = data.nickname?.length > 0 ? data.nickname : user.nickname
  74. user.avatar = user.avatar?.length > 0 ? data.avatar : user.avatar
  75. accountId.value = data.accountId
  76. reply.value.accountId = data.accountId
  77. refreshChange()
  78. })
  79. // 执行发送
  80. const sendMsg = async () => {
  81. if (!unref(reply)) {
  82. return
  83. }
  84. // 公众号限制:客服消息,公众号只允许发送一条
  85. if (
  86. reply.value.type === ReplyType.News &&
  87. reply.value.articles &&
  88. reply.value.articles.length > 1
  89. ) {
  90. reply.value.articles = [reply.value.articles[0]]
  91. message.success('图文消息条数限制在 1 条以内,已默认发送第一条')
  92. }
  93. const data = await sendMessage({ userId: props.userId, ...reply.value })
  94. sendLoading.value = false
  95. list.value = [...list.value, ...[data]]
  96. await scrollToBottom()
  97. // 发送后清空数据
  98. replySelectRef.value?.clear()
  99. }
  100. const loadMore = () => {
  101. queryParams.pageNo++
  102. getPage(queryParams, null)
  103. }
  104. const getPage = async (page: any, params: any = null) => {
  105. loading.value = true
  106. let dataTemp = await getMessagePage(
  107. Object.assign(
  108. {
  109. pageNo: page.pageNo,
  110. pageSize: page.pageSize,
  111. userId: props.userId,
  112. accountId: page.accountId
  113. },
  114. params
  115. )
  116. )
  117. const scrollHeight = msgDivRef.value?.scrollHeight ?? 0
  118. // 处理数据
  119. const data = dataTemp.list.reverse()
  120. list.value = [...data, ...list.value]
  121. loading.value = false
  122. if (data.length < queryParams.pageSize || data.length === 0) {
  123. hasMore.value = false
  124. }
  125. queryParams.pageNo = page.pageNo
  126. queryParams.pageSize = page.pageSize
  127. // 滚动到原来的位置
  128. if (queryParams.pageNo === 1) {
  129. // 定位到消息底部
  130. await scrollToBottom()
  131. } else if (data.length !== 0) {
  132. // 定位滚动条
  133. await nextTick()
  134. if (scrollHeight !== 0) {
  135. if (msgDivRef.value) {
  136. msgDivRef.value.scrollTop = msgDivRef.value.scrollHeight - scrollHeight - 100
  137. }
  138. }
  139. }
  140. }
  141. const refreshChange = () => {
  142. getPage(queryParams)
  143. }
  144. /** 定位到消息底部 */
  145. const scrollToBottom = async () => {
  146. await nextTick()
  147. if (msgDivRef.value) {
  148. msgDivRef.value.scrollTop = msgDivRef.value.scrollHeight
  149. }
  150. }
  151. </script>
  152. <style lang="scss" scoped>
  153. .msg-div {
  154. height: 50vh;
  155. margin-right: 10px;
  156. margin-left: 10px;
  157. overflow: auto;
  158. background-color: #eaeaea;
  159. }
  160. .msg-send {
  161. padding: 10px;
  162. }
  163. .send-but {
  164. float: right;
  165. margin-top: 8px;
  166. margin-bottom: 8px;
  167. }
  168. </style>