main.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  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" :id="'msg-div' + nowStr">
  11. <!-- 加载更多 -->
  12. <div v-loading="loading"></div>
  13. <div v-if="!loading">
  14. <div class="el-table__empty-block" v-if="loadMore" @click="loadingMore"
  15. ><span class="el-table__empty-text">点击加载更多</span></div
  16. >
  17. <div class="el-table__empty-block" v-if="!loadMore"
  18. ><span class="el-table__empty-text">没有更多了</span></div
  19. >
  20. </div>
  21. <!-- 消息列表 -->
  22. <div class="execution" v-for="item in list" :key="item.id">
  23. <div class="avue-comment" :class="item.sendFrom === 2 ? 'avue-comment--reverse' : ''">
  24. <div class="avatar-div">
  25. <img
  26. :src="item.sendFrom === 1 ? user.avatar : mp.avatar"
  27. class="avue-comment__avatar"
  28. />
  29. <div class="avue-comment__author"
  30. >{{ item.sendFrom === 1 ? user.nickname : mp.nickname }}
  31. </div>
  32. </div>
  33. <div class="avue-comment__main">
  34. <div class="avue-comment__header">
  35. <div class="avue-comment__create_time">{{ formatDate(item.createTime) }}</div>
  36. </div>
  37. <div
  38. class="avue-comment__body"
  39. :style="item.sendFrom === 2 ? 'background: #6BED72;' : ''"
  40. >
  41. <!-- 【事件】区域 -->
  42. <div v-if="item.type === MsgType.Event && item.event === 'subscribe'">
  43. <el-tag type="success">关注</el-tag>
  44. </div>
  45. <div v-else-if="item.type === MsgType.Event && item.event === 'unsubscribe'">
  46. <el-tag type="danger">取消关注</el-tag>
  47. </div>
  48. <div v-else-if="item.type === MsgType.Event && item.event === 'CLICK'">
  49. <el-tag>点击菜单</el-tag>
  50. 【{{ item.eventKey }}】
  51. </div>
  52. <div v-else-if="item.type === MsgType.Event && item.event === 'VIEW'">
  53. <el-tag>点击菜单链接</el-tag>
  54. 【{{ item.eventKey }}】
  55. </div>
  56. <div v-else-if="item.type === MsgType.Event && item.event === 'scancode_waitmsg'">
  57. <el-tag>扫码结果</el-tag>
  58. 【{{ item.eventKey }}】
  59. </div>
  60. <div v-else-if="item.type === MsgType.Event && item.event === 'scancode_push'">
  61. <el-tag>扫码结果</el-tag>
  62. 【{{ item.eventKey }}】
  63. </div>
  64. <div v-else-if="item.type === MsgType.Event && item.event === 'pic_sysphoto'">
  65. <el-tag>系统拍照发图</el-tag>
  66. </div>
  67. <div v-else-if="item.type === MsgType.Event && item.event === 'pic_photo_or_album'">
  68. <el-tag>拍照或者相册</el-tag>
  69. </div>
  70. <div v-else-if="item.type === MsgType.Event && item.event === 'pic_weixin'">
  71. <el-tag>微信相册</el-tag>
  72. </div>
  73. <div v-else-if="item.type === MsgType.Event && item.event === 'location_select'">
  74. <el-tag>选择地理位置</el-tag>
  75. </div>
  76. <div v-else-if="item.type === MsgType.Event">
  77. <el-tag type="danger">未知事件类型</el-tag>
  78. </div>
  79. <!-- 【消息】区域 -->
  80. <div v-else-if="item.type === MsgType.Text">{{ item.content }}</div>
  81. <div v-else-if="item.type === MsgType.Voice">
  82. <WxVoicePlayer :url="item.mediaUrl" :content="item.recognition" />
  83. </div>
  84. <div v-else-if="item.type === MsgType.Image">
  85. <a target="_blank" :href="item.mediaUrl">
  86. <img :src="item.mediaUrl" style="width: 100px" />
  87. </a>
  88. </div>
  89. <div
  90. v-else-if="item.type === MsgType.Video || item.type === 'shortvideo'"
  91. style="text-align: center"
  92. >
  93. <WxVideoPlayer :url="item.mediaUrl" />
  94. </div>
  95. <div v-else-if="item.type === MsgType.Link" class="avue-card__detail">
  96. <el-link type="success" :underline="false" target="_blank" :href="item.url">
  97. <div class="avue-card__title"><i class="el-icon-link"></i>{{ item.title }}</div>
  98. </el-link>
  99. <div class="avue-card__info" style="height: unset">{{ item.description }}</div>
  100. </div>
  101. <!-- TODO 芋艿:待完善 -->
  102. <div v-else-if="item.type === MsgType.Location">
  103. <WxLocation
  104. :label="item.label"
  105. :location-y="item.locationY"
  106. :location-x="item.locationX"
  107. />
  108. </div>
  109. <div v-else-if="item.type === MsgType.News" style="width: 300px">
  110. <!-- TODO 芋艿:待测试;详情页也存在类似的情况 -->
  111. <WxNews :articles="item.articles" />
  112. </div>
  113. <div v-else-if="item.type === MsgType.Music">
  114. <WxMusic
  115. :title="item.title"
  116. :description="item.description"
  117. :thumb-media-url="item.thumbMediaUrl"
  118. :music-url="item.musicUrl"
  119. :hq-music-url="item.hqMusicUrl"
  120. />
  121. </div>
  122. </div>
  123. </div>
  124. </div>
  125. </div>
  126. </div>
  127. <div class="msg-send" v-loading="sendLoading">
  128. <WxReplySelect ref="replySelectRef" :objData="objData" />
  129. <el-button type="success" size="small" class="send-but" @click="sendMsg">发送(S)</el-button>
  130. </div>
  131. </ContentWrap>
  132. </template>
  133. <script setup lang="ts" name="WxMsg">
  134. import WxReplySelect from '@/views/mp/components/wx-reply/main.vue'
  135. import WxVideoPlayer from '@/views/mp/components/wx-video-play/main.vue'
  136. import WxVoicePlayer from '@/views/mp/components/wx-voice-play/main.vue'
  137. import WxNews from '@/views/mp/components/wx-news/main.vue'
  138. import WxLocation from '@/views/mp/components/wx-location/main.vue'
  139. import WxMusic from '@/views/mp/components/wx-music/main.vue'
  140. import { getMessagePage, sendMessage } from '@/api/mp/message'
  141. import { getUser } from '@/api/mp/user'
  142. import { formatDate } from '@/utils/formatTime'
  143. import profile from '@/assets/imgs/profile.jpg'
  144. import wechat from '@/assets/imgs/wechat.png'
  145. import { MsgType } from './types'
  146. const message = useMessage() // 消息弹窗
  147. const props = defineProps({
  148. userId: {
  149. type: Number,
  150. required: true
  151. }
  152. })
  153. const nowStr = ref(new Date().getTime()) // 当前的时间戳,用于每次消息加载后,回到原位置;具体见 :id="'msg-div' + nowStr" 处
  154. const loading = ref(false) // 消息列表是否正在加载中
  155. const loadMore = ref(true) // 是否可以加载更多
  156. const list = ref<any[]>([]) // 消息列表
  157. const queryParams = reactive({
  158. pageNo: 1, // 当前页数
  159. pageSize: 14, // 每页显示多少条
  160. accountId: undefined
  161. })
  162. interface User {
  163. nickname: string
  164. avatar: string
  165. accountId: number
  166. }
  167. // 由于微信不再提供昵称,直接使用“用户”展示
  168. const user: User = reactive({
  169. nickname: '用户',
  170. avatar: profile,
  171. accountId: 0 // 公众号账号编号
  172. })
  173. interface Mp {
  174. nickname: string
  175. avatar: string
  176. }
  177. const mp: Mp = reactive({
  178. nickname: '公众号',
  179. avatar: wechat
  180. })
  181. // ========= 消息发送 =========
  182. const sendLoading = ref(false) // 发送消息是否加载中
  183. interface ObjData {
  184. type: MsgType
  185. accountId: number | null
  186. articles: any[]
  187. }
  188. // 微信发送消息
  189. const objData: ObjData = reactive({
  190. type: MsgType.Text,
  191. accountId: null,
  192. articles: []
  193. })
  194. const replySelectRef = ref<InstanceType<typeof WxReplySelect> | null>(null)
  195. /** 完成加载 */
  196. onMounted(async () => {
  197. const data = await getUser(props.userId)
  198. user.nickname = data.nickname?.length > 0 ? data.nickname : user.nickname
  199. user.avatar = user.avatar?.length > 0 ? data.avatar : user.avatar
  200. user.accountId = data.accountId
  201. queryParams.accountId = data.accountId
  202. objData.accountId = data.accountId
  203. refreshChange()
  204. })
  205. // 执行发送
  206. const sendMsg = async () => {
  207. if (!objData) {
  208. return
  209. }
  210. // 公众号限制:客服消息,公众号只允许发送一条
  211. if (objData.type === MsgType.News && objData.articles.length > 1) {
  212. objData.articles = [objData.articles[0]]
  213. message.success('图文消息条数限制在 1 条以内,已默认发送第一条')
  214. }
  215. const data = await sendMessage(Object.assign({ userId: props.userId }, { ...objData }))
  216. sendLoading.value = false
  217. list.value = [...list.value, ...[data]]
  218. scrollToBottom()
  219. //ts检查的時候会判断这个组件可能是空的,所以需要进行断言。
  220. //避免 tab 的数据未清理
  221. const deleteObj = replySelectRef.value?.deleteObj
  222. if (deleteObj) {
  223. deleteObj()
  224. }
  225. }
  226. const loadingMore = () => {
  227. queryParams.pageNo++
  228. getPage(queryParams, null)
  229. }
  230. const getPage = async (page, params) => {
  231. loading.value = true
  232. let dataTemp = await getMessagePage(
  233. Object.assign(
  234. {
  235. pageNo: page.pageNo,
  236. pageSize: page.pageSize,
  237. userId: props.userId,
  238. accountId: page.accountId
  239. },
  240. params
  241. )
  242. )
  243. const msgDiv = document.getElementById('msg-div' + nowStr.value)
  244. let scrollHeight = 0
  245. if (msgDiv) {
  246. scrollHeight = msgDiv.scrollHeight
  247. }
  248. // 处理数据
  249. const data = dataTemp.list.reverse()
  250. list.value = [...data, ...list.value]
  251. loading.value = false
  252. if (data.length < queryParams.pageSize || data.length === 0) {
  253. loadMore.value = false
  254. }
  255. queryParams.pageNo = page.pageNo
  256. queryParams.pageSize = page.pageSize
  257. // 滚动到原来的位置
  258. if (queryParams.pageNo === 1) {
  259. // 定位到消息底部
  260. scrollToBottom()
  261. } else if (data.length !== 0) {
  262. // 定位滚动条
  263. await nextTick(() => {
  264. if (scrollHeight !== 0) {
  265. let div = document.getElementById('msg-div' + nowStr.value)
  266. if (div && msgDiv) {
  267. msgDiv.scrollTop = div.scrollHeight - scrollHeight - 100
  268. }
  269. }
  270. })
  271. }
  272. }
  273. const refreshChange = () => {
  274. getPage(queryParams, null)
  275. }
  276. /** 定位到消息底部 */
  277. const scrollToBottom = () => {
  278. nextTick(() => {
  279. let div = document.getElementById('msg-div' + nowStr.value)
  280. if (div) {
  281. div.scrollTop = div.scrollHeight
  282. }
  283. })
  284. }
  285. </script>
  286. <style lang="scss" scoped>
  287. /* 因为 joolun 实现依赖 avue 组件,该页面使用了 comment.scss、card.scc */
  288. @import './comment.scss';
  289. @import './card.scss';
  290. .msg-main {
  291. margin-top: -30px;
  292. padding: 10px;
  293. }
  294. .msg-div {
  295. height: 50vh;
  296. overflow: auto;
  297. background-color: #eaeaea;
  298. margin-left: 10px;
  299. margin-right: 10px;
  300. }
  301. .msg-send {
  302. padding: 10px;
  303. }
  304. .avatar-div {
  305. text-align: center;
  306. width: 80px;
  307. }
  308. .send-but {
  309. float: right;
  310. margin-top: 8px !important;
  311. }
  312. </style>