index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. <template>
  2. <div>
  3. <div>
  4. <div class="item" @click="person">
  5. <div>
  6. <img :src="lxr" class="img" />
  7. </div>
  8. <div class="title">新的联系人</div>
  9. </div>
  10. <div class="item" @click="chat">
  11. <div>
  12. <img :src="ql" class="img" />
  13. </div>
  14. <div class="title">我的群聊</div>
  15. </div>
  16. <div class="item" @click="Blacklist">
  17. <div>
  18. <img :src="hmd" class="img" />
  19. </div>
  20. <div class="title">黑名单</div>
  21. </div>
  22. <liu-indexed-list :dataList="dataList" @click="selectItem"></liu-indexed-list>
  23. </div>
  24. <liu-drag-button @clickBtn="clickBtn">
  25. <Icon :file="addSVG" :width="'14px'" :height="'14px'" />
  26. </liu-drag-button>
  27. <!-- <ul
  28. v-if="!contactSearchingStatus"
  29. :class="['tui-contact-list', !isPC && 'tui-contact-list-h5']"
  30. >
  31. <li
  32. v-for="(contactListObj, key) in contactListMap"
  33. :key="key"
  34. class="tui-contact-list-item"
  35. >
  36. <header
  37. class="tui-contact-list-item-header"
  38. @click="toggleCurrentContactList(key)"
  39. >
  40. <div class="tui-contact-list-item-header-left">
  41. <Icon
  42. :file="currentContactListKey === key ? downSVG : rightSVG"
  43. width="16px"
  44. height="16px"
  45. />
  46. <div>{{ TUITranslateService.t(`TUIContact.${contactListObj.title}`) }}</div>
  47. </div>
  48. <div class="tui-contact-list-item-header-right">
  49. <span
  50. v-if="contactListObj.unreadCount"
  51. class="tui-contact-list-item-header-right-unread"
  52. >
  53. {{ contactListObj.unreadCount }}
  54. </span>
  55. </div>
  56. </header>
  57. <ul
  58. :class="[
  59. 'tui-contact-list-item-main',
  60. currentContactListKey === key ? '' : 'hidden',
  61. ]"
  62. >
  63. <li
  64. v-for="contactListItem in contactListObj.list"
  65. :key="contactListItem.renderKey"
  66. class="tui-contact-list-item-main-item"
  67. :class="['selected']"
  68. @click="selectItem(contactListItem)"
  69. >
  70. <ContactListItem
  71. :item="contactListItem"
  72. :displayOnlineStatus="displayOnlineStatus && key === 'friendList'"
  73. />
  74. </li>
  75. </ul>
  76. </li>
  77. </ul>
  78. <ul v-else class="tui-contact-list">
  79. <li
  80. v-for="(item, key) in contactSearchResult"
  81. :key="key"
  82. class="tui-contact-list-item"
  83. >
  84. <div v-if="item.list[0]" class="tui-contact-search-list">
  85. <div class="tui-contact-search-list-title">
  86. {{ TUITranslateService.t(`TUIContact.${item.label}`) }}
  87. </div>
  88. <div
  89. v-for="(listItem, index) in item.list"
  90. :key="index"
  91. class="tui-contact-search-list-item"
  92. :class="['selected']"
  93. @click="selectItem(listItem)"
  94. >
  95. <ContactListItem :item="listItem" :displayOnlineStatus="false" />
  96. </div>
  97. </div>
  98. </li>
  99. <div v-if="isContactSearchNoResult" class="tui-contact-search-list-default">
  100. {{ TUITranslateService.t("TUIContact.无搜索结果") }}
  101. </div>
  102. </ul> -->
  103. </div>
  104. </template>
  105. <script setup lang="ts">
  106. import {
  107. TUITranslateService,
  108. TUIStore,
  109. StoreName,
  110. IGroupModel,
  111. TUIFriendService,
  112. Friend,
  113. FriendApplication,
  114. TUIUserService,
  115. } from "@tencentcloud/chat-uikit-engine";
  116. import TUICore, { TUIConstants } from "@tencentcloud/tui-core";
  117. import { ref, computed, onMounted, onUnmounted, provide } from "../../../adapter-vue";
  118. import Icon from "../../common/Icon.vue";
  119. import downSVG from "../../../assets/icon/down-icon.svg";
  120. import rightSVG from "../../../assets/icon/right-icon.svg";
  121. import addSVG from "../../../assets/icon/add.svg";
  122. import lxr from "../../../../static/img/lxr.png";
  123. import ql from "../../../../static/img/ql.png";
  124. import hmd from "../../../../static/img/hmd.png";
  125. import { TUIGlobal } from "@tencentcloud/universal-api";
  126. import {
  127. IContactList,
  128. IContactSearchResult,
  129. IBlackListUserItem,
  130. IUserStatus,
  131. IUserStatusMap,
  132. IContactInfoType,
  133. } from "../../../interface";
  134. import ContactListItem from "./contact-list-item/index.vue";
  135. import { isPC } from "../../../utils/env";
  136. const currentContactListKey = ref<keyof IContactList>("");
  137. const currentContactInfo = ref<IContactInfoType>({} as IContactInfoType);
  138. const dataList = ref([]);
  139. const contactListMap = ref<IContactList>({
  140. friendApplicationList: {
  141. key: "friendApplicationList",
  142. title: "新的联系人",
  143. list: [] as FriendApplication[],
  144. unreadCount: 0,
  145. },
  146. blackList: {
  147. key: "blackList",
  148. title: "黑名单",
  149. list: [] as IBlackListUserItem[],
  150. },
  151. groupList: {
  152. key: "groupList",
  153. title: "我的群聊",
  154. list: [] as IGroupModel[],
  155. },
  156. friendList: {
  157. key: "friendList",
  158. title: "我的好友",
  159. list: [] as Friend[],
  160. },
  161. });
  162. const contactSearchingStatus = ref<boolean>(false);
  163. const contactSearchResult = ref<IContactSearchResult>();
  164. const displayOnlineStatus = ref<boolean>(false);
  165. const userOnlineStatusMap = ref<IUserStatusMap>();
  166. const show = ref(false);
  167. const clickBtn = () => {
  168. // show.value = true;
  169. TUIGlobal?.navigateTo({
  170. url: "/pages/group/add",
  171. });
  172. };
  173. const isContactSearchNoResult = computed((): boolean => {
  174. return (
  175. !contactSearchResult?.value?.user?.list[0] &&
  176. !contactSearchResult?.value?.group?.list[0]
  177. );
  178. });
  179. const Blacklist = () => {
  180. TUIGlobal?.navigateTo({
  181. url: "/pages/group/black-list?list=" + JSON.stringify(contactListMap.value),
  182. });
  183. };
  184. const person = () => {
  185. TUIGlobal?.navigateTo({
  186. url: "/pages/group/person?list=" + JSON.stringify(contactListMap.value),
  187. });
  188. };
  189. const chat = () => {
  190. TUIGlobal?.navigateTo({
  191. url: "/pages/group/chat?list=" + JSON.stringify(contactListMap.value),
  192. });
  193. };
  194. onMounted(() => {
  195. console.log('触发----------------------');
  196. TUIStore.watch(StoreName.APP, {
  197. enabledCustomerServicePlugin: onCustomerServiceCommercialPluginUpdated,
  198. });
  199. TUIStore.watch(StoreName.GRP, {
  200. groupList: onGroupListUpdated,
  201. });
  202. TUIStore.watch(StoreName.USER, {
  203. userBlacklist: onUserBlacklistUpdated,
  204. displayOnlineStatus: onDisplayOnlineStatusUpdated,
  205. userStatusList: onUserStatusListUpdated,
  206. });
  207. TUIStore.watch(StoreName.FRIEND, {
  208. friendList: onFriendListUpdated,
  209. friendApplicationList: onFriendApplicationListUpdated,
  210. friendApplicationUnreadCount: onFriendApplicationUnreadCountUpdated,
  211. });
  212. TUIStore.watch(StoreName.CUSTOM, {
  213. currentContactSearchingStatus: onCurrentContactSearchingStatusUpdated,
  214. currentContactSearchResult: onCurrentContactSearchResultUpdated,
  215. currentContactListKey: onCurrentContactListKeyUpdated,
  216. currentContactInfo: onCurrentContactInfoUpdated,
  217. });
  218. console.log(contactListMap.value);
  219. let list = [];
  220. contactListMap.value.friendList.list.forEach((e) => {
  221. list.push(e.profile);
  222. });
  223. dataList.value = list;
  224. console.log("dataList.value", contactListMap.value);
  225. });
  226. onUnmounted(() => {
  227. TUIStore.unwatch(StoreName.APP, {
  228. enabledCustomerServicePlugin: onCustomerServiceCommercialPluginUpdated,
  229. });
  230. TUIStore.unwatch(StoreName.GRP, {
  231. groupList: onGroupListUpdated,
  232. });
  233. TUIStore.unwatch(StoreName.USER, {
  234. userBlacklist: onUserBlacklistUpdated,
  235. displayOnlineStatus: onDisplayOnlineStatusUpdated,
  236. userStatusList: onUserStatusListUpdated,
  237. });
  238. TUIStore.unwatch(StoreName.FRIEND, {
  239. friendList: onFriendListUpdated,
  240. friendApplicationList: onFriendApplicationListUpdated,
  241. friendApplicationUnreadCount: onFriendApplicationUnreadCountUpdated,
  242. });
  243. TUIStore.unwatch(StoreName.CUSTOM, {
  244. currentContactSearchingStatus: onCurrentContactSearchingStatusUpdated,
  245. currentContactSearchResult: onCurrentContactSearchResultUpdated,
  246. currentContactListKey: onCurrentContactListKeyUpdated,
  247. currentContactInfo: onCurrentContactInfoUpdated,
  248. });
  249. });
  250. function toggleCurrentContactList(key: keyof IContactList) {
  251. if (currentContactListKey.value === key) {
  252. currentContactListKey.value = "";
  253. currentContactInfo.value = {} as IContactInfoType;
  254. TUIStore.update(StoreName.CUSTOM, "currentContactListKey", "");
  255. TUIStore.update(StoreName.CUSTOM, "currentContactInfo", {} as IContactInfoType);
  256. } else {
  257. currentContactListKey.value = key;
  258. TUIStore.update(StoreName.CUSTOM, "currentContactListKey", key);
  259. if (key === "friendApplicationList") {
  260. TUIFriendService.setFriendApplicationRead();
  261. }
  262. }
  263. }
  264. function selectItem(item: any) {
  265. currentContactInfo.value = item;
  266. // For a result in the search list, before viewing the contactInfo details,
  267. // it is necessary to update the data for the "already in the group list/already in the friend list" situation to obtain more detailed information
  268. if (contactSearchingStatus.value) {
  269. let targetListItem;
  270. if ((currentContactInfo.value as Friend)?.userID) {
  271. targetListItem = contactListMap.value?.friendList?.list?.find(
  272. (item: IContactInfoType) =>
  273. (item as Friend)?.userID === (currentContactInfo.value as Friend)?.userID
  274. );
  275. } else if ((currentContactInfo.value as IGroupModel)?.groupID) {
  276. targetListItem = contactListMap.value?.groupList?.list?.find(
  277. (item: IContactInfoType) =>
  278. (item as IGroupModel)?.groupID ===
  279. (currentContactInfo.value as IGroupModel)?.groupID
  280. );
  281. }
  282. if (targetListItem) {
  283. currentContactInfo.value = targetListItem;
  284. }
  285. }
  286. TUIStore.update(StoreName.CUSTOM, "currentContactInfo", currentContactInfo.value);
  287. }
  288. function onDisplayOnlineStatusUpdated(status: boolean) {
  289. displayOnlineStatus.value = status;
  290. }
  291. function onUserStatusListUpdated(list: Map<string, IUserStatus>) {
  292. if (list?.size > 0) {
  293. userOnlineStatusMap.value = Object.fromEntries(list?.entries());
  294. }
  295. }
  296. function onCustomerServiceCommercialPluginUpdated(isEnabled: boolean) {
  297. if (!isEnabled) {
  298. return;
  299. }
  300. // After the customer purchases the customer service plug-in,
  301. // the engine updates the enabledCustomerServicePlugin to true through the commercial capability bit.
  302. const contactListExtensionID = TUIConstants.TUIContact.EXTENSION.CONTACT_LIST.EXT_ID;
  303. const tuiContactExtensionList = TUICore.getExtensionList(contactListExtensionID);
  304. const customerData = tuiContactExtensionList.find((extension: any) => {
  305. const { name, accountList = [] } = extension.data || {};
  306. return name === "customer" && accountList.length > 0;
  307. });
  308. if (customerData) {
  309. const { data, text } = customerData;
  310. const { accountList } = (data || {}) as { accountList: string[] };
  311. TUIUserService.getUserProfile({ userIDList: accountList })
  312. .then((res) => {
  313. if (res.data.length > 0) {
  314. const customerList = {
  315. title: text,
  316. list: res.data.map((item: any, index: number) => {
  317. return {
  318. ...item,
  319. renderKey: generateRenderKey("customerList", item, index),
  320. infoKeyList: [],
  321. btnKeyList: ["enterC2CConversation"],
  322. };
  323. }),
  324. key: "customerList",
  325. };
  326. contactListMap.value = { ...contactListMap.value, customerList };
  327. }
  328. })
  329. .catch(() => {});
  330. }
  331. }
  332. function onGroupListUpdated(groupList: IGroupModel[]) {
  333. updateContactListMap("groupList", groupList);
  334. }
  335. function onUserBlacklistUpdated(userBlacklist: IBlackListUserItem[]) {
  336. updateContactListMap("blackList", userBlacklist);
  337. }
  338. function onFriendApplicationUnreadCountUpdated(friendApplicationUnreadCount: number) {
  339. contactListMap.value.friendApplicationList.unreadCount = friendApplicationUnreadCount;
  340. }
  341. function onFriendListUpdated(friendList: Friend[]) {
  342. updateContactListMap("friendList", friendList);
  343. }
  344. function onFriendApplicationListUpdated(friendApplicationList: FriendApplication[]) {
  345. updateContactListMap("friendApplicationList", friendApplicationList);
  346. }
  347. function updateContactListMap(key: keyof IContactList, list: IContactInfoType[]) {
  348. contactListMap.value[key].list = list;
  349. contactListMap.value[key].list.map(
  350. (item: IContactInfoType, index: number) =>
  351. (item.renderKey = generateRenderKey(key, item, index))
  352. );
  353. updateCurrentContactInfoFromList(contactListMap.value[key].list, key);
  354. }
  355. function updateCurrentContactInfoFromList(
  356. list: IContactInfoType[],
  357. type: keyof IContactList
  358. ) {
  359. if (
  360. !(currentContactInfo.value as Friend)?.userID &&
  361. !(currentContactInfo.value as IGroupModel)?.groupID
  362. ) {
  363. return;
  364. }
  365. if (type === currentContactListKey.value || contactSearchingStatus.value) {
  366. currentContactInfo.value =
  367. list?.find(
  368. (item: any) =>
  369. (item?.groupID &&
  370. item?.groupID === (currentContactInfo.value as IGroupModel)?.groupID) ||
  371. (item?.userID && item?.userID === (currentContactInfo.value as Friend)?.userID)
  372. ) || ({} as IContactInfoType);
  373. TUIStore.update(StoreName.CUSTOM, "currentContactInfo", currentContactInfo.value);
  374. }
  375. }
  376. function generateRenderKey(
  377. contactListMapKey: keyof IContactList,
  378. contactInfo: IContactInfoType,
  379. index: number
  380. ) {
  381. return `${contactListMapKey}-${
  382. (contactInfo as Friend).userID ||
  383. (contactInfo as IGroupModel).groupID ||
  384. "index" + index
  385. }`;
  386. }
  387. function onCurrentContactSearchResultUpdated(searchResult: IContactSearchResult) {
  388. contactSearchResult.value = searchResult;
  389. }
  390. function onCurrentContactSearchingStatusUpdated(searchingStatus: boolean) {
  391. contactSearchingStatus.value = searchingStatus;
  392. TUIStore.update(StoreName.CUSTOM, "currentContactInfo", {} as IContactInfoType);
  393. TUIStore.update(StoreName.CUSTOM, "currentContactListKey", "");
  394. }
  395. function onCurrentContactInfoUpdated(contactInfo: IContactInfoType) {
  396. currentContactInfo.value = contactInfo;
  397. }
  398. function onCurrentContactListKeyUpdated(contactListKey: string) {
  399. currentContactListKey.value = contactListKey;
  400. }
  401. provide("userOnlineStatusMap", userOnlineStatusMap);
  402. </script>
  403. <style lang="scss" scoped src="./style/index.scss"></style>