UNPKG

ngx-tencent-im

Version:
752 lines (733 loc) 283 kB
import * as i0 from '@angular/core'; import { InjectionToken, Injectable, Inject, Component, ChangeDetectionStrategy, Input, ViewChild, EventEmitter, Output, NgModule } from '@angular/core'; import { Subject } from 'rxjs'; import * as i1 from '@ngrx/store'; import { createAction, props, createSelector, createReducer, on, StoreModule } from '@ngrx/store'; import TIM$1 from 'tim-js-sdk'; import TIMUploadPlugin from 'tim-upload-plugin'; import * as i1$1 from '@angular/forms'; import { Validators, FormsModule, ReactiveFormsModule } from '@angular/forms'; import * as i2 from 'ng-zorro-antd/modal'; import { NzModalModule } from 'ng-zorro-antd/modal'; import * as i5 from 'ng-zorro-antd/form'; import { NzFormModule } from 'ng-zorro-antd/form'; import * as i6 from 'ng-zorro-antd/radio'; import { NzRadioModule } from 'ng-zorro-antd/radio'; import * as i7 from 'ng-zorro-antd/button'; import { NzButtonModule } from 'ng-zorro-antd/button'; import * as i8 from 'ng-zorro-antd/grid'; import * as i9 from 'ng-zorro-antd/input'; import { NzInputModule } from 'ng-zorro-antd/input'; import * as i5$1 from '@angular/common'; import { CommonModule } from '@angular/common'; import * as i10 from 'ng-zorro-antd/core/wave'; import * as i6$1 from 'ng-zorro-antd/core/transition-patch'; import * as i1$2 from 'ng-zorro-antd/avatar'; import { NzAvatarModule } from 'ng-zorro-antd/avatar'; import * as i6$2 from 'ng-zorro-antd/icon'; import { NzIconModule } from 'ng-zorro-antd/icon'; import * as i7$1 from 'ng-zorro-antd/popover'; import { NzPopoverModule } from 'ng-zorro-antd/popover'; import { debounceTime, distinctUntilChanged } from 'rxjs/operators'; import * as i6$3 from 'ng-zorro-antd/select'; import { NzSelectModule } from 'ng-zorro-antd/select'; import * as i1$3 from 'ng-zorro-antd/dropdown'; import { NzDropDownModule } from 'ng-zorro-antd/dropdown'; import * as i6$4 from 'ng-zorro-antd/menu'; import * as i2$1 from 'ng-zorro-antd/image'; import { NzImageModule } from 'ng-zorro-antd/image'; import * as i11 from 'ng-zorro-antd/tooltip'; import { NzToolTipModule } from 'ng-zorro-antd/tooltip'; import * as i3 from 'ng-zorro-antd/input-number'; import { NzInputNumberModule } from 'ng-zorro-antd/input-number'; import * as i6$5 from 'ng-zorro-antd/switch'; import { NzSwitchModule } from 'ng-zorro-antd/switch'; import * as i6$6 from '@angular/cdk/drag-drop'; import { DragDropModule } from '@angular/cdk/drag-drop'; import { NzBadgeModule } from 'ng-zorro-antd/badge'; import { NzCollapseModule } from 'ng-zorro-antd/collapse'; import { NzMessageModule } from 'ng-zorro-antd/message'; import { NzPopconfirmModule } from 'ng-zorro-antd/popconfirm'; var LoginActionTypes; (function (LoginActionTypes) { LoginActionTypes["Login"] = "[toggleIsLogin]"; })(LoginActionTypes || (LoginActionTypes = {})); const loginAction = createAction(LoginActionTypes.Login, props()); var UserActionTypes; (function (UserActionTypes) { UserActionTypes["SDKReady"] = "[user] toggleIsSDKReady"; UserActionTypes["updateCurrentUserProfile"] = "[user] updateCurrentUserProfile"; UserActionTypes["resetUser"] = "[user] resetUser"; })(UserActionTypes || (UserActionTypes = {})); const SDKReadyAction = createAction(UserActionTypes.SDKReady, props()); const updateCurrentUserProfileAction = createAction(UserActionTypes.updateCurrentUserProfile, props()); const resetUserAction = createAction(UserActionTypes.resetUser); const showAction = createAction('[showMessage]', props()); const updateMessageAction = createAction('[updateMessage]', props()); ; const updateCurrentConversationAction = createAction('[conversation] updateCurrentConversation', props()); const updateConversationListAction = createAction('[conversation] updateConversationList', props()); const resetCurrentConversationAction = createAction('[conversation] resetCurrentConversation'); const pushCurrentMessageListAction = createAction('[conversation] pushCurrentMessageList', props()); const removeMessageAction = createAction('[conversation] removeMessage', props()); const resetConversationAction = createAction('[conversation] reset'); const checkoutConversationAction = createAction('[conversation] checkoutConversation', props()); const selectPlayerStates$3 = (state) => state.currentConversationID; const getCurrentConversationID = createSelector(selectPlayerStates$3, (state) => state.conversationID); const selectConversation = (state) => state.conversation; const conversationSelector = createSelector(selectConversation, (state) => state); const selectCurrentConversation = (state) => state.conversation.currentConversation; const currentConversationSelector = createSelector(selectCurrentConversation, (state) => state); const selectConversationList = (state) => state.conversation.conversationList; const conversationListSelector = createSelector(selectConversationList, (state) => state); const selectPlayerStates$2 = (state) => state.login; const getLogin = createSelector(selectPlayerStates$2, (state) => state.isLogin); const selectPlayerStates$1 = (state) => state.message; const getMessage = createSelector(selectPlayerStates$1, (state) => state); const selectPlayerStates = (state) => state.user; const getSDkReady = createSelector(selectPlayerStates, (state) => state.isSDKReady); const currentUserProfileSelector = createSelector(selectPlayerStates, (state) => state.currentUserProfile); const updateGroupListAction = createAction('[group] updateGroupList', props()); const updateCurrentMemberListAction = createAction('[group] updateCurrentMemberList', props()); const resetCurrentMemberListAction = createAction('[group] resetCurrentMemberList'); const selectGroupListStates = (state) => state.group.groupList; const groupListSelector = createSelector(selectGroupListStates, (state) => state); const currentMemberListStates = (state) => state.group.currentMemberList; const currentMemberListSelector = createSelector(currentMemberListStates, (state) => state); const NG_Tim_CONFIG = new InjectionToken('config'); var MESSAGE_STATUS; (function (MESSAGE_STATUS) { MESSAGE_STATUS["success"] = "success"; MESSAGE_STATUS["info"] = "info"; MESSAGE_STATUS["warning"] = "warning"; MESSAGE_STATUS["error"] = "error"; })(MESSAGE_STATUS || (MESSAGE_STATUS = {})); var CONVERSATION_TYPE; (function (CONVERSATION_TYPE) { CONVERSATION_TYPE["client"] = "C2C"; CONVERSATION_TYPE["group"] = "GROUP"; CONVERSATION_TYPE["system"] = "@TIM#SYSTEM"; })(CONVERSATION_TYPE || (CONVERSATION_TYPE = {})); var TIM_TYPES; (function (TIM_TYPES) { TIM_TYPES["GRP_WORK"] = "Private"; TIM_TYPES["GRP_PUBLIC"] = "Public"; TIM_TYPES["GRP_MEETING"] = "ChatRoom"; TIM_TYPES["GRP_AVCHATROOM"] = "AVChatRoom"; })(TIM_TYPES || (TIM_TYPES = {})); const TIM = { TYPES: { "MSG_TEXT": "TIMTextElem", "MSG_IMAGE": "TIMImageElem", "MSG_SOUND": "TIMSoundElem", "MSG_AUDIO": "TIMSoundElem", "MSG_FILE": "TIMFileElem", "MSG_FACE": "TIMFaceElem", "MSG_VIDEO": "TIMVideoFileElem", "MSG_GEO": "TIMLocationElem", "MSG_GRP_TIP": "TIMGroupTipElem", "MSG_GRP_SYS_NOTICE": "TIMGroupSystemNoticeElem", "MSG_CUSTOM": "TIMCustomElem", "MSG_MERGER": "TIMRelayElem", "MSG_PRIORITY_HIGH": "High", "MSG_PRIORITY_NORMAL": "Normal", "MSG_PRIORITY_LOW": "Low", "MSG_PRIORITY_LOWEST": "Lowest", "CONV_C2C": "C2C", "CONV_GROUP": "GROUP", "CONV_SYSTEM": "@TIM%23SYSTEM", "CONV_AT_ME": 1, "CONV_AT_ALL": 2, "CONV_AT_ALL_AT_ME": 3, "GRP_PRIVATE": "Private", "GRP_WORK": "Private", "GRP_PUBLIC": "Public", "GRP_CHATROOM": "ChatRoom", "GRP_MEETING": "ChatRoom", "GRP_AVCHATROOM": "AVChatRoom", "GRP_MBR_ROLE_OWNER": "Owner", "GRP_MBR_ROLE_ADMIN": "Admin", "GRP_MBR_ROLE_MEMBER": "Member", "GRP_TIP_MBR_JOIN": 1, "GRP_TIP_MBR_QUIT": 2, "GRP_TIP_MBR_KICKED_OUT": 3, "GRP_TIP_MBR_SET_ADMIN": 4, "GRP_TIP_MBR_CANCELED_ADMIN": 5, "GRP_TIP_GRP_PROFILE_UPDATED": 6, "GRP_TIP_MBR_PROFILE_UPDATED": 7, "MSG_REMIND_ACPT_AND_NOTE": "AcceptAndNotify", "MSG_REMIND_ACPT_NOT_NOTE": "AcceptNotNotify", "MSG_REMIND_DISCARD": "Discard", "GENDER_UNKNOWN": "Gender_Type_Unknown", "GENDER_FEMALE": "Gender_Type_Female", "GENDER_MALE": "Gender_Type_Male", "KICKED_OUT_MULT_ACCOUNT": "multipleAccount", "KICKED_OUT_MULT_DEVICE": "multipleDevice", "KICKED_OUT_USERSIG_EXPIRED": "userSigExpired", "ALLOW_TYPE_ALLOW_ANY": "AllowType_Type_AllowAny", "ALLOW_TYPE_NEED_CONFIRM": "AllowType_Type_NeedConfirm", "ALLOW_TYPE_DENY_ANY": "AllowType_Type_DenyAny", "FORBID_TYPE_NONE": "AdminForbid_Type_None", "FORBID_TYPE_SEND_OUT": "AdminForbid_Type_SendOut", "JOIN_OPTIONS_FREE_ACCESS": "FreeAccess", "JOIN_OPTIONS_NEED_PERMISSION": "NeedPermission", "JOIN_OPTIONS_DISABLE_APPLY": "DisableApply", "JOIN_STATUS_SUCCESS": "JoinedSuccess", "JOIN_STATUS_ALREADY_IN_GROUP": "AlreadyInGroup", "JOIN_STATUS_WAIT_APPROVAL": "WaitAdminApproval", "GRP_PROFILE_OWNER_ID": "ownerID", "GRP_PROFILE_CREATE_TIME": "createTime", "GRP_PROFILE_LAST_INFO_TIME": "lastInfoTime", "GRP_PROFILE_MEMBER_NUM": "memberNum", "GRP_PROFILE_MAX_MEMBER_NUM": "maxMemberNum", "GRP_PROFILE_JOIN_OPTION": "joinOption", "GRP_PROFILE_INTRODUCTION": "introduction", "GRP_PROFILE_NOTIFICATION": "notification", "GRP_PROFILE_MUTE_ALL_MBRS": "muteAllMembers", "NET_STATE_CONNECTED": "connected", "NET_STATE_CONNECTING": "connecting", "NET_STATE_DISCONNECTED": "disconnected", "MSG_AT_ALL": "__kImSDK_MesssageAtALL__" }, EVENT: { "SDK_READY": "sdkStateReady", "SDK_NOT_READY": "sdkStateNotReady", "SDK_DESTROY": "sdkDestroy", "MESSAGE_RECEIVED": "onMessageReceived", "MESSAGE_REVOKED": "onMessageRevoked", "MESSAGE_READ_BY_PEER": "onMessageReadByPeer", "CONVERSATION_LIST_UPDATED": "onConversationListUpdated", "GROUP_LIST_UPDATED": "onGroupListUpdated", "GROUP_SYSTEM_NOTICE_RECEIVED": "receiveGroupSystemNotice", "PROFILE_UPDATED": "onProfileUpdated", "BLACKLIST_UPDATED": "blacklistUpdated", "KICKED_OUT": "kickedOut", "ERROR": "error", "NET_STATE_CHANGE": "netStateChange", "SDK_RELOAD": "sdkReload" }, VERSION: "2.10.2" }; class TimHelperService { constructor(store, config) { this.store = store; this.config = config; this.eventBus$ = new Subject(); this.totalUnRead = new Subject(); this.initTim(config); // 初始化监听器 this.initListener(); // 获取当前会话 this.store.select(conversationSelector).subscribe(res => { this.conversation = res; }); // 获取当前成员 this.store.select(currentMemberListSelector).subscribe(res => { this.currentMemberList = res; }); } login(userId, userSig) { if (!userSig) { throw new Error('请配置签名!'); } this.tim.login({ userID: userId, userSig }) .then((imResponse) => { this.eventBus$.next('login'); this.store.dispatch(loginAction({ isLogin: true })); // this.store.dispatch(startComputeCurrentAction()); this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.success, message: '登录成功!' })); if (imResponse.data.repeatLogin === true) { // 标识账号已登录,本次登录操作为重复登录。v2.5.1 起支持 console.log(imResponse.data.errorInfo); } }).catch((imError) => { console.warn('login error:', imError); // 登录失败的相关信息 }); } logout() { // 若有当前会话,在退出登录时已读上报 if (this.conversation.currentConversation.conversationID) { this.tim.setMessageRead({ conversationID: this.conversation.currentConversation.conversationID }); } this.tim.logout().then((res) => { this.eventBus$.next('logout'); // this.store.dispatch(stopComputeCurrentAction()); this.store.dispatch(loginAction({ isLogin: false })); this.store.dispatch(resetUserAction()); this.store.dispatch(resetConversationAction()); this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.success, message: '已退出!' })); }); } // 初始化tim监听函数 initListener() { // sdk ready this.tim.on(TIM$1.EVENT.SDK_READY, this.onReadyStateUpdate, this); // SDK NOT READT this.tim.on(TIM$1.EVENT.SDK_NOT_READY, this.onReadyStateUpdate, this); // 被踢出 this.tim.on(TIM$1.EVENT.KICKED_OUT, this.onKickOut, this); // SDK内部出错 this.tim.on(TIM$1.EVENT.ERROR, this.onError, this); // 收到新消息 this.tim.on(TIM$1.EVENT.MESSAGE_RECEIVED, this.onReceiveMessage, this); // 会话列表更新 this.tim.on(TIM$1.EVENT.CONVERSATION_LIST_UPDATED, this.onUpdateConversationList, this); // 群组列表更新 this.tim.on(TIM$1.EVENT.GROUP_LIST_UPDATED, this.onUpdateGroupList, this); // 网络监测 this.tim.on(TIM$1.EVENT.NET_STATE_CHANGE, this.onNetStateChange, this); // 已读回执 this.tim.on(TIM$1.EVENT.MESSAGE_READ_BY_PEER, this.onMessageReadByPeer, this); } onReadyStateUpdate({ name }) { const isSDKReady = name === TIM$1.EVENT.SDK_READY ? true : false; this.store.dispatch(SDKReadyAction({ SDKReadyState: isSDKReady })); if (isSDKReady) { this.tim.getMyProfile() .then(({ data }) => { this.store.dispatch(updateCurrentUserProfileAction({ profile: data })); }) .catch(error => { this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.warning, message: error.message })); }); } } onKickOut(event) { this.eventBus$.next('logout'); // this.store.dispatch(stopComputeCurrentAction()); this.store.dispatch(loginAction({ isLogin: false })); this.store.dispatch(resetUserAction()); this.store.dispatch(resetConversationAction()); this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.warning, message: '由于多实例登录被踢出,请重新登录!' })); } onError({ data }) { if (data.message !== 'Network Error') { console.log('%c error', 'color:red;font-size:20px;', data); } } onMessageReadByPeer(event) { console.log('已回执', event); } onReceiveMessage({ data: messageList }) { // this.handleVideoMessage(messageList); // this.handleAt(messageList); // this.handleQuitGroupTip(messageList); this.store.dispatch(pushCurrentMessageListAction({ message: messageList })); } // 会话列表更新 onUpdateConversationList(event) { this.store.dispatch(updateConversationListAction({ conversationList: event.data })); } // 群列表更新 onUpdateGroupList(event) { this.store.dispatch(updateGroupListAction({ groupList: event.data })); } onNetStateChange(event) { console.log('网络监测::', event); } /** * 切换会话 * 调用时机:切换会话时 */ checkoutConversation(conversationID) { this.store.dispatch(resetCurrentMemberListAction()); // 1.切换会话前,将切换前的会话进行已读上报 if (this.conversation.currentConversation.conversationID) { const prevConversationID = this.conversation.currentConversation.conversationID; this.tim.setMessageRead({ conversationID: prevConversationID }); } // 2.待切换的会话也进行已读上报 this.tim.setMessageRead({ conversationID }); // 3. 获取会话信息 return this.tim.getConversationProfile(conversationID).then((res) => { // 3.1 更新当前会话 this.store.dispatch(updateCurrentConversationAction({ conversation: res.data.conversation })); // 3.2 获取消息列表 this.getMessageList(conversationID); if (res.data.conversation.type === TIM$1.TYPES.CONV_GROUP) { this.getGroupMemberList(res.data.conversation.groupProfile.groupID); } return Promise.resolve(); }); // .catch(err => { // this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.error, message: err })); // }); } /** * @description 获取消息 */ getMessageList(conversationID) { if (this.conversation.isCompleted) { this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.info, message: '已经没有更多的历史消息了哦' })); return; } const { nextReqMessageID, currentMessageList } = this.conversation; this.tim.getMessageList({ conversationID, nextReqMessageID, count: 15 }) .then((imResponse) => { this.store.dispatch(updateMessageAction({ nextReqMessageID: imResponse.data.nextReqMessageID, isCompleted: imResponse.data.isCompleted, currentMessageList: [...imResponse.data.messageList, ...currentMessageList] })); }); } /** * @description 获取群成员 */ getGroupMemberList(groupID) { this.tim.getGroupMemberList({ groupID, offset: this.currentMemberList.length, count: 30 }).then((imResponse) => { this.store.dispatch(updateCurrentMemberListAction({ currentMemberList: imResponse.data.memberList })); }); } initTim(config) { // const cosImport = await import('cos-js-sdk-v5'); // const timImport = await import('tim-js-sdk'); this.tim = TIM$1.create({ SDKAppID: config.sdkAppId, oversea: config.oversea, }); // 无日志级别 this.tim.setLogLevel(config.level || 1); // 注册 cos this.tim.registerPlugin({ 'tim-upload-plugin': TIMUploadPlugin }); } } TimHelperService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: TimHelperService, deps: [{ token: i1.Store }, { token: NG_Tim_CONFIG }], target: i0.ɵɵFactoryTarget.Injectable }); TimHelperService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: TimHelperService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: TimHelperService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: function () { return [{ type: i1.Store }, { type: undefined, decorators: [{ type: Inject, args: [NG_Tim_CONFIG] }] }]; } }); function titleNotify(count) { const hasNewMessage = count > 0; if (hasNewMessage) { if (document.title.search(/\((.*?)\)/) >= 0) { document.title = document.title.replace(/\((.*?)\)/, `(${count > 99 ? '99+' : count})`); } else { document.title = `(${count})${document.title}`; } } else { document.title = document.title.replace(/\((.*?)\)/, ''); } } class EditProfileComponent { constructor(fb, modalRef, store, timHelperService) { this.fb = fb; this.modalRef = modalRef; this.store = store; this.timHelperService = timHelperService; this.radioOption = [ { label: '男', value: TIM.TYPES.GENDER_MALE }, { label: '女', value: TIM.TYPES.GENDER_FEMALE }, { label: '不显示', value: TIM.TYPES.GENDER_UNKNOWN }, ]; } ngOnInit() { this.form = this.fb.group({ avatar: [this.userProfile.avatar], nick: [this.userProfile.nick], gender: [this.userProfile.gender] }); } submitForm() { const formValue = this.form.value; if (formValue.avatar && formValue.avatar.indexOf('http') === -1) { this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.warning, message: '头像应该是 Url 地址' })); formValue.avatar = ''; return; } const options = {}; // 过滤空串 Object.keys(formValue).forEach(key => { if (formValue[key]) { options[key] = formValue[key]; } }); this.timHelperService.tim .updateMyProfile(options) .then(() => { this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.success, message: '修改成功' })); this.modalRef.destroy(); }) .catch(imError => { this.store.dispatch(showAction({ msgType: MESSAGE_STATUS.error, message: imError.message })); }); } } EditProfileComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: EditProfileComponent, deps: [{ token: i1$1.FormBuilder }, { token: i2.NzModalRef }, { token: i1.Store }, { token: TimHelperService }], target: i0.ɵɵFactoryTarget.Component }); EditProfileComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: EditProfileComponent, selector: "im-edit-profile", inputs: { userProfile: "userProfile" }, ngImport: i0, template: "<form nz-form [formGroup]=\"form\" (ngSubmit)=\"submitForm()\">\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\">\r\n <span>\u5934\u50CF</span>\r\n </nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <input nz-input formControlName=\"avatar\" placeholder=\"\u5934\u50CF\u5730\u5740\" />\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\" nzFor=\"avatar\">\u6635\u79F0</nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <input nz-input formControlName=\"nick\" placeholder=\"\u6635\u79F0\" />\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\">\u6027\u522B\r\n </nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <nz-radio-group formControlName=\"gender\">\r\n <label *ngFor=\"let item of radioOption\" nz-radio\r\n [nzValue]=\"item.value\">{{item.label}}</label>\r\n </nz-radio-group>\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n\r\n <nz-form-item nz-row nzJustify=\"center\">\r\n <nz-form-control [nzSpan]=\"12\" nzPush=\"8\">\r\n <button nz-button nzType=\"primary\">\u786E\u5B9A</button>\r\n </nz-form-control>\r\n <nz-form-control [nzSpan]=\"12\">\r\n <button nz-button nzType=\"default\" type=\"button\" (click)=\"modalRef.destroy()\">\u53D6\u6D88</button>\r\n </nz-form-control>\r\n </nz-form-item>\r\n</form>\r\n", styles: [""], components: [{ type: i5.NzFormItemComponent, selector: "nz-form-item", exportAs: ["nzFormItem"] }, { type: i5.NzFormLabelComponent, selector: "nz-form-label", inputs: ["nzFor", "nzRequired", "nzNoColon", "nzTooltipTitle", "nzTooltipIcon"], exportAs: ["nzFormLabel"] }, { type: i5.NzFormControlComponent, selector: "nz-form-control", inputs: ["nzSuccessTip", "nzWarningTip", "nzErrorTip", "nzValidatingTip", "nzExtra", "nzAutoTips", "nzDisableAutoTips", "nzHasFeedback", "nzValidateStatus"], exportAs: ["nzFormControl"] }, { type: i6.NzRadioGroupComponent, selector: "nz-radio-group", inputs: ["nzDisabled", "nzButtonStyle", "nzSize", "nzName"], exportAs: ["nzRadioGroup"] }, { type: i6.NzRadioComponent, selector: "[nz-radio],[nz-radio-button]", inputs: ["nzValue", "nzDisabled", "nzAutoFocus"], exportAs: ["nzRadio"] }, { type: i7.NzButtonComponent, selector: "button[nz-button], a[nz-button]", inputs: ["nzBlock", "nzGhost", "nzSearch", "nzLoading", "nzDanger", "disabled", "tabIndex", "nzType", "nzShape", "nzSize"], exportAs: ["nzButton"] }], directives: [{ type: i1$1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i1$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i5.NzFormDirective, selector: "[nz-form]", inputs: ["nzLayout", "nzNoColon", "nzAutoTips", "nzDisableAutoTips", "nzTooltipIcon"], exportAs: ["nzForm"] }, { type: i1$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i8.NzRowDirective, selector: "[nz-row],nz-row,nz-form-item", inputs: ["nzAlign", "nzJustify", "nzGutter"], exportAs: ["nzRow"] }, { type: i8.NzColDirective, selector: "[nz-col],nz-col,nz-form-control,nz-form-label", inputs: ["nzFlex", "nzSpan", "nzOrder", "nzOffset", "nzPush", "nzPull", "nzXs", "nzSm", "nzMd", "nzLg", "nzXl", "nzXXl"], exportAs: ["nzCol"] }, { type: i9.NzInputDirective, selector: "input[nz-input],textarea[nz-input]", inputs: ["nzBorderless", "nzSize", "disabled"], exportAs: ["nzInput"] }, { type: i1$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i1$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { type: i5$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i10.NzWaveDirective, selector: "[nz-wave],button[nz-button]:not([nzType=\"link\"]):not([nzType=\"text\"])", inputs: ["nzWaveExtraNode"], exportAs: ["nzWave"] }, { type: i6$1.ɵNzTransitionPatchDirective, selector: "[nz-button], nz-button-group, [nz-icon], [nz-menu-item], [nz-submenu], nz-select-top-control, nz-select-placeholder, nz-input-group", inputs: ["hidden"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: EditProfileComponent, decorators: [{ type: Component, args: [{ selector: 'im-edit-profile', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form nz-form [formGroup]=\"form\" (ngSubmit)=\"submitForm()\">\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\">\r\n <span>\u5934\u50CF</span>\r\n </nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <input nz-input formControlName=\"avatar\" placeholder=\"\u5934\u50CF\u5730\u5740\" />\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\" nzFor=\"avatar\">\u6635\u79F0</nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <input nz-input formControlName=\"nick\" placeholder=\"\u6635\u79F0\" />\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n <nz-form-item>\r\n <nz-form-label [nzSm]=\"6\" [nzXs]=\"24\">\u6027\u522B\r\n </nz-form-label>\r\n <nz-form-control [nzSm]=\"14\" [nzXs]=\"24\">\r\n <nz-radio-group formControlName=\"gender\">\r\n <label *ngFor=\"let item of radioOption\" nz-radio\r\n [nzValue]=\"item.value\">{{item.label}}</label>\r\n </nz-radio-group>\r\n </nz-form-control>\r\n </nz-form-item>\r\n\r\n\r\n <nz-form-item nz-row nzJustify=\"center\">\r\n <nz-form-control [nzSpan]=\"12\" nzPush=\"8\">\r\n <button nz-button nzType=\"primary\">\u786E\u5B9A</button>\r\n </nz-form-control>\r\n <nz-form-control [nzSpan]=\"12\">\r\n <button nz-button nzType=\"default\" type=\"button\" (click)=\"modalRef.destroy()\">\u53D6\u6D88</button>\r\n </nz-form-control>\r\n </nz-form-item>\r\n</form>\r\n", styles: [""] }] }], ctorParameters: function () { return [{ type: i1$1.FormBuilder }, { type: i2.NzModalRef }, { type: i1.Store }, { type: TimHelperService }]; }, propDecorators: { userProfile: [{ type: Input }] } }); class AvatarComponent { constructor() { this.shape = 'circle'; this.size = 'large'; this.type = 'C2C'; } set src(value) { this._src = value; if (/^(https:|http:|\/\/)/.test(value)) { this.avatarSrc = value; } else { this.avatarSrc = this.getDefaultAvatar(); } } ; get src() { return this._src; } ngOnInit() { } getDefaultAvatar() { switch (this.type) { case 'C2C': // 个人头像 return 'https://imgcache.qq.com/open/qcloud/video/act/webim-avatar/avatar-2.png'; case 'GROUP': // 群默认头像 return 'https://imgcache.qq.com/open/qcloud/video/act/webim-avatar/avatar-3.png'; case TIM.TYPES.CONV_SYSTEM: return ''; default: // 默认头像 return 'https://imgcache.qq.com/open/qcloud/video/act/webim-avatar/avatar-1.png'; } } } AvatarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: AvatarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: AvatarComponent, selector: "im-avatar", inputs: { shape: "shape", size: "size", type: "type", title: "title", src: "src" }, ngImport: i0, template: "<nz-avatar [nzSize]=\"size\" [nzShape]=\"shape\" [nzSrc]=\"avatarSrc\" [nzAlt]=\"title\" [title]=\"title\"></nz-avatar>\r\n", styles: [""], components: [{ type: i1$2.NzAvatarComponent, selector: "nz-avatar", inputs: ["nzShape", "nzSize", "nzGap", "nzText", "nzSrc", "nzSrcSet", "nzAlt", "nzIcon"], outputs: ["nzError"], exportAs: ["nzAvatar"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: AvatarComponent, decorators: [{ type: Component, args: [{ selector: 'im-avatar', template: "<nz-avatar [nzSize]=\"size\" [nzShape]=\"shape\" [nzSrc]=\"avatarSrc\" [nzAlt]=\"title\" [title]=\"title\"></nz-avatar>\r\n", styles: [""] }] }], ctorParameters: function () { return []; }, propDecorators: { shape: [{ type: Input }], size: [{ type: Input }], type: [{ type: Input }], title: [{ type: Input }], src: [{ type: Input }] } }); class ProfileCardComponent { constructor() { } set userProfile(value) { this._userProfile = value; switch (value.gender) { case TIM.TYPES.GENDER_MALE: this.className = 'icon-male'; break; case TIM.TYPES.GENDER_FEMALE: this.className = 'icon-female'; break; default: this.className = null; } } ; get userProfile() { return this._userProfile; } ; ngOnInit() { } } ProfileCardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: ProfileCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); ProfileCardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: ProfileCardComponent, selector: "im-profile-card", inputs: { userProfile: "userProfile" }, ngImport: i0, template: "<div class=\"profile-card-wrapper\">\r\n <div class=\"content\">\r\n <im-avatar [src]=\"userProfile?.avatar\"></im-avatar>\r\n <div class=\"basic\">\r\n <span class=\"nick text-ellipsis\">{{ userProfile?.nick || userProfile?.userID }}</span>\r\n <span class=\"iconfont\" [class]=\"className\"></span>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".profile-header{display:flex;margin-bottom:12px}.profile-card-wrapper .content{display:flex;justify-content:center;flex-direction:column;align-items:center}.basic{display:flex;align-items:center;margin-top:12px}.icon-male{color:#6391f3}.icon-female{color:#ff8096}.nick{font-size:18px;margin-right:8px;max-width:100px;display:inline-block}im-avatar::ng-deep .avatar{width:70px;height:70px}\n"], components: [{ type: AvatarComponent, selector: "im-avatar", inputs: ["shape", "size", "type", "title", "src"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: ProfileCardComponent, decorators: [{ type: Component, args: [{ selector: 'im-profile-card', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"profile-card-wrapper\">\r\n <div class=\"content\">\r\n <im-avatar [src]=\"userProfile?.avatar\"></im-avatar>\r\n <div class=\"basic\">\r\n <span class=\"nick text-ellipsis\">{{ userProfile?.nick || userProfile?.userID }}</span>\r\n <span class=\"iconfont\" [class]=\"className\"></span>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".profile-header{display:flex;margin-bottom:12px}.profile-card-wrapper .content{display:flex;justify-content:center;flex-direction:column;align-items:center}.basic{display:flex;align-items:center;margin-top:12px}.icon-male{color:#6391f3}.icon-female{color:#ff8096}.nick{font-size:18px;margin-right:8px;max-width:100px;display:inline-block}im-avatar::ng-deep .avatar{width:70px;height:70px}\n"] }] }], ctorParameters: function () { return []; }, propDecorators: { userProfile: [{ type: Input }] } }); class MyProfileComponent { constructor(store, modal) { this.store = store; this.modal = modal; } ngOnInit() { this.subscription = this.store.select(currentUserProfileSelector) .subscribe(res => { if (res) { this.currentUserProfile = res; } }); } editProfile() { this.modal.create({ nzTitle: `编辑资料`, nzContent: EditProfileComponent, nzMaskClosable: false, nzFooter: null, nzWidth: '40%', nzStyle: { top: '20px' }, nzComponentParams: { userProfile: this.currentUserProfile } }); } ngOnDestroy() { if (this.subscription) { this.subscription.unsubscribe(); } } } MyProfileComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: MyProfileComponent, deps: [{ token: i1.Store }, { token: i2.NzModalService }], target: i0.ɵɵFactoryTarget.Component }); MyProfileComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: MyProfileComponent, selector: "app-my-profile", ngImport: i0, template: "<ng-template #titleTemplate>\r\n <div class=\"title-container\">\r\n <span>\u5F53\u524D\u7528\u6237</span>\r\n <i nz-icon nzType=\"setting\" nzTheme=\"outline\" (click)=\"editProfile()\"></i>\r\n </div>\r\n</ng-template>\r\n\r\n<ng-template #contentTemplate>\r\n <im-profile-card [userProfile]=\"currentUserProfile\"></im-profile-card>\r\n</ng-template>\r\n\r\n<div class=\"my-profile-wrapper\">\r\n <div nz-button nz-popover nzType=\"primary\" nzPopoverTrigger=\"click\"\r\n [nzPopoverTitle]=\"titleTemplate\" nzPopoverPlacement=\"right\"\r\n [nzPopoverContent]=\"contentTemplate\">\r\n <im-avatar [src]=\"currentUserProfile.avatar\"></im-avatar>\r\n </div>\r\n</div>\r\n", styles: [".title-container{display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between;align-items:center}.title-container i{cursor:pointer}.my-profile-wrapper{margin:15px}.my-profile-wrapper im-avatar{cursor:pointer}.edit-my-profile{position:absolute;top:10px;right:10px;cursor:pointer}\n"], components: [{ type: ProfileCardComponent, selector: "im-profile-card", inputs: ["userProfile"] }, { type: AvatarComponent, selector: "im-avatar", inputs: ["shape", "size", "type", "title", "src"] }], directives: [{ type: i6$1.ɵNzTransitionPatchDirective, selector: "[nz-button], nz-button-group, [nz-icon], [nz-menu-item], [nz-submenu], nz-select-top-control, nz-select-placeholder, nz-input-group", inputs: ["hidden"] }, { type: i6$2.NzIconDirective, selector: "[nz-icon]", inputs: ["nzSpin", "nzRotate", "nzType", "nzTheme", "nzTwotoneColor", "nzIconfont"], exportAs: ["nzIcon"] }, { type: i7$1.NzPopoverDirective, selector: "[nz-popover]", inputs: ["nzPopoverArrowPointAtCenter", "nzPopoverTitle", "nzPopoverContent", "nz-popover", "nzPopoverTrigger", "nzPopoverPlacement", "nzPopoverOrigin", "nzPopoverVisible", "nzPopoverMouseEnterDelay", "nzPopoverMouseLeaveDelay", "nzPopoverOverlayClassName", "nzPopoverOverlayStyle", "nzPopoverBackdrop"], outputs: ["nzPopoverVisibleChange"], exportAs: ["nzPopover"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: MyProfileComponent, decorators: [{ type: Component, args: [{ selector: 'app-my-profile', template: "<ng-template #titleTemplate>\r\n <div class=\"title-container\">\r\n <span>\u5F53\u524D\u7528\u6237</span>\r\n <i nz-icon nzType=\"setting\" nzTheme=\"outline\" (click)=\"editProfile()\"></i>\r\n </div>\r\n</ng-template>\r\n\r\n<ng-template #contentTemplate>\r\n <im-profile-card [userProfile]=\"currentUserProfile\"></im-profile-card>\r\n</ng-template>\r\n\r\n<div class=\"my-profile-wrapper\">\r\n <div nz-button nz-popover nzType=\"primary\" nzPopoverTrigger=\"click\"\r\n [nzPopoverTitle]=\"titleTemplate\" nzPopoverPlacement=\"right\"\r\n [nzPopoverContent]=\"contentTemplate\">\r\n <im-avatar [src]=\"currentUserProfile.avatar\"></im-avatar>\r\n </div>\r\n</div>\r\n", styles: [".title-container{display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between;align-items:center}.title-container i{cursor:pointer}.my-profile-wrapper{margin:15px}.my-profile-wrapper im-avatar{cursor:pointer}.edit-my-profile{position:absolute;top:10px;right:10px;cursor:pointer}\n"] }] }], ctorParameters: function () { return [{ type: i1.Store }, { type: i2.NzModalService }]; } }); /** * 返回年月日 * @export * @param {Date} date * @param {string} [splitor='-'] * @returns */ function getDate(date, splitor = '-') { const year = date.getFullYear(); const month = date.getMonth() + 1; const day = date.getDate(); return `${year}${splitor}${addZeroPrefix(month)}${splitor}${addZeroPrefix(day)}`; } /** * 返回时分秒/时分 * @export * @param {*} date * @param {boolean} [withSecond=false] * @returns */ function getTime(date, withSecond = false) { const hour = date.getHours(); const minute = date.getMinutes(); const second = date.getSeconds(); return withSecond ? `${addZeroPrefix(hour)}:${addZeroPrefix(minute)}:${addZeroPrefix(second)}` : `${hour}:${addZeroPrefix(minute)}`; } function getFullDate(date) { return `${getDate(date)} ${getTime(date)}`; } function isToday(date) { return date.toDateString() === new Date().toDateString(); } /** * 个位数,加0前缀 * @param {*} number * @returns */ function addZeroPrefix(number) { return number < 10 ? `0${number}` : number; } class ConversationItemComponent { constructor(store, timHelperService) { this.store = store; this.timHelperService = timHelperService; this.TIM = TIM$1; } ngOnInit() { this.profileSubscription = this.store.select(currentUserProfileSelector) .subscribe(res => { this.currentUserProfile = res; }); } selectConversation() { if (this.conversation.conversationID !== this.currentConversation.conversationID) { this.timHelperService.checkoutConversation(this.conversation.conversationID); // this.timHelperService.eventBus$.next('select-item'); } } get avatarSrc() { switch (this.conversation.type) { case 'GROUP': return this.conversation.groupProfile?.avatar; case 'C2C': return this.conversation.userProfile?.avatar; default: return null; } } ; get date() { if (!this.conversation.lastMessage || !this.conversation.lastMessage.lastTime) { return ''; } const date = new Date(this.conversation.lastMessage.lastTime * 1000); if (isToday(date)) { return getTime(date); } return getDate(date); } get messageForShow() { if (this.conversation.lastMessage.isRevoked) { if (this.conversation.lastMessage.fromAccount === this.currentUserProfile?.userID) { return '你撤回了一条消息'; } if (this.conversation.type === TIM$1.TYPES.CONV_C2C) { return '对方撤回了一条消息'; } return `${this.conversation.lastMessage.fromAccount}撤回了一条消息`; } return this.conversation.lastMessage.messageForShow; } get conversationName() { if (this.conversation.type === TIM$1.TYPES.CONV_C2C) { return this.conversation.userProfile.nick || this.conversation.userProfile.userID; } if (this.conversation.type === TIM$1.TYPES.CONV_GROUP) { return this.conversation.groupProfile.name || this.conversation.groupProfile.groupID; } if (this.conversation.type === TIM$1.TYPES.CONV_SYSTEM) { return '系统通知'; } return ''; } deleteConversation(event) { // 停止冒泡,避免和点击会话的事件冲突 event.stopPropagation(); this.timHelperService.tim .deleteConversation(this.conversation.conversationID) .then(() => { this.store.dispatch(showAction({ message: `会话【${this.conversationName}】删除成功!`, msgType: MESSAGE_STATUS.success })); this.store.dispatch(resetCurrentConversationAction()); }) .catch(error => { this.store.dispatch(showAction({ message: `会话【${this.conversationName}】删除失败!, error=${error.message}`, msgType: MESSAGE_STATUS.error })); }); } ngOnDestroy() { if (this.profileSubscription) { this.profileSubscription.unsubscribe(); } } } ConversationItemComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: ConversationItemComponent, deps: [{ token: i1.Store }, { token: TimHelperService }], target: i0.ɵɵFactoryTarget.Component }); ConversationItemComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: ConversationItemComponent, selector: "app-conversation-item", inputs: { currentConversation: "currentConversation", conversation: "conversation" }, ngImport: i0, template: "<div class=\"conversation-item-container\"\r\n [ngClass]=\"{'choose': conversation.conversationID === currentConversation.conversationID }\"\r\n (click)=\"selectConversation()\">\r\n <div class=\"close-btn\">\r\n <span class=\"tim-icon-close\" title=\"\u5220\u9664\u4F1A\u8BDD\" (click)=\"deleteConversation($event)\"></span>\r\n </div>\r\n <div class=\"warp\">\r\n <im-avatar [src]=\"avatarSrc\" [type]=\"conversation.type\"></im-avatar>\r\n <div class=\"content\">\r\n <div class=\"row-1\">\r\n <div class=\"name\">\r\n <div class=\"text-ellipsis\">\r\n <span [title]=\"conversation.userProfile.nick || conversation.userProfile.userID\"\r\n *ngIf=\"conversation.type === TIM.TYPES.CONV_C2C\">\r\n {{conversation.userProfile.nick || conversation.userProfile.userID}}\r\n </span>\r\n <span [title]=\"conversation.groupProfile.name || conversation.groupProfile.groupID\"\r\n *ngIf=\"conversation.type===TIM.TYPES.CONV_GROUP\">\r\n {{conversation.groupProfile.name || conversation.groupProfile.groupID}}\r\n </span>\r\n <span *ngIf=\"conversation.type === TIM.TYPES.CONV_SYSTEM\">\u7CFB\u7EDF\u901A\u77E5\r\n </span>\r\n </div>\r\n </div>\r\n <div class=\"unread-count\">\r\n <span class=\"badge\" *ngIf=\"this.conversation.unreadCount > 0\">\r\n {{conversation.unreadCount > 99 ? '99+' : conversation.unreadCount}}\r\n </span>\r\n </div>\r\n </div>\r\n <div class=\"row-2\">\r\n <div class=\"summary\">\r\n <div v-if=\"conversation.lastMessage\" class=\"text-ellipsis\">\r\n <!-- <span class=\"remind\" style=\"color:red;\">[\u6709\u4EBA\u63D0\u5230\u6211]</span> -->\r\n <span class=\"text\" [title]=\"conversation.lastMessage.messageForShow\">\r\n {{messageForShow}}\r\n </span>\r\n </div>\r\n </div>\r\n <div class=\"date\">\r\n {{date}}\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".conversation-item-container{padding:15px 10px;cursor:pointer;position:relative;overflow:hidden;transition:.2s}.conversation-item-container:hover{background-color:#404953}.conversation-item-container:hover .close-btn{right:3px}.conversation-item-container .close-btn{position:absolute;right:-20px;top:3px;color:#76828c;transition:all .2s ease}.conversation-item-container .close-btn:hover{color:#f35f5f}.conversation-item-container .warp{display:flex}.conversation-item-container .warp im-avatar{padding-right:10px}.conversation-item-container .content{flex:1;height:40px;overflow:hidden}.conversation-item-container .content .row-1{display:flex;line-height:21px}.conversation-item-container .content .row-1 .name{color:#f7f7f8;flex:1;min-width:0px}.conversation-item-container .content .row-1 .unread-count{padding-left:10px;flex-shrink:0;color:#76828c;font-size:12px}.conversation-item-container .content .row-1 .unread-count .badge{vertical-align:bottom;background-color:#f35f5f;border-radius:10px;color:#fff;display:inl