index.vue 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937
  1. <template>
  2. <div
  3. :class="{
  4. 'tui-chat': true,
  5. 'tui-chat-h5': isMobile,
  6. }"
  7. @click="onMessageListBackgroundClick"
  8. >
  9. <!-- <JoinGroupCard /> -->
  10. <div class="tui-chat-main">
  11. <div
  12. v-if="isMerchantShow"
  13. class="tui-chat-safe-tips"
  14. style="text-align: right"
  15. @click="goGroup(currentConversationID)"
  16. >
  17. 进入店铺
  18. </div>
  19. <MessageGroupApplication
  20. v-if="isGroup"
  21. :key="props.groupID"
  22. :groupID="props.groupID"
  23. />
  24. <scroll-view
  25. id="messageScrollList"
  26. class="tui-message-list"
  27. style="background: #f1f1f1"
  28. scroll-y="true"
  29. :scroll-top="scrollTop"
  30. :scroll-into-view="`tui-${historyFirstMessageID}`"
  31. @scroll="handelScrollListScroll"
  32. >
  33. <p
  34. v-if="!isCompleted"
  35. class="message-more"
  36. @click="getHistoryMessageList"
  37. >
  38. {{ TUITranslateService.t("TUIChat.查看更多") }}
  39. </p>
  40. <li
  41. v-for="(item, index) in messageList"
  42. :id="`tui-${item.ID}`"
  43. :key="item.vueForRenderKey"
  44. :class="'message-li ' + item.flow"
  45. >
  46. <MessageTimestamp
  47. :currTime="item.time"
  48. :prevTime="index > 0 ? messageList[index - 1].time : 0"
  49. />
  50. <div class="message-item" @click="toggleID = ''">
  51. <MessageTip
  52. v-if="
  53. item.type === TYPES.MSG_GRP_TIP ||
  54. isCreateGroupCustomMessage(item)
  55. "
  56. :content="item.getMessageContent()"
  57. />
  58. <div
  59. v-else-if="!item.isRevoked && !isPluginMessage(item)"
  60. :id="`msg-bubble-${item.ID}`"
  61. class="message-bubble-container"
  62. @longpress="handleToggleMessageItem($event, item, index, true)"
  63. @touchstart="handleH5LongPress($event, item, index, 'touchstart')"
  64. @touchend="handleH5LongPress($event, item, index, 'touchend')"
  65. @mouseover="handleH5LongPress($event, item, index, 'touchend')"
  66. >
  67. <MessageBubble
  68. :messageItem="item"
  69. :content="item.getMessageContent()"
  70. :isAudioPlayed="audioPlayedMapping[item.ID]"
  71. :blinkMessageIDList="blinkMessageIDList"
  72. :isMultipleSelectMode="isMultipleSelectMode"
  73. :multipleSelectedMessageIDList="multipleSelectedMessageIDList"
  74. @resendMessage="resendMessage(item)"
  75. @blinkMessage="blinkMessage"
  76. @scrollTo="scrollTo"
  77. @changeSelectMessageIDList="changeSelectMessageIDList"
  78. @setReadReceiptPanelVisible="setReadReceiptPanelVisible"
  79. >
  80. <MessageText
  81. v-if="item.type === TYPES.MSG_TEXT"
  82. :content="item.getMessageContent()"
  83. />
  84. <ProgressMessage
  85. v-else-if="item.type === TYPES.MSG_IMAGE"
  86. :content="item.getMessageContent()"
  87. :messageItem="item"
  88. >
  89. <MessageImage
  90. :content="item.getMessageContent()"
  91. :messageItem="item"
  92. @previewImage="handleImagePreview(index)"
  93. />
  94. </ProgressMessage>
  95. <ProgressMessage
  96. v-else-if="item.type === TYPES.MSG_VIDEO"
  97. :content="item.getMessageContent()"
  98. :messageItem="item"
  99. >
  100. <MessageVideo
  101. :content="item.getMessageContent()"
  102. :messageItem="item"
  103. />
  104. </ProgressMessage>
  105. <MessageAudio
  106. v-else-if="item.type === TYPES.MSG_AUDIO"
  107. :content="item.getMessageContent()"
  108. :messageItem="item"
  109. :broadcastNewAudioSrc="broadcastNewAudioSrc"
  110. @setAudioPlayed="setAudioPlayed"
  111. @getGlobalAudioContext="getGlobalAudioContext"
  112. />
  113. <MessageRecord
  114. v-else-if="item.type === TYPES.MSG_MERGER"
  115. :renderData="item.payload"
  116. :messageItem="item"
  117. @assignMessageIDInUniapp="assignMessageIDInUniapp"
  118. />
  119. <MessageFile
  120. v-else-if="item.type === TYPES.MSG_FILE"
  121. :content="item.getMessageContent()"
  122. />
  123. <MessageFace
  124. v-else-if="item.type === TYPES.MSG_FACE"
  125. :content="item.getMessageContent()"
  126. />
  127. <MessageLocation
  128. v-else-if="item.type === TYPES.MSG_LOCATION"
  129. :content="item.getMessageContent()"
  130. />
  131. <MessageCustom
  132. @handleDraw="handleDraw"
  133. class="MessageCustom"
  134. v-else-if="item.payload.data"
  135. :content="item.getMessageContent()"
  136. :messageItem="item"
  137. />
  138. </MessageBubble>
  139. </div>
  140. <MessagePlugin
  141. v-else-if="!item.isRevoked && isPluginMessage(item)"
  142. :message="item"
  143. @resendMessage="resendMessage"
  144. @handleToggleMessageItem="handleToggleMessageItem"
  145. @handleH5LongPress="handleH5LongPress"
  146. />
  147. <MessageRevoked
  148. v-else
  149. :isEdit="item.type === TYPES.MSG_TEXT"
  150. :messageItem="item"
  151. @messageEdit="handleEdit(item)"
  152. />
  153. <!-- message tool -->
  154. <MessageTool
  155. v-if="item.ID === toggleID"
  156. :class="{
  157. 'message-tool': true,
  158. 'message-tool-out': item.flow === 'out',
  159. 'message-tool-in': item.flow === 'in',
  160. }"
  161. :messageItem="item"
  162. :isMultipleSelectMode="isMultipleSelectMode"
  163. @toggleMultipleSelectMode="
  164. () => emits('toggleMultipleSelectMode')
  165. "
  166. />
  167. </div>
  168. </li>
  169. </scroll-view>
  170. <!-- scroll button -->
  171. <ScrollButton
  172. ref="scrollButtonInstanceRef"
  173. @scrollToLatestMessage="scrollToLatestMessage"
  174. />
  175. <Dialog
  176. v-if="reSendDialogShow"
  177. :show="reSendDialogShow"
  178. :isH5="!isPC"
  179. :center="true"
  180. :isHeaderShow="isPC"
  181. @submit="resendMessageConfirm()"
  182. @update:show="(e) => (reSendDialogShow = e)"
  183. >
  184. <p class="delDialog-title">
  185. {{ TUITranslateService.t("TUIChat.确认重发该消息?") }}
  186. </p>
  187. </Dialog>
  188. <!-- read receipt panel -->
  189. <ReadReceiptPanel
  190. v-if="isShowReadUserStatusPanel"
  191. :message="Object.assign({}, readStatusMessage)"
  192. @setReadReceiptPanelVisible="setReadReceiptPanelVisible"
  193. />
  194. <!-- simple message list -->
  195. <Drawer
  196. :visible="isShowSimpleMessageList"
  197. :overlayColor="'transparent'"
  198. :popDirection="'right'"
  199. >
  200. <SimpleMessageList
  201. :style="{ height: '100%' }"
  202. :isMounted="isShowSimpleMessageList"
  203. :messageID="simpleMessageListRenderMessageID"
  204. @closeOverlay="isShowSimpleMessageList = false"
  205. />
  206. </Drawer>
  207. </div>
  208. <div class="mask" v-if="showMask">
  209. <div class="back">
  210. <div class="header">恭喜您获得现金红包</div>
  211. <div class="main">
  212. <span class="price">{{ redEnvelopeAmount }}</span>
  213. <span class="unit">元</span>
  214. </div>
  215. <div class="footer">
  216. <span class="text">红包可在【个人中心-我的钱包】中查看</span>
  217. <button class="btn" @click="take">开心收下</button>
  218. </div>
  219. </div>
  220. </div>
  221. </div>
  222. </template>
  223. <script lang="ts" setup>
  224. import {
  225. ref,
  226. watch,
  227. nextTick,
  228. onMounted,
  229. onUnmounted,
  230. getCurrentInstance,
  231. } from "../../../adapter-vue";
  232. import TUIChatEngine, {
  233. IMessageModel,
  234. TUIStore,
  235. StoreName,
  236. TUITranslateService,
  237. TUIChatService,
  238. } from "@tencentcloud/chat-uikit-engine";
  239. import { throttle } from "lodash";
  240. import {
  241. setInstanceMapping,
  242. getBoundingClientRect,
  243. getScrollInfo,
  244. } from "@tencentcloud/universal-api";
  245. // import { JoinGroupCard } from '@tencentcloud/call-uikit-wechat';
  246. import Link from "./link";
  247. import SimpleMessageList from "./message-elements/simple-message-list/index.vue";
  248. import MessageGroupApplication from "./message-group-application/index.vue";
  249. import MessageText from "./message-elements/message-text.vue";
  250. import MessageImage from "./message-elements/message-image.vue";
  251. import MessageAudio from "./message-elements/message-audio.vue";
  252. import MessageRecord from "./message-elements/message-record/index.vue";
  253. import MessageFile from "./message-elements/message-file.vue";
  254. import MessageFace from "./message-elements/message-face.vue";
  255. import MessageCustom from "./message-elements/message-custom.vue";
  256. import MessageTip from "./message-elements/message-tip.vue";
  257. import MessageBubble from "./message-elements/message-bubble.vue";
  258. import MessageLocation from "./message-elements/message-location.vue";
  259. import MessageTimestamp from "./message-elements/message-timestamp.vue";
  260. import MessageVideo from "./message-elements/message-video.vue";
  261. import MessageTool from "./message-tool/index.vue";
  262. import MessageRevoked from "./message-tool/message-revoked.vue";
  263. import MessagePlugin from "../../../plugins/plugin-components/message-plugin.vue";
  264. import ReadReceiptPanel from "./read-receipt-panel/index.vue";
  265. import ScrollButton from "./scroll-button/index.vue";
  266. import { isPluginMessage } from "../../../plugins/plugin-components/index";
  267. import Dialog from "../../common/Dialog/index.vue";
  268. import Drawer from "../../common/Drawer/index.vue";
  269. import { Toast, TOAST_TYPE } from "../../common/Toast/index";
  270. import ProgressMessage from "../../common/ProgressMessage/index.vue";
  271. import { isCreateGroupCustomMessage } from "../utils/utils";
  272. import { isEnabledMessageReadReceiptGlobal } from "../utils/utils";
  273. import { isPC, isH5, isMobile } from "../../../utils/env";
  274. import chatStorage from "../utils/chatStorage";
  275. import { IAudioContext } from "../../../interface";
  276. import { requestConfig } from "../../../../app.config";
  277. import * as mesApi from "../../../../api/message/index";
  278. interface IEmits {
  279. (e: "closeInputToolBar"): void;
  280. (e: "handleEditor", message: IMessageModel, type: string): void;
  281. (key: "toggleMultipleSelectMode"): void;
  282. }
  283. interface IProps {
  284. isGroup: boolean;
  285. groupID: string;
  286. isNotInGroup: boolean;
  287. isMultipleSelectMode: boolean;
  288. }
  289. const showMask = ref(false);
  290. const emits = defineEmits<IEmits>();
  291. const props = withDefaults(defineProps<IProps>(), {
  292. isGroup: false,
  293. groupID: "",
  294. isNotInGroup: false,
  295. isMultipleSelectMode: false,
  296. });
  297. let selfAddValue = 0;
  298. let observer: any = null;
  299. let groupType: string | undefined;
  300. const sentReceiptMessageID = new Set<string>();
  301. const isOfficial = TUIStore.getData(StoreName.APP, "isOfficial");
  302. const thisInstance = getCurrentInstance()?.proxy || getCurrentInstance();
  303. const messageList = ref<IMessageModel[]>();
  304. const multipleSelectedMessageIDList = ref<string[]>([]);
  305. const isCompleted = ref(false);
  306. const currentConversationID = ref("");
  307. const toggleID = ref("");
  308. const scrollTop = ref(5000); // The initial number of messages is 15, and the maximum message height is 300.
  309. const TYPES = ref(TUIChatEngine.TYPES);
  310. const isLoadingMessage = ref(false);
  311. const isLongpressing = ref(false);
  312. const blinkMessageIDList = ref<string[]>([]);
  313. const messageTarget = ref<IMessageModel>();
  314. const scrollButtonInstanceRef = ref<InstanceType<typeof ScrollButton>>();
  315. const historyFirstMessageID = ref<string>("");
  316. const isShowSimpleMessageList = ref<boolean>(false);
  317. const simpleMessageListRenderMessageID = ref<string>();
  318. const audioPlayedMapping = ref<Record<string, boolean>>({});
  319. // audio control
  320. const broadcastNewAudioSrc = ref<string>("");
  321. const readStatusMessage = ref<IMessageModel>();
  322. const isShowReadUserStatusPanel = ref<boolean>(false);
  323. // Resend Message Dialog
  324. const reSendDialogShow = ref(false);
  325. const resendMessageData = ref();
  326. const scrollToBottom = () => {
  327. scrollTop.value += 300;
  328. // Solve the issue where swiping to the bottom for the first time after packaging Uniapp into an app has a delay,
  329. // which can be set to 300 ms.
  330. const timer = setTimeout(() => {
  331. scrollTop.value += 1;
  332. clearTimeout(timer);
  333. }, 300);
  334. };
  335. const onCurrentConversationIDUpdated = (conversationID: string) => {
  336. currentConversationID.value = conversationID;
  337. if (isEnabledMessageReadReceiptGlobal()) {
  338. const { groupProfile } =
  339. TUIStore.getConversationModel(conversationID) || {};
  340. groupType = groupProfile?.type;
  341. }
  342. if (Object.keys(audioPlayedMapping.value).length > 0) {
  343. // Synchronize storage about whether the audio has been played when converstaion switched
  344. chatStorage.setChatStorage("audioPlayedMapping", audioPlayedMapping.value);
  345. }
  346. };
  347. const redEnvelopeAmount = ref("");
  348. const redInfo = ref();
  349. const handleDraw = (record) => {
  350. redInfo.value = record;
  351. redEnvelopeAmount.value = record.info.redEnvelopeAmount;
  352. if(!record.isOk){
  353. Toast({
  354. message: TUITranslateService.t(record.message),
  355. type: TOAST_TYPE.ERROR,
  356. });
  357. }else{
  358. showMask.value = true;
  359. }
  360. };
  361. const take = () => {
  362. if (redInfo.value.isOk) {
  363. Toast({
  364. message: TUITranslateService.t("红包可在【个人中心-我的钱包】中查看!"),
  365. type: TOAST_TYPE.SUCCESS,
  366. });
  367. } else {
  368. Toast({
  369. message: TUITranslateService.t("红包领取失败,请联系客服反馈!"),
  370. type: TOAST_TYPE.ERROR,
  371. });
  372. }
  373. showMask.value = false;
  374. };
  375. onMounted(() => {
  376. // Retrieve the information about whether the audio has been played from localStorage
  377. audioPlayedMapping.value =
  378. chatStorage.getChatStorage("audioPlayedMapping") || {};
  379. TUIStore.watch(StoreName.CHAT, {
  380. messageList: onMessageListUpdated,
  381. messageSource: onMessageSourceUpdated,
  382. isCompleted: onChatCompletedUpdated,
  383. });
  384. TUIStore.watch(StoreName.CONV, {
  385. currentConversationID: onCurrentConversationIDUpdated,
  386. });
  387. setInstanceMapping("messageList", thisInstance);
  388. uni.$on("scroll-to-bottom", scrollToLatestMessage);
  389. let personId = uni.getStorageSync("personId");
  390. if (personId != requestConfig.customerId) {
  391. isMerchant();
  392. }
  393. });
  394. const goGroup = async () => {
  395. let id = currentConversationID.value.slice(3);
  396. let res = await mesApi.default.getByOwnerId(id);
  397. console.log("123123", res);
  398. if (res.code == 200) {
  399. webUni.webView.navigateTo({
  400. url: `/subpages/my/product-orders/in-store-look?shopId=` + res.data.id,
  401. });
  402. }
  403. };
  404. const GetQueryString = (name: string) => {
  405. var reg = new RegExp("(^|&)" + name + "=([^&]*)(&|$)");
  406. var r = window.location.search.substr(1).match(reg);
  407. if (r != null) return unescape(r[2]);
  408. return null;
  409. };
  410. const isMerchantShow = ref(false);
  411. const isMerchant = async () => {
  412. if (currentConversationID.value.includes("GROUP")) {
  413. isMerchantShow.value = false;
  414. } else {
  415. // let id = currentConversationID.value.slice(3);
  416. let id = GetQueryString("id");
  417. let res = await mesApi.default.getUserInfo(id);
  418. if (res.data.userTypes.includes("2")) {
  419. isMerchantShow.value = true;
  420. } else {
  421. isMerchantShow.value = false;
  422. }
  423. }
  424. console.log("xxxxx");
  425. };
  426. onUnmounted(() => {
  427. TUIStore.unwatch(StoreName.CHAT, {
  428. messageList: onMessageListUpdated,
  429. isCompleted: onChatCompletedUpdated,
  430. });
  431. TUIStore.unwatch(StoreName.CONV, {
  432. currentConversationID: onCurrentConversationIDUpdated,
  433. });
  434. observer?.disconnect();
  435. observer = null;
  436. uni.$off("scroll-to-bottom");
  437. if (Object.keys(audioPlayedMapping.value).length > 0) {
  438. // Synchronize storage about whether the audio has been played when the component is unmounted
  439. chatStorage.setChatStorage("audioPlayedMapping", audioPlayedMapping.value);
  440. }
  441. });
  442. const handelScrollListScroll = throttle(
  443. function (e: Event) {
  444. scrollButtonInstanceRef.value?.judgeScrollOverOneScreen(e);
  445. },
  446. 500,
  447. { leading: true }
  448. );
  449. function getGlobalAudioContext(
  450. audioMap: Map<string, IAudioContext>,
  451. options?: { newAudioSrc: string }
  452. ) {
  453. if (options?.newAudioSrc) {
  454. broadcastNewAudioSrc.value = options.newAudioSrc;
  455. }
  456. }
  457. async function onMessageListUpdated(list: IMessageModel[]) {
  458. observer?.disconnect();
  459. messageList.value = list
  460. .filter((message) => !message.isDeleted)
  461. .map((message) => {
  462. message.vueForRenderKey = `${message.ID}`;
  463. return message;
  464. });
  465. const newLastMessage = messageList.value?.[messageList.value?.length - 1];
  466. if (messageTarget.value) {
  467. // scroll to target message
  468. scrollAndBlinkMessage(messageTarget.value);
  469. } else if (
  470. !isLoadingMessage.value &&
  471. !(
  472. scrollButtonInstanceRef.value?.isScrollButtonVisible &&
  473. newLastMessage?.flow === "in"
  474. )
  475. ) {
  476. // scroll to bottom
  477. nextTick(() => {
  478. scrollToBottom();
  479. });
  480. }
  481. if (isEnabledMessageReadReceiptGlobal()) {
  482. nextTick(() => bindIntersectionObserver());
  483. }
  484. }
  485. async function scrollToLatestMessage() {
  486. try {
  487. const { scrollHeight } = await getScrollInfo(
  488. "#messageScrollList",
  489. "messageList"
  490. );
  491. if (scrollHeight) {
  492. scrollTop.value === scrollHeight
  493. ? (scrollTop.value = scrollHeight + 1)
  494. : (scrollTop.value = scrollHeight);
  495. } else {
  496. scrollToBottom();
  497. }
  498. } catch (error) {
  499. scrollToBottom();
  500. }
  501. }
  502. async function onMessageSourceUpdated(message: IMessageModel) {
  503. messageTarget.value = message;
  504. scrollAndBlinkMessage(messageTarget.value);
  505. }
  506. function scrollAndBlinkMessage(message: IMessageModel) {
  507. if (
  508. messageList.value?.some(
  509. (messageListItem) => messageListItem?.ID === message?.ID
  510. )
  511. ) {
  512. nextTick(async () => {
  513. await scrollToTargetMessage(message);
  514. await blinkMessage(message?.ID);
  515. messageTarget.value = undefined;
  516. });
  517. }
  518. }
  519. function onChatCompletedUpdated(flag: boolean) {
  520. isCompleted.value = flag;
  521. }
  522. const getHistoryMessageList = () => {
  523. isLoadingMessage.value = true;
  524. const currentFirstMessageID = messageList.value?.[0]?.ID || "";
  525. TUIChatService.getMessageList().then(() => {
  526. nextTick(() => {
  527. historyFirstMessageID.value = currentFirstMessageID;
  528. const timer = setTimeout(() => {
  529. historyFirstMessageID.value = "";
  530. isLoadingMessage.value = false;
  531. clearTimeout(timer);
  532. }, 500);
  533. });
  534. });
  535. };
  536. const openComplaintLink = () => {};
  537. // toggle message
  538. const handleToggleMessageItem = (
  539. e: any,
  540. message: IMessageModel,
  541. index: number,
  542. isLongpress = false
  543. ) => {
  544. if (props.isMultipleSelectMode || props.isNotInGroup) {
  545. return;
  546. }
  547. if (isLongpress) {
  548. isLongpressing.value = true;
  549. }
  550. toggleID.value = message.ID;
  551. };
  552. // h5 long press
  553. let timer: number;
  554. const handleH5LongPress = (
  555. e: any,
  556. message: IMessageModel,
  557. index: number,
  558. type: string
  559. ) => {
  560. if (props.isMultipleSelectMode || props.isNotInGroup) {
  561. return;
  562. }
  563. if (!isH5) return;
  564. function longPressHandler() {
  565. clearTimeout(timer);
  566. handleToggleMessageItem(e, message, index, true);
  567. }
  568. function touchStartHandler() {
  569. timer = setTimeout(longPressHandler, 500);
  570. }
  571. function touchEndHandler() {
  572. clearTimeout(timer);
  573. }
  574. switch (type) {
  575. case "touchstart":
  576. touchStartHandler();
  577. break;
  578. case "touchend":
  579. touchEndHandler();
  580. setTimeout(() => {
  581. isLongpressing.value = false;
  582. }, 200);
  583. break;
  584. }
  585. };
  586. // reedit message
  587. const handleEdit = (message: IMessageModel) => {
  588. emits("handleEditor", message, "reedit");
  589. };
  590. const resendMessage = (message: IMessageModel) => {
  591. reSendDialogShow.value = true;
  592. resendMessageData.value = message;
  593. };
  594. const handleImagePreview = (index: number) => {
  595. if (!messageList.value) {
  596. return;
  597. }
  598. const imageMessageIndex: number[] = [];
  599. const imageMessageList: IMessageModel[] = messageList.value.filter(
  600. (item, index) => {
  601. if (
  602. !item.isRevoked &&
  603. !item.hasRiskContent &&
  604. item.type === TYPES.value.MSG_IMAGE
  605. ) {
  606. imageMessageIndex.push(index);
  607. return true;
  608. }
  609. return false;
  610. }
  611. );
  612. uni.previewImage({
  613. current: imageMessageIndex.indexOf(index),
  614. urls: imageMessageList.map(
  615. (message) => message.payload.imageInfoArray?.[2].url
  616. ),
  617. // #ifdef APP-PLUS
  618. indicator: "number",
  619. // #endif
  620. });
  621. };
  622. const resendMessageConfirm = () => {
  623. reSendDialogShow.value = !reSendDialogShow.value;
  624. const messageModel = resendMessageData.value;
  625. messageModel.resendMessage();
  626. };
  627. function blinkMessage(messageID: string): Promise<void> {
  628. return new Promise((resolve) => {
  629. const index = blinkMessageIDList.value.indexOf(messageID);
  630. if (index < 0) {
  631. blinkMessageIDList.value.push(messageID);
  632. const timer = setTimeout(() => {
  633. blinkMessageIDList.value.splice(
  634. blinkMessageIDList.value.indexOf(messageID),
  635. 1
  636. );
  637. clearTimeout(timer);
  638. resolve();
  639. }, 3000);
  640. }
  641. });
  642. }
  643. function scrollTo(scrollHeight: number) {
  644. scrollTop.value = scrollHeight;
  645. }
  646. async function bindIntersectionObserver() {
  647. if (!messageList.value || messageList.value.length === 0) {
  648. return;
  649. }
  650. if (
  651. groupType === TYPES.value.GRP_AVCHATROOM ||
  652. groupType === TYPES.value.GRP_COMMUNITY
  653. ) {
  654. // AVCHATROOM and COMMUNITY chats do not monitor read receipts for messages.
  655. return;
  656. }
  657. observer?.disconnect();
  658. observer = uni
  659. .createIntersectionObserver(thisInstance, {
  660. threshold: [0.7],
  661. observeAll: true,
  662. // In Uni-app, the `safetip` is also included, so a negative margin is needed to exclude it.
  663. })
  664. .relativeTo("#messageScrollList", { top: -70 });
  665. observer?.observe(".message-li.in .message-bubble-container", (res: any) => {
  666. if (sentReceiptMessageID.has(res.id)) {
  667. return;
  668. }
  669. const matchingMessage = messageList.value.find((message: IMessageModel) => {
  670. return res.id.indexOf(message.ID) > -1;
  671. });
  672. if (
  673. matchingMessage &&
  674. matchingMessage.needReadReceipt &&
  675. matchingMessage.flow === "in" &&
  676. !matchingMessage.readReceiptInfo?.isPeerRead
  677. ) {
  678. TUIChatService.sendMessageReadReceipt([matchingMessage]);
  679. sentReceiptMessageID.add(res.id);
  680. }
  681. });
  682. }
  683. function setReadReceiptPanelVisible(visible: boolean, message?: IMessageModel) {
  684. if (visible && props.isNotInGroup) {
  685. return;
  686. }
  687. if (!visible) {
  688. readStatusMessage.value = undefined;
  689. } else {
  690. readStatusMessage.value = message;
  691. }
  692. isShowReadUserStatusPanel.value = visible;
  693. }
  694. async function scrollToTargetMessage(message: IMessageModel) {
  695. const targetMessageID = message.ID;
  696. const isTargetMessageInScreen =
  697. messageList.value &&
  698. messageList.value.some((msg) => msg.ID === targetMessageID);
  699. if (targetMessageID && isTargetMessageInScreen) {
  700. const timer = setTimeout(async () => {
  701. try {
  702. const scrollViewRect = await getBoundingClientRect(
  703. "#messageScrollList",
  704. "messageList"
  705. );
  706. const originalMessageRect = await getBoundingClientRect(
  707. "#tui-" + targetMessageID,
  708. "messageList"
  709. );
  710. const { scrollTop } = await getScrollInfo(
  711. "#messageScrollList",
  712. "messageList"
  713. );
  714. const finalScrollTop =
  715. originalMessageRect.top +
  716. scrollTop -
  717. scrollViewRect.top -
  718. (selfAddValue++ % 2);
  719. scrollTo(finalScrollTop);
  720. clearTimeout(timer);
  721. } catch (error) {
  722. // todo
  723. }
  724. }, 500);
  725. } else {
  726. Toast({
  727. message: TUITranslateService.t("TUIChat.无法定位到原消息"),
  728. type: TOAST_TYPE.WARNING,
  729. });
  730. }
  731. }
  732. function onMessageListBackgroundClick() {
  733. emits("closeInputToolBar");
  734. }
  735. watch(
  736. () => props.isMultipleSelectMode,
  737. (newValue) => {
  738. if (!newValue) {
  739. changeSelectMessageIDList({
  740. type: "clearAll",
  741. messageID: "",
  742. });
  743. }
  744. }
  745. );
  746. function changeSelectMessageIDList({
  747. type,
  748. messageID,
  749. }: {
  750. type: "add" | "remove" | "clearAll";
  751. messageID: string;
  752. }) {
  753. // TODO need to delete this
  754. if (type === "clearAll") {
  755. multipleSelectedMessageIDList.value = [];
  756. } else if (
  757. type === "add" &&
  758. !multipleSelectedMessageIDList.value.includes(messageID)
  759. ) {
  760. multipleSelectedMessageIDList.value.push(messageID);
  761. } else if (type === "remove") {
  762. multipleSelectedMessageIDList.value =
  763. multipleSelectedMessageIDList.value.filter((id) => id !== messageID);
  764. }
  765. }
  766. function mergeForwardMessage() {
  767. TUIStore.update(StoreName.CUSTOM, "multipleForwardMessageID", {
  768. isMergeForward: true,
  769. messageIDList: multipleSelectedMessageIDList.value,
  770. });
  771. }
  772. function oneByOneForwardMessage() {
  773. TUIStore.update(StoreName.CUSTOM, "multipleForwardMessageID", {
  774. isMergeForward: false,
  775. messageIDList: multipleSelectedMessageIDList.value,
  776. });
  777. }
  778. function assignMessageIDInUniapp(messageID: string) {
  779. simpleMessageListRenderMessageID.value = messageID;
  780. isShowSimpleMessageList.value = true;
  781. }
  782. function setAudioPlayed(messageID: string) {
  783. audioPlayedMapping.value[messageID] = true;
  784. }
  785. defineExpose({
  786. oneByOneForwardMessage,
  787. mergeForwardMessage,
  788. scrollToLatestMessage,
  789. });
  790. </script>
  791. <style lang="scss" scoped src="./style/index.scss"></style>
  792. <style lang="scss" scoped>
  793. .mask {
  794. width: 100vw;
  795. height: 100vh;
  796. position: absolute;
  797. background: #00000080;
  798. left: 0px;
  799. top: 0px;
  800. right: 0px;
  801. z-index: 9999;
  802. display: flex;
  803. flex-direction: row;
  804. justify-content: center;
  805. align-items: center;
  806. }
  807. .back {
  808. position: relative;
  809. bottom: 100px;
  810. height: 400px;
  811. width: 300px;
  812. // background: linear-gradient(225deg, #f74d30 0%, #ff7633 100%);
  813. background: url("@/static/img/lqhbBack.png") no-repeat;
  814. background-size: 100% 100%;
  815. padding: 150px 20px 0px 20px !important;
  816. display: block !important;
  817. box-shadow: 0rpx 4rpx 12rpx 0rpx rgba(226, 226, 226, 0.23);
  818. border-radius: 0px 20px 20px 20px;
  819. .header {
  820. font-weight: bold;
  821. font-size: 36rpx;
  822. color: #000000;
  823. text-align: left;
  824. font-style: normal;
  825. text-align: center;
  826. }
  827. .main {
  828. text-align: center;
  829. height: 100px;
  830. display: flex;
  831. align-items: center;
  832. flex-direction: row;
  833. justify-content: center;
  834. .price {
  835. font-weight: 800;
  836. font-size: 88rpx;
  837. color: #ff2f2f;
  838. text-align: left;
  839. font-style: normal;
  840. }
  841. .unit {
  842. font-weight: 800;
  843. font-size: 19px;
  844. color: #ff2f2f;
  845. text-align: left;
  846. font-style: normal;
  847. position: relative;
  848. bottom: -7px;
  849. }
  850. }
  851. .footer {
  852. text-align: center;
  853. .text {
  854. font-weight: 500;
  855. font-size: 28rpx;
  856. color: #ff2f2f;
  857. text-align: center;
  858. font-style: normal;
  859. }
  860. .btn {
  861. margin-top: 20px;
  862. width: 506rpx;
  863. height: 100rpx;
  864. background: linear-gradient(90deg, #ff3737 0%, #ff7b1d 100%);
  865. border-radius: 56rpx;
  866. display: flex;
  867. flex-direction: row;
  868. justify-content: center;
  869. align-items: center;
  870. font-weight: 500;
  871. font-size: 40rpx;
  872. color: #ffffff;
  873. font-style: normal;
  874. }
  875. }
  876. }
  877. </style>