MessageList.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. <template>
  2. <div class="flex radius-base message-list-modal">
  3. <div class="message-list p-base scroll-y-auto">
  4. <template v-if="!!(messageList || []).length">
  5. <div
  6. v-for="message in messageList"
  7. :key="message.id"
  8. class="radius-base fill-blank p-base m-b-mini relative message-item"
  9. :class="{ active: currentMessage?.sendUserId === message.sendUserId, 'un-read': message.unReadCount > 0 }"
  10. @click="checkMessage(message)"
  11. >
  12. <div class="flex items-center m-b-base message-title">
  13. <div class="message-send-user">{{ message.sendUserName }}</div>
  14. <div class="m-l-auto message-send-time">{{ dayjs(message.sendTime).format('HH:mm') }}</div>
  15. </div>
  16. <!-- <pre class="message-content" v-html="message.content"></pre> -->
  17. <pre class="message-content">{{ transHtmlContent(message.content) }}</pre>
  18. </div>
  19. </template>
  20. <div v-else class="none-msg-box">
  21. <div class="center-box">
  22. <svg-icon name="none_message" style="font-size: 120px"></svg-icon>
  23. <p class="tip">暂无消息</p>
  24. </div>
  25. </div>
  26. </div>
  27. <div class="radius-base message-info-container">
  28. <div class="flex direction-column message-info">
  29. <div class="flex items-center p-base message-info-header">
  30. <div class="flex items-center send-user">
  31. <span class="m-r-mini">发件人</span>
  32. <span class="radius-base user-name">{{ currentMessage?.sendUserName }}</span>
  33. </div>
  34. <div class="grid pointer m-l-auto close-icon" @click="$emit('close')">
  35. <el-icon><close /></el-icon>
  36. </div>
  37. </div>
  38. <div class="flex-1 overflow-hidden p-base">
  39. <pre
  40. class="full-h radius-base p-extra-base scroll-y-auto message-info-content"
  41. @click="onContentClick"
  42. v-html="currentMessage?.content"
  43. ></pre>
  44. </div>
  45. <div class="p-base flex items-center justify-end">
  46. <el-button size="small" type="primary" :disabled="!currentMessage" @click="onReply">回复</el-button>
  47. <el-button size="small" plain :disabled="!currentMessage" @click="toggleHistory">历史消息</el-button>
  48. </div>
  49. </div>
  50. <message-history v-if="showHistory" :send-user-id="currentMessage?.sendUserId"></message-history>
  51. </div>
  52. </div>
  53. <image-preview v-model="previewModalVisible" :url="paperPath" :is-big="true"></image-preview>
  54. </template>
  55. <script setup lang="ts" name="MessageList">
  56. /** 消息列表*/
  57. import { ref, watch, onUnmounted } from 'vue'
  58. import { ElButton, ElIcon } from 'element-plus'
  59. import dayjs from 'dayjs'
  60. import { Close } from '@element-plus/icons-vue'
  61. import useFetch from '@/hooks/useFetch'
  62. import MessageHistory from '@/components/shared/message/MessageHistory.vue'
  63. import ImagePreview from '../ImagePreview.vue'
  64. import useMainStore from '@/store/main'
  65. import uesFetch from '@/hooks/useFetch'
  66. import { setLastMsgs } from '@/hooks/useMessageLoop'
  67. import SvgIcon from '@/components/common/SvgIcon.vue'
  68. import { transHtmlContent } from '@/utils/common'
  69. import type { ExtractApiResponse } from '@/api/api'
  70. type MessageType = ExtractArrayValue<ExtractApiResponse<'getMessageList'>>
  71. const mainStore = useMainStore()
  72. const emits = defineEmits(['close', 'change-type', 'reply'])
  73. /** 图片预览 */
  74. const previewModalVisible = ref<boolean>(false)
  75. /** 图片路径 */
  76. const paperPath = ref<string>('')
  77. const showHistory = ref<boolean>(false)
  78. const { fetch: getMessageList, result: messageList } = useFetch('getMessageList')
  79. const currentMessage = ref<MessageType>()
  80. watch(currentMessage, () => {
  81. if (currentMessage.value) {
  82. useFetch('handleReadMessage')
  83. .fetch({ id: currentMessage.value.id })
  84. .then(() => {
  85. if (currentMessage.value?.unReadCount) {
  86. currentMessage.value.unReadCount--
  87. }
  88. })
  89. }
  90. })
  91. const checkMessage = (message: MessageType) => {
  92. currentMessage.value = message
  93. }
  94. const toggleHistory = () => {
  95. showHistory.value = !showHistory.value
  96. }
  97. const onReply = () => {
  98. if (currentMessage.value) {
  99. emits('change-type', 'send')
  100. emits('reply', currentMessage.value.sendUserId)
  101. }
  102. }
  103. const onContentClick = (e: Event) => {
  104. const target = e.target as HTMLButtonElement
  105. const path = target.getAttribute('data-path')
  106. if (path) {
  107. previewModalVisible.value = true
  108. paperPath.value = path
  109. }
  110. }
  111. getMessageList().then((result) => (currentMessage.value = result?.[0]))
  112. watch(
  113. () => mainStore.newMsgs,
  114. () => {
  115. getMessageList().then((res) => {
  116. if (res && Array.isArray(res)) {
  117. let find = res.find((item) => item.sendUserId == currentMessage.value?.sendUserId)
  118. if (!!find) {
  119. currentMessage.value = find
  120. }
  121. }
  122. })
  123. }
  124. )
  125. onUnmounted(() => {
  126. uesFetch('getUnReadMessage')
  127. .fetch()
  128. .then((res) => {
  129. setLastMsgs(res)
  130. })
  131. })
  132. </script>
  133. <style scoped lang="scss">
  134. .message-list-modal {
  135. width: 750px;
  136. background-color: transparent;
  137. .message-list {
  138. width: 260px;
  139. height: 446px;
  140. background: #fafafa;
  141. box-shadow: 0px 6px 6px 0px rgba(0, 0, 0, 0.1);
  142. .none-msg-box {
  143. width: 100%;
  144. height: 100%;
  145. display: flex;
  146. justify-content: center;
  147. align-items: center;
  148. .center-box {
  149. text-align: center;
  150. .tip {
  151. color: #999;
  152. margin-top: 5px;
  153. }
  154. }
  155. }
  156. .message-item {
  157. &.un-read:after {
  158. content: '';
  159. position: absolute;
  160. right: 0;
  161. top: 0;
  162. border-radius: 50%;
  163. width: 8px;
  164. height: 8px;
  165. background-color: $DangerColor;
  166. }
  167. &.active {
  168. background-color: $color--primary;
  169. color: $color--white;
  170. }
  171. &:not(.active) {
  172. .message-title {
  173. color: $NormalColor;
  174. .message-send-time {
  175. color: $RegularFontColor;
  176. }
  177. }
  178. .message-content {
  179. color: $RegularFontColor;
  180. }
  181. }
  182. .message-title {
  183. font-size: $BaseFont;
  184. .message-send-time {
  185. font-size: $SmallFont;
  186. font-weight: 400;
  187. }
  188. }
  189. .message-content {
  190. font-weight: 400;
  191. overflow: hidden;
  192. text-overflow: ellipsis;
  193. display: -webkit-box;
  194. -webkit-line-clamp: 2;
  195. -webkit-box-orient: vertical;
  196. font-size: $SmallFont;
  197. white-space: pre-wrap;
  198. }
  199. }
  200. }
  201. .message-info-container {
  202. // width: 600px;
  203. width: calc(100% - 260px);
  204. position: relative;
  205. .message-info {
  206. height: 446px;
  207. background-color: $color--white;
  208. .message-info-header {
  209. border-bottom: $OnePixelLine;
  210. .send-user {
  211. font-size: $SmallFont;
  212. color: $RegularFontColor;
  213. .user-name {
  214. display: inline-block;
  215. width: 160px;
  216. padding: 0 12px;
  217. border: $OnePixelLine;
  218. height: 38px;
  219. line-height: 36px;
  220. }
  221. }
  222. .close-icon {
  223. width: 20px;
  224. height: 20px;
  225. place-items: center;
  226. font-size: 18px;
  227. color: $RegularFontColor;
  228. &:hover {
  229. color: $NormalColor;
  230. }
  231. }
  232. }
  233. .message-info-content {
  234. border: $OnePixelLine;
  235. font-size: $SmallFont;
  236. white-space: pre-wrap;
  237. }
  238. }
  239. }
  240. }
  241. </style>