index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. <template>
  2. <transition @before-leave="init">
  3. <div class="TUI-contact" :class="[env.isH5 ? 'TUI-contact-H5' : '']">
  4. <aside class="TUI-contact-left">
  5. <!-- <header class="TUI-contact-left-header">
  6. <div class="search">
  7. <div class="search-box" @click="toggleSearch" v-if="!isSearch">
  8. <i class="plus"></i>
  9. <h1>{{ $t('TUIContact.添加群聊') }}</h1>
  10. </div>
  11. <div class="search-box" v-else>
  12. <div class="input-box">
  13. <input
  14. type="text"
  15. v-model="searchID"
  16. :placeholder="$t('TUIContact.输入群ID,按回车搜索')"
  17. @keyup.enter="handleSearchGroup"
  18. enterkeyhint="search"
  19. />
  20. <i
  21. class="icon icon-cancel"
  22. v-if="!!searchID"
  23. @click="searchID = ''"
  24. ></i>
  25. </div>
  26. <span class="search-cancel" @click="toggleSearch">
  27. {{ $t('取消') }}
  28. </span>
  29. </div>
  30. </div>
  31. </header> -->
  32. <ul class="TUI-contact-column" v-if="!isSearch">
  33. <li class="TUI-contact-column-item">
  34. <header @click="select('system')">
  35. <i class="icon icon-right" :class="[columnName === 'system' && 'icon-down']"></i>
  36. <main>
  37. <label>{{ $t("TUIContact.群聊通知") }}</label>
  38. <span class="num" v-if="systemConversation && systemConversation.unreadCount > 0">
  39. {{ systemConversation.unreadCount }}
  40. </span>
  41. </main>
  42. </header>
  43. <ul class="TUI-contact-list" v-if="columnName === 'system'">
  44. <li class="TUI-contact-list-item selected not-aside">
  45. <label>{{ $t("TUIContact.系统通知") }}</label>
  46. <span class="num" v-if="systemConversation && systemConversation.unreadCount > 0">
  47. {{ systemConversation.unreadCount }}
  48. </span>
  49. </li>
  50. </ul>
  51. </li>
  52. <li class="TUI-contact-column-item">
  53. <header @click="select('group')">
  54. <i class="icon icon-right" :class="[columnName === 'group' && 'icon-down']"></i>
  55. <main>
  56. <label>{{ $t("TUIContact.我的群聊") }}</label>
  57. </main>
  58. </header>
  59. <ul class="TUI-contact-list" v-show="columnName === 'group'">
  60. <li class="TUI-contact-list-item" :class="[currentGroup?.groupID === item?.groupID && 'selected']" v-for="(item, index) in groupList" :key="index" @click="handleListItem(item)">
  61. <aside class="left">
  62. <img class="avatar" :src="item?.avatar || 'https://oss.dayaedu.com/news-info/07/1690775328089.png'" onerror="this.src='https://oss.dayaedu.com/news-info/07/1690775328089.png'" />
  63. </aside>
  64. <main class="content">
  65. <ul>
  66. <li class="name">{{ item?.name }}</li>
  67. <li class="ID">
  68. <label>ID:</label>
  69. <span>{{ item?.groupID }}</span>
  70. </li>
  71. </ul>
  72. <span class="type">{{ item?.type }}</span>
  73. </main>
  74. </li>
  75. </ul>
  76. </li>
  77. <li class="TUI-contact-column-item">
  78. <header @click="select('friend')">
  79. <i class="icon icon-right" :class="[columnName === 'friend' && 'icon-down']"></i>
  80. <main>
  81. <label>{{ $t("TUIContact.我的好友") }}</label>
  82. </main>
  83. </header>
  84. <ul class="TUI-contact-list" v-show="columnName === 'friend'">
  85. <li class="TUI-contact-list-item" :class="[currentFriend?.userID === item?.userID && 'selected']" v-for="(item, index) in friendList" :key="index" @click="handleListItem(item)">
  86. <aside class="left">
  87. <img class="avatar" :src="item?.profile?.avatar || 'https://oss.dayaedu.com/news-info/07/1690787574969.png'" onerror="this.src='https://oss.dayaedu.com/news-info/07/1690787574969.png'" />
  88. <div class="online-status" :class="userStatusList.get(item?.userID)?.statusType === 1 ? 'online-status-online' : 'online-status-offline'" v-if="displayOnlineStatus"></div>
  89. </aside>
  90. <main class="content">
  91. <ul>
  92. <li class="name">
  93. {{ item?.profile?.nick || item?.userID }}
  94. </li>
  95. </ul>
  96. </main>
  97. </li>
  98. </ul>
  99. </li>
  100. </ul>
  101. <ul class="TUI-contact-list" v-else>
  102. <li v-if="!!searchGroup?.groupID" class="TUI-contact-list-item" :class="[currentGroup?.groupID === searchGroup?.groupID && 'selected']" @click="handleListItem(searchGroup)">
  103. <aside class="left">
  104. <img class="avatar" :src="searchGroup?.avatar || 'https://oss.dayaedu.com/news-info/07/1690775328089.png'" onerror="this.src='https://oss.dayaedu.com/news-info/07/1690775328089.png'" />
  105. </aside>
  106. <main class="content">
  107. <ul>
  108. <li class="name">{{ searchGroup?.name }}</li>
  109. <li class="ID">
  110. <label>ID:</label>
  111. <span>{{ searchGroup?.groupID }}</span>
  112. </li>
  113. </ul>
  114. <span class="type">{{ searchGroup?.type }}</span>
  115. </main>
  116. </li>
  117. </ul>
  118. </aside>
  119. <!-- <main
  120. class="TUI-contact-main"
  121. v-show="
  122. !!currentGroup?.groupID ||
  123. !!currentFriend?.userID ||
  124. columnName === 'system'
  125. "
  126. >
  127. <header class="TUI-contact-main-h5-title" v-if="env.isH5">
  128. <i class="icon icon-back" @click="back"></i>
  129. <h1>{{ currentGroup?.name || $t('TUIContact.系统通知') }}</h1>
  130. </header>
  131. <div v-if="!!currentGroup?.groupID" class="TUI-contact-main-info">
  132. <header class="TUI-contact-main-info-header">
  133. <ul class="list">
  134. <h1>{{ currentGroup?.name }}</h1>
  135. <li>
  136. <label>{{ $t('TUIContact.群ID') }}:</label>
  137. <span>{{ currentGroup?.groupID }}</span>
  138. </li>
  139. <li>
  140. <label>{{ $t('TUIContact.群类型') }}:</label>
  141. <span>{{ currentGroup?.type }}</span>
  142. </li>
  143. </ul>
  144. <img
  145. class="avatar"
  146. :src="
  147. currentGroup?.avatar ||
  148. 'https://web.sdk.qcloud.com/component/TUIKit/assets/group_avatar.png'
  149. "
  150. onerror="this.src='https://oss.dayaedu.com/news-info/07/1690775328089.png'"
  151. />
  152. </header>
  153. <main class="TUI-contact-main-info-main" v-if="isNeedPermission">
  154. <label>{{ $t('TUIContact.请填写验证信息') }}</label>
  155. <textarea
  156. v-model="currentGroup.applyMessage"
  157. :disabled="currentGroup.apply"
  158. ></textarea>
  159. </main>
  160. <footer class="TUI-contact-main-info-footer">
  161. <p v-if="currentGroup.apply && currentGroup.type === 'AVChatRoom'">
  162. {{ $t('TUIContact.已加入') }}
  163. </p>
  164. <p v-else-if="currentGroup.apply">{{ $t('TUIContact.已申请') }}</p>
  165. <button
  166. class="btn btn-default"
  167. v-else-if="isNeedPermission"
  168. @click="join(currentGroup)"
  169. >
  170. {{ $t('TUIContact.申请加入') }}
  171. </button>
  172. <button
  173. class="btn btn-default"
  174. v-else-if="!currentGroup.selfInfo.userID"
  175. @click="join(currentGroup)"
  176. >
  177. {{ $t('TUIContact.加入群聊') }}
  178. </button>
  179. <button
  180. class="btn btn-cancel"
  181. v-else-if="
  182. currentGroup.selfInfo.userID &&
  183. currentGroup.selfInfo.role === 'Owner' &&
  184. currentGroup.type !== 'Private'
  185. "
  186. @click="dismiss(currentGroup)"
  187. >
  188. {{ $t('TUIContact.解散群聊') }}
  189. </button>
  190. <button class="btn btn-cancel" v-else @click="quit(currentGroup)">
  191. {{ $t('TUIContact.退出群聊') }}
  192. </button>
  193. <button
  194. v-if="currentGroup.selfInfo.userID"
  195. class="btn btn-default"
  196. @click="enter(currentGroup.groupID, 'GROUP')"
  197. >
  198. {{ $t('TUIContact.进入群聊') }}
  199. </button>
  200. </footer>
  201. </div>
  202. <div
  203. v-else-if="currentFriend?.userID && columnName === 'friend'"
  204. class="TUI-contact-main-info"
  205. >
  206. <header class="TUI-contact-main-info-header">
  207. <ul class="list">
  208. <h1>
  209. {{ currentFriend?.profile?.nick || currentFriend?.userID }}
  210. </h1>
  211. <li>
  212. <label>ID:</label>
  213. <span>{{ currentFriend?.profile?.userID }}</span>
  214. </li>
  215. <li>
  216. <label>{{ $t('TUIContact.个性签名') }}:</label>
  217. <span>{{ currentFriend?.profile?.selfSignature }}</span>
  218. </li>
  219. </ul>
  220. <img
  221. class="avatar"
  222. :src="
  223. currentFriend?.profile?.avatar ||
  224. 'https://oss.dayaedu.com/news-info/07/1690787574969.png'
  225. "
  226. onerror="this.src='https://oss.dayaedu.com/news-info/07/1690787574969.png'"
  227. />
  228. </header>
  229. <footer class="TUI-contact-main-info-footer">
  230. <button
  231. class="btn btn-default"
  232. @click="enter(currentFriend.userID, 'C2C')"
  233. >
  234. {{ $t('TUIContact.发送消息') }}
  235. </button>
  236. </footer>
  237. </div>
  238. <div class="TUI-contact-system" v-else-if="columnName === 'system'">
  239. <header class="TUI-contact-system-header" v-if="!env.isH5">
  240. <h1>{{ $t('TUIContact.系统通知') }}</h1>
  241. </header>
  242. <MessageSystem
  243. :isH5="env.isH5"
  244. :data="systemMessageList"
  245. :types="types"
  246. @application="handleGroupApplication"
  247. />
  248. </div>
  249. <slot v-else />
  250. </main> -->
  251. </div>
  252. </transition>
  253. </template>
  254. <script lang="ts">
  255. import { computed, defineComponent, reactive, toRefs, watch } from "vue";
  256. // import MessageSystem from './components/message-system.vue';
  257. import { handleErrorPrompts, isArrayEqual } from "../utils";
  258. const TUIContact = defineComponent({
  259. name: "TUIContact",
  260. components: {
  261. // MessageSystem
  262. },
  263. props: {
  264. displayOnlineStatus: {
  265. type: Boolean,
  266. default: false,
  267. },
  268. },
  269. setup(props: any, ctx: any) {
  270. const TUIServer: any = TUIContact.TUIServer;
  271. const { t } = TUIServer.TUICore.config.i18n.useI18n();
  272. const data = reactive({
  273. groupList: [],
  274. searchGroup: {},
  275. searchID: "",
  276. currentGroup: null,
  277. systemConversation: {
  278. unreadCount: 0,
  279. },
  280. systemMessageList: [],
  281. columnName: "",
  282. types: TUIServer.TUICore.TIM.TYPES,
  283. isSearch: false,
  284. env: TUIServer.TUICore.TUIEnv,
  285. friendList: [],
  286. userIDList: [],
  287. currentFriend: {},
  288. displayOnlineStatus: false,
  289. onlineStatus: false,
  290. userStatusList: new Map(),
  291. });
  292. TUIServer.bind(data);
  293. watch(
  294. () => props.displayOnlineStatus,
  295. async (newVal: any, oldVal: any) => {
  296. if (newVal === oldVal) return;
  297. data.displayOnlineStatus = newVal;
  298. TUIServer.handleUserStatus(data.displayOnlineStatus, data.userIDList);
  299. },
  300. {
  301. immediate: true,
  302. }
  303. );
  304. watch(
  305. () => data.userIDList,
  306. (newVal: Array<string>, oldVal: Array<string>) => {
  307. if (isArrayEqual(newVal, oldVal)) return;
  308. TUIServer.handleUserStatus(data.displayOnlineStatus, data.userIDList);
  309. }
  310. );
  311. const isNeedPermission = computed(() => {
  312. const isHaveSeif = (data.currentGroup as any).selfInfo.userID;
  313. const isPermission = (data.currentGroup as any).joinOption === TUIServer.TUICore.TIM.TYPES.JOIN_OPTIONS_NEED_PERMISSION;
  314. return !isHaveSeif && isPermission;
  315. });
  316. const handleListItem = async (item: any) => {
  317. switch (data.columnName) {
  318. case "group":
  319. data.currentGroup = item;
  320. break;
  321. case "friend":
  322. data.currentFriend = item;
  323. break;
  324. }
  325. if (data.isSearch) {
  326. data.currentGroup = item;
  327. }
  328. };
  329. const handleSearchGroup = async (e: any) => {
  330. data.currentGroup = null;
  331. if (data.searchID.trim()) {
  332. try {
  333. await TUIServer.searchGroupByID(data.searchID.trim());
  334. } catch (error) {
  335. const message = t("TUIContact.该群组不存在");
  336. handleErrorPrompts(message, data.env);
  337. }
  338. }
  339. };
  340. const join = async (group: any) => {
  341. const options: any = {
  342. groupID: group.groupID,
  343. applyMessage: group.applyMessage || t("TUIContact.加群"),
  344. type: group?.type,
  345. };
  346. await TUIServer.joinGroup(options);
  347. (data.currentGroup as any).apply = true;
  348. };
  349. const quit = async (group: any) => {
  350. await TUIServer.quitGroup(group.groupID);
  351. data.currentGroup = null;
  352. };
  353. const enter = async (ID: any, type: string) => {
  354. const name = `${type}${ID}`;
  355. TUIServer.TUICore.TUIServer.TUIConversation.getConversationProfile(name).then((imResponse: any) => {
  356. // 通知 TUIConversation 添加当前会话
  357. // Notify TUIConversation to toggle the current conversation
  358. TUIServer.TUICore.TUIServer.TUIConversation.handleCurrentConversation(imResponse.data.conversation);
  359. back();
  360. });
  361. };
  362. const dismiss = (group: any) => {
  363. TUIServer.dismissGroup(group.groupID);
  364. data.currentGroup = null;
  365. };
  366. const select = async (name: string) => {
  367. if (data.columnName !== "system" && name === "system" && (data.systemConversation as any)?.conversationID) {
  368. await TUIServer.getSystemMessageList();
  369. await TUIServer.setMessageRead();
  370. }
  371. (data.currentGroup as any) = {};
  372. if (data.columnName !== "group" && name === "group" && !data.env.isH5) {
  373. (data.currentGroup as any) = data.groupList[0];
  374. } else {
  375. (data.currentGroup as any) = {};
  376. }
  377. data.searchID = "";
  378. data.columnName = data.columnName === name ? "" : name;
  379. };
  380. const toggleSearch = () => {
  381. data.isSearch = !data.isSearch;
  382. data.columnName = "";
  383. data.searchID = "";
  384. data.searchGroup = {};
  385. (data.currentGroup as any) = {};
  386. };
  387. const init = () => {
  388. data.isSearch = false;
  389. data.columnName = "";
  390. data.searchID = "";
  391. data.searchGroup = {};
  392. (data.currentGroup as any) = {};
  393. };
  394. const handleGroupApplication = (params: any) => {
  395. TUIServer.handleGroupApplication(params);
  396. };
  397. const back = () => {
  398. (data.currentGroup as any) = {};
  399. (data.currentFriend as any) = {};
  400. data.columnName = "";
  401. };
  402. const getUserStatusList = (userList: Array<string>) => {
  403. TUIServer.TUICore.getUserStatusList(userList);
  404. };
  405. return {
  406. ...toRefs(data),
  407. handleListItem,
  408. handleSearchGroup,
  409. join,
  410. quit,
  411. dismiss,
  412. isNeedPermission,
  413. select,
  414. handleGroupApplication,
  415. toggleSearch,
  416. init,
  417. back,
  418. enter,
  419. getUserStatusList,
  420. };
  421. },
  422. });
  423. export default TUIContact;
  424. </script>
  425. <style lang="scss" scoped src="./style/index.scss"></style>