Compare commits
	
		
			17 Commits
		
	
	
		
			9360ecaaf9
			...
			3eaac91ba8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3eaac91ba8 | |||
|  | b04d25a243 | ||
|  | 9e31271cc3 | ||
|  | 6d08dbe42f | ||
|  | 478336c2fe | ||
|  | 419bde4db2 | ||
|  | fca127b42b | ||
|  | 94cf0f9f63 | ||
|  | fad84e5bf3 | ||
|  | 661472a70a | ||
|  | 651baafd0f | ||
|  | c9794c3f25 | ||
|  | 067312cd5c | ||
|  | 51a406e5e5 | ||
|  | fed311c76e | ||
|  | b84430a7e3 | ||
|  | d413a6b9fe | 
							
								
								
									
										75
									
								
								auto-imports.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,75 @@ | ||||
| /* eslint-disable */ | ||||
| /* prettier-ignore */ | ||||
| // @ts-nocheck
 | ||||
| // noinspection JSUnusedGlobalSymbols
 | ||||
| // Generated by unplugin-auto-import
 | ||||
| // biome-ignore lint: disable
 | ||||
| export {} | ||||
| declare global { | ||||
|   const EffectScope: typeof import('vue')['EffectScope'] | ||||
|   const computed: typeof import('vue')['computed'] | ||||
|   const createApp: typeof import('vue')['createApp'] | ||||
|   const customRef: typeof import('vue')['customRef'] | ||||
|   const defineAsyncComponent: typeof import('vue')['defineAsyncComponent'] | ||||
|   const defineComponent: typeof import('vue')['defineComponent'] | ||||
|   const effectScope: typeof import('vue')['effectScope'] | ||||
|   const getCurrentInstance: typeof import('vue')['getCurrentInstance'] | ||||
|   const getCurrentScope: typeof import('vue')['getCurrentScope'] | ||||
|   const h: typeof import('vue')['h'] | ||||
|   const inject: typeof import('vue')['inject'] | ||||
|   const isProxy: typeof import('vue')['isProxy'] | ||||
|   const isReactive: typeof import('vue')['isReactive'] | ||||
|   const isReadonly: typeof import('vue')['isReadonly'] | ||||
|   const isRef: typeof import('vue')['isRef'] | ||||
|   const markRaw: typeof import('vue')['markRaw'] | ||||
|   const nextTick: typeof import('vue')['nextTick'] | ||||
|   const onActivated: typeof import('vue')['onActivated'] | ||||
|   const onBeforeMount: typeof import('vue')['onBeforeMount'] | ||||
|   const onBeforeUnmount: typeof import('vue')['onBeforeUnmount'] | ||||
|   const onBeforeUpdate: typeof import('vue')['onBeforeUpdate'] | ||||
|   const onDeactivated: typeof import('vue')['onDeactivated'] | ||||
|   const onErrorCaptured: typeof import('vue')['onErrorCaptured'] | ||||
|   const onMounted: typeof import('vue')['onMounted'] | ||||
|   const onRenderTracked: typeof import('vue')['onRenderTracked'] | ||||
|   const onRenderTriggered: typeof import('vue')['onRenderTriggered'] | ||||
|   const onScopeDispose: typeof import('vue')['onScopeDispose'] | ||||
|   const onServerPrefetch: typeof import('vue')['onServerPrefetch'] | ||||
|   const onUnmounted: typeof import('vue')['onUnmounted'] | ||||
|   const onUpdated: typeof import('vue')['onUpdated'] | ||||
|   const onWatcherCleanup: typeof import('vue')['onWatcherCleanup'] | ||||
|   const provide: typeof import('vue')['provide'] | ||||
|   const reactive: typeof import('vue')['reactive'] | ||||
|   const readonly: typeof import('vue')['readonly'] | ||||
|   const ref: typeof import('vue')['ref'] | ||||
|   const resolveComponent: typeof import('vue')['resolveComponent'] | ||||
|   const shallowReactive: typeof import('vue')['shallowReactive'] | ||||
|   const shallowReadonly: typeof import('vue')['shallowReadonly'] | ||||
|   const shallowRef: typeof import('vue')['shallowRef'] | ||||
|   const toRaw: typeof import('vue')['toRaw'] | ||||
|   const toRef: typeof import('vue')['toRef'] | ||||
|   const toRefs: typeof import('vue')['toRefs'] | ||||
|   const toValue: typeof import('vue')['toValue'] | ||||
|   const triggerRef: typeof import('vue')['triggerRef'] | ||||
|   const unref: typeof import('vue')['unref'] | ||||
|   const useAttrs: typeof import('vue')['useAttrs'] | ||||
|   const useCssModule: typeof import('vue')['useCssModule'] | ||||
|   const useCssVars: typeof import('vue')['useCssVars'] | ||||
|   const useDialog: typeof import('naive-ui')['useDialog'] | ||||
|   const useId: typeof import('vue')['useId'] | ||||
|   const useLoadingBar: typeof import('naive-ui')['useLoadingBar'] | ||||
|   const useMessage: typeof import('naive-ui')['useMessage'] | ||||
|   const useModel: typeof import('vue')['useModel'] | ||||
|   const useNotification: typeof import('naive-ui')['useNotification'] | ||||
|   const useSlots: typeof import('vue')['useSlots'] | ||||
|   const useTemplateRef: typeof import('vue')['useTemplateRef'] | ||||
|   const watch: typeof import('vue')['watch'] | ||||
|   const watchEffect: typeof import('vue')['watchEffect'] | ||||
|   const watchPostEffect: typeof import('vue')['watchPostEffect'] | ||||
|   const watchSyncEffect: typeof import('vue')['watchSyncEffect'] | ||||
| } | ||||
| // for type re-export
 | ||||
| declare global { | ||||
|   // @ts-ignore
 | ||||
|   export type { Component, Slot, Slots, ComponentPublicInstance, ComputedRef, DirectiveBinding, ExtractDefaultPropTypes, ExtractPropTypes, ExtractPublicPropTypes, InjectionKey, PropType, Ref, MaybeRef, MaybeRefOrGetter, VNode, WritableComputedRef } from 'vue' | ||||
|   import('vue') | ||||
| } | ||||
							
								
								
									
										96
									
								
								components.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,96 @@ | ||||
| /* eslint-disable */ | ||||
| // @ts-nocheck
 | ||||
| // Generated by unplugin-vue-components
 | ||||
| // Read more: https://github.com/vuejs/core/pull/3399
 | ||||
| // biome-ignore lint: disable
 | ||||
| export {} | ||||
| 
 | ||||
| /* prettier-ignore */ | ||||
| declare module 'vue' { | ||||
|   export interface GlobalComponents { | ||||
|     ApplyTab: typeof import('./src/components/group/manage/ApplyTab.vue')['default'] | ||||
|     AudioMessage: typeof import('./src/components/talk/message/AudioMessage.vue')['default'] | ||||
|     Avatar: typeof import('./src/components/base/Avatar.vue')['default'] | ||||
|     AvatarCropper: typeof import('./src/components/base/AvatarCropper.vue')['default'] | ||||
|     AvatarModule: typeof import('./src/components/avatar-module/index.vue')['default'] | ||||
|     CodeMessage: typeof import('./src/components/talk/message/CodeMessage.vue')['default'] | ||||
|     ConfigTab: typeof import('./src/components/group/manage/ConfigTab.vue')['default'] | ||||
|     ContactModal: typeof import('./src/components/user/ContactModal.vue')['default'] | ||||
|     CustomBtn: typeof import('./src/components/common/customBtn.vue')['default'] | ||||
|     CustomModal: typeof import('./src/components/common/customModal.vue')['default'] | ||||
|     DetailTab: typeof import('./src/components/group/manage/DetailTab.vue')['default'] | ||||
|     DialogApi: typeof import('./src/components/common/DialogApi.vue')['default'] | ||||
|     Editor: typeof import('./src/components/editor/Editor.vue')['default'] | ||||
|     EditorEmail: typeof import('./src/components/user/EditorEmail.vue')['default'] | ||||
|     EditorMobile: typeof import('./src/components/user/EditorMobile.vue')['default'] | ||||
|     EditorPassword: typeof import('./src/components/user/EditorPassword.vue')['default'] | ||||
|     FileMessage: typeof import('./src/components/talk/message/FileMessage.vue')['default'] | ||||
|     Flnindex: typeof import('./src/components/flnlayout/tree/flnindex.vue')['default'] | ||||
|     ForwardMessage: typeof import('./src/components/talk/message/ForwardMessage.vue')['default'] | ||||
|     ForwardRecord: typeof import('./src/components/talk/ForwardRecord.vue')['default'] | ||||
|     GroupApply: typeof import('./src/components/group/GroupApply.vue')['default'] | ||||
|     GroupLaunch: typeof import('./src/components/group/GroupLaunch.vue')['default'] | ||||
|     GroupNotice: typeof import('./src/components/group/GroupNotice.vue')['default'] | ||||
|     GroupNoticeMessage: typeof import('./src/components/talk/message/GroupNoticeMessage.vue')['default'] | ||||
|     GroupPanel: typeof import('./src/components/group/GroupPanel.vue')['default'] | ||||
|     HistoryRecord: typeof import('./src/components/talk/HistoryRecord.vue')['default'] | ||||
|     ImageMessage: typeof import('./src/components/talk/message/ImageMessage.vue')['default'] | ||||
|     LinkMessage: typeof import('./src/components/talk/message/LinkMessage.vue')['default'] | ||||
|     Loading: typeof import('./src/components/base/Loading.vue')['default'] | ||||
|     LoginMessage: typeof import('./src/components/talk/message/LoginMessage.vue')['default'] | ||||
|     Manage: typeof import('./src/components/group/manage/index.vue')['default'] | ||||
|     MeEditorCode: typeof import('./src/components/editor/MeEditorCode.vue')['default'] | ||||
|     MeEditorEmoticon: typeof import('./src/components/editor/MeEditorEmoticon.vue')['default'] | ||||
|     MeEditorImage: typeof import('./src/components/editor/MeEditorImage.vue')['default'] | ||||
|     MeEditorLocation: typeof import('./src/components/editor/MeEditorLocation.vue')['default'] | ||||
|     MeEditorRecorder: typeof import('./src/components/editor/MeEditorRecorder.vue')['default'] | ||||
|     MeEditorVote: typeof import('./src/components/editor/MeEditorVote.vue')['default'] | ||||
|     MemberTab: typeof import('./src/components/group/manage/MemberTab.vue')['default'] | ||||
|     MessageApi: typeof import('./src/components/common/MessageApi.vue')['default'] | ||||
|     MixedMessage: typeof import('./src/components/talk/message/MixedMessage.vue')['default'] | ||||
|     NAvatar: typeof import('naive-ui')['NAvatar'] | ||||
|     NButton: typeof import('naive-ui')['NButton'] | ||||
|     NCheckbox: typeof import('naive-ui')['NCheckbox'] | ||||
|     NEmpty: typeof import('naive-ui')['NEmpty'] | ||||
|     NIcon: typeof import('naive-ui')['NIcon'] | ||||
|     NImage: typeof import('naive-ui')['NImage'] | ||||
|     NInput: typeof import('naive-ui')['NInput'] | ||||
|     NModal: typeof import('naive-ui')['NModal'] | ||||
|     NoticeEditor: typeof import('./src/components/group/manage/NoticeEditor.vue')['default'] | ||||
|     NoticeTab: typeof import('./src/components/group/manage/NoticeTab.vue')['default'] | ||||
|     NotificationApi: typeof import('./src/components/common/NotificationApi.vue')['default'] | ||||
|     NRadio: typeof import('naive-ui')['NRadio'] | ||||
|     NSpin: typeof import('naive-ui')['NSpin'] | ||||
|     NVirtualList: typeof import('naive-ui')['NVirtualList'] | ||||
|     RevokeMessage: typeof import('./src/components/talk/message/RevokeMessage.vue')['default'] | ||||
|     RouterLink: typeof import('vue-router')['RouterLink'] | ||||
|     RouterView: typeof import('vue-router')['RouterView'] | ||||
|     SysGroupAdminMessage: typeof import('./src/components/talk/message/system/SysGroupAdminMessage.vue')['default'] | ||||
|     SysGroupCancelMutedMessage: typeof import('./src/components/talk/message/system/SysGroupCancelMutedMessage.vue')['default'] | ||||
|     SysGroupCreateMessage: typeof import('./src/components/talk/message/system/SysGroupCreateMessage.vue')['default'] | ||||
|     SysGroupDismissed: typeof import('./src/components/talk/message/system/SysGroupDismissed.vue')['default'] | ||||
|     SysGroupInfoChangeMessage: typeof import('./src/components/talk/message/system/SysGroupInfoChangeMessage.vue')['default'] | ||||
|     SysGroupJoinMessage: typeof import('./src/components/talk/message/system/SysGroupJoinMessage.vue')['default'] | ||||
|     SysGroupMemberCancelMutedMessage: typeof import('./src/components/talk/message/system/SysGroupMemberCancelMutedMessage.vue')['default'] | ||||
|     SysGroupMemberKickedMessage: typeof import('./src/components/talk/message/system/SysGroupMemberKickedMessage.vue')['default'] | ||||
|     SysGroupMemberMutedMessage: typeof import('./src/components/talk/message/system/SysGroupMemberMutedMessage.vue')['default'] | ||||
|     SysGroupMemberQuitMessage: typeof import('./src/components/talk/message/system/SysGroupMemberQuitMessage.vue')['default'] | ||||
|     SysGroupMemberRemovedMessage: typeof import('./src/components/talk/message/system/SysGroupMemberRemovedMessage.vue')['default'] | ||||
|     SysGroupMutedMessage: typeof import('./src/components/talk/message/system/SysGroupMutedMessage.vue')['default'] | ||||
|     SysGroupTransferMessage: typeof import('./src/components/talk/message/system/SysGroupTransferMessage.vue')['default'] | ||||
|     SysTextMessage: typeof import('./src/components/talk/message/system/SysTextMessage.vue')['default'] | ||||
|     TextMessage: typeof import('./src/components/talk/message/TextMessage.vue')['default'] | ||||
|     Treelabel: typeof import('./src/components/flnlayout/tree/treelabel.vue')['default'] | ||||
|     UnknownMessage: typeof import('./src/components/talk/message/UnknownMessage.vue')['default'] | ||||
|     UploadsModal: typeof import('./src/components/base/UploadsModal.vue')['default'] | ||||
|     UserCardModal: typeof import('./src/components/user/UserCardModal.vue')['default'] | ||||
|     VideoMessage: typeof import('./src/components/talk/message/VideoMessage.vue')['default'] | ||||
|     VoteMessage: typeof import('./src/components/talk/message/VoteMessage.vue')['default'] | ||||
|     XAddressSelect: typeof import('./src/components/x-naive-ui/x-address-select/index.vue')['default'] | ||||
|     XNDataTable: typeof import('./src/components/x-naive-ui/x-n-data-table/index.vue')['default'] | ||||
|     XNModal: typeof import('./src/components/x-naive-ui/x-n-modal/index.vue')['default'] | ||||
|     XNUpload: typeof import('./src/components/x-naive-ui/x-n-upload/index.vue')['default'] | ||||
|     XSearchForm: typeof import('./src/components/x-naive-ui/x-search-form/index.vue')['default'] | ||||
|     Xtime: typeof import('./src/components/base/Xtime.vue')['default'] | ||||
|   } | ||||
| } | ||||
							
								
								
									
										6
									
								
								env/.env.test
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -2,9 +2,7 @@ ENV = 'development' | ||||
| 
 | ||||
| VITE_BASE=/ | ||||
| VUE_APP_PREVIEW=false | ||||
| VITE_BASE_API=http://172.16.100.93:8503 | ||||
| # VITE_BASE_API=http://192.168.88.21:9503 | ||||
| VITE_BASE_API=http://114.218.158.24:8503 | ||||
| VITE_EPR_BASEURL=http://114.218.158.24:9020 | ||||
| VITE_SOCKET_API=ws://172.16.100.93:8504 | ||||
| # VITE_SOCKET_API=ws://192.168.88.21:9504 | ||||
| VITE_SOCKET_API=ws://114.218.158.24:8504 | ||||
| VUE_APP_WEBSITE_NAME="Lumen IM" | ||||
| @ -32,6 +32,8 @@ | ||||
|     "quill": "^1.3.7", | ||||
|     "quill-image-uploader": "^1.3.0", | ||||
|     "quill-mention": "^4.1.0", | ||||
|     "sortablejs": "^1.15.6", | ||||
|     "viewerjs": "^1.11.7", | ||||
|     "vue": "^3.3.11", | ||||
|     "vue-cropper": "^1.1.1", | ||||
|     "vue-router": "^4.2.5", | ||||
| @ -44,6 +46,7 @@ | ||||
|     "@tsconfig/node18": "^18.2.2", | ||||
|     "@types/node": "^18.18.5", | ||||
|     "@types/vue": "^2.0.0", | ||||
|     "@unocss/reset": "^66.1.1", | ||||
|     "@vitejs/plugin-vue": "^4.4.0", | ||||
|     "@vitejs/plugin-vue-jsx": "^3.0.2", | ||||
|     "@vue/tsconfig": "^0.4.0", | ||||
| @ -57,6 +60,8 @@ | ||||
|     "sass": "^1.88.0", | ||||
|     "typescript": "~5.2.0", | ||||
|     "unocss": "0.58.0", | ||||
|     "unplugin-auto-import": "^19.2.0", | ||||
|     "unplugin-vue-components": "^28.5.0", | ||||
|     "vite": "^4.5.1", | ||||
|     "vite-plugin-compression": "^0.5.1", | ||||
|     "vite-plugin-vue-devtools": "^7.7.6", | ||||
|  | ||||
							
								
								
									
										178
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
							
						
						| @ -62,6 +62,12 @@ importers: | ||||
|       quill-mention: | ||||
|         specifier: ^4.1.0 | ||||
|         version: 4.1.0 | ||||
|       sortablejs: | ||||
|         specifier: ^1.15.6 | ||||
|         version: 1.15.6 | ||||
|       viewerjs: | ||||
|         specifier: ^1.11.7 | ||||
|         version: 1.11.7 | ||||
|       vue: | ||||
|         specifier: ^3.3.11 | ||||
|         version: 3.5.13(typescript@5.2.2) | ||||
| @ -93,6 +99,9 @@ importers: | ||||
|       '@types/vue': | ||||
|         specifier: ^2.0.0 | ||||
|         version: 2.0.0(typescript@5.2.2) | ||||
|       '@unocss/reset': | ||||
|         specifier: ^66.1.1 | ||||
|         version: 66.1.1 | ||||
|       '@vitejs/plugin-vue': | ||||
|         specifier: ^4.4.0 | ||||
|         version: 4.6.2(vite@4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0))(vue@3.5.13(typescript@5.2.2)) | ||||
| @ -132,6 +141,12 @@ importers: | ||||
|       unocss: | ||||
|         specifier: 0.58.0 | ||||
|         version: 0.58.0(postcss@8.5.3)(rollup@3.29.5)(vite@4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0)) | ||||
|       unplugin-auto-import: | ||||
|         specifier: ^19.2.0 | ||||
|         version: 19.2.0(@vueuse/core@10.11.1(vue@3.5.13(typescript@5.2.2))) | ||||
|       unplugin-vue-components: | ||||
|         specifier: ^28.5.0 | ||||
|         version: 28.5.0(@babel/parser@7.27.2)(vue@3.5.13(typescript@5.2.2)) | ||||
|       vite: | ||||
|         specifier: ^4.5.1 | ||||
|         version: 4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0) | ||||
| @ -803,6 +818,9 @@ packages: | ||||
|   '@unocss/reset@0.58.0': | ||||
|     resolution: {integrity: sha512-UVZ5kz37JGbwAA06k/gjKYcekcTwi6oIhev1EpTtCvHLL6XYcYqcwb/u4Wjzprd3L3lxDGYXvGdjREGm2u7vbQ==} | ||||
| 
 | ||||
|   '@unocss/reset@66.1.1': | ||||
|     resolution: {integrity: sha512-WrI3sStMd/EXTcb3SaTVH10Wc9NKutW4+/HktQy470wEpncXdvihrXgCYwJH6LEEL4KOto3o+KKSD5xenWE7Aw==} | ||||
| 
 | ||||
|   '@unocss/rule-utils@0.58.0': | ||||
|     resolution: {integrity: sha512-LBJ9dJ/j5UIMzJF7pmIig55MtJAYtG+tn/zQRveZuPRVahzP+KqwlyB7u3uCUnQhdgo/MJODMcqyr0jl6+kTuA==} | ||||
|     engines: {node: '>=14'} | ||||
| @ -1691,6 +1709,10 @@ packages: | ||||
|     resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} | ||||
|     engines: {node: '>=0.8.0'} | ||||
| 
 | ||||
|   escape-string-regexp@5.0.0: | ||||
|     resolution: {integrity: sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==} | ||||
|     engines: {node: '>=12'} | ||||
| 
 | ||||
|   eslint-scope@5.1.1: | ||||
|     resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} | ||||
|     engines: {node: '>=8.0.0'} | ||||
| @ -1719,6 +1741,9 @@ packages: | ||||
|   estree-walker@2.0.2: | ||||
|     resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==} | ||||
| 
 | ||||
|   estree-walker@3.0.3: | ||||
|     resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} | ||||
| 
 | ||||
|   event-emitter@0.3.5: | ||||
|     resolution: {integrity: sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==} | ||||
| 
 | ||||
| @ -1787,6 +1812,14 @@ packages: | ||||
|   fastq@1.19.1: | ||||
|     resolution: {integrity: sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==} | ||||
| 
 | ||||
|   fdir@6.4.4: | ||||
|     resolution: {integrity: sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==} | ||||
|     peerDependencies: | ||||
|       picomatch: ^3 || ^4 | ||||
|     peerDependenciesMeta: | ||||
|       picomatch: | ||||
|         optional: true | ||||
| 
 | ||||
|   figures@6.1.0: | ||||
|     resolution: {integrity: sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg==} | ||||
|     engines: {node: '>=18'} | ||||
| @ -2153,6 +2186,9 @@ packages: | ||||
|   js-tokens@4.0.0: | ||||
|     resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} | ||||
| 
 | ||||
|   js-tokens@9.0.1: | ||||
|     resolution: {integrity: sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==} | ||||
| 
 | ||||
|   js-yaml@3.14.1: | ||||
|     resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} | ||||
|     hasBin: true | ||||
| @ -2813,6 +2849,9 @@ packages: | ||||
|   scroll-into-view-if-needed@2.2.31: | ||||
|     resolution: {integrity: sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA==} | ||||
| 
 | ||||
|   scule@1.3.0: | ||||
|     resolution: {integrity: sha512-6FtHJEvt+pVMIB9IBY+IcCJ6Z5f1iQnytgyfKMhDKgmzYG+TeH/wx1y3l27rshSbLiSanrR9ffZDrEsmjlQF2g==} | ||||
| 
 | ||||
|   section-matter@1.0.0: | ||||
|     resolution: {integrity: sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==} | ||||
|     engines: {node: '>=4'} | ||||
| @ -2894,6 +2933,9 @@ packages: | ||||
|   sortablejs@1.14.0: | ||||
|     resolution: {integrity: sha512-pBXvQCs5/33fdN1/39pPL0NZF20LeRbLQ5jtnheIPN9JQAaufGjKdWduZn4U7wCtVuzKhmRkI0DFYHYRbB2H1w==} | ||||
| 
 | ||||
|   sortablejs@1.15.6: | ||||
|     resolution: {integrity: sha512-aNfiuwMEpfBM/CN6LY0ibyhxPfPbyFeBTYJKCvzkJ2GkUpazIt3H+QIPAMHwqQ7tMKaHz1Qj+rJJCqljnf4p3A==} | ||||
| 
 | ||||
|   source-map-js@1.2.1: | ||||
|     resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} | ||||
|     engines: {node: '>=0.10.0'} | ||||
| @ -2951,6 +2993,9 @@ packages: | ||||
|     resolution: {integrity: sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw==} | ||||
|     engines: {node: '>=18'} | ||||
| 
 | ||||
|   strip-literal@3.0.0: | ||||
|     resolution: {integrity: sha512-TcccoMhJOM3OebGhSBEmp3UZ2SfDMZUEBdRA/9ynfLi8yYajyWX3JiXArcJt4Umh4vISpspkQIY8ZZoCqjbviA==} | ||||
| 
 | ||||
|   stylis@4.3.6: | ||||
|     resolution: {integrity: sha512-yQ3rwFWRfwNUY7H5vpU0wfdkNSnvnJinhF9830Swlaxl03zsOjCfmX0ugac+3LtK0lYSgwL/KXc8oYL3mG4YFQ==} | ||||
| 
 | ||||
| @ -3002,6 +3047,10 @@ packages: | ||||
|   tinyexec@1.0.1: | ||||
|     resolution: {integrity: sha512-5uC6DDlmeqiOwCPmK9jMSdOuZTh8bU39Ys6yidB+UTt5hfZUPGAypSgFRiEp+jbi9qH40BLDvy85jIU88wKSqw==} | ||||
| 
 | ||||
|   tinyglobby@0.2.13: | ||||
|     resolution: {integrity: sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw==} | ||||
|     engines: {node: '>=12.0.0'} | ||||
| 
 | ||||
|   to-object-path@0.3.0: | ||||
|     resolution: {integrity: sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==} | ||||
|     engines: {node: '>=0.10.0'} | ||||
| @ -3066,6 +3115,10 @@ packages: | ||||
|     resolution: {integrity: sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==} | ||||
|     engines: {node: '>=18'} | ||||
| 
 | ||||
|   unimport@4.2.0: | ||||
|     resolution: {integrity: sha512-mYVtA0nmzrysnYnyb3ALMbByJ+Maosee2+WyE0puXl+Xm2bUwPorPaaeZt0ETfuroPOtG8jj1g/qeFZ6buFnag==} | ||||
|     engines: {node: '>=18.12.0'} | ||||
| 
 | ||||
|   union-value@1.0.1: | ||||
|     resolution: {integrity: sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==} | ||||
|     engines: {node: '>=0.10.0'} | ||||
| @ -3093,6 +3146,39 @@ packages: | ||||
|       vite: | ||||
|         optional: true | ||||
| 
 | ||||
|   unplugin-auto-import@19.2.0: | ||||
|     resolution: {integrity: sha512-DGRHg86nUDKEYpny1p2kFZjeLg7kHQmknsPQ8krAshvpeypps7dFxNBsAqhBaxYINjetbgQilF8wbjuZxpdomg==} | ||||
|     engines: {node: '>=14'} | ||||
|     peerDependencies: | ||||
|       '@nuxt/kit': ^3.2.2 | ||||
|       '@vueuse/core': '*' | ||||
|     peerDependenciesMeta: | ||||
|       '@nuxt/kit': | ||||
|         optional: true | ||||
|       '@vueuse/core': | ||||
|         optional: true | ||||
| 
 | ||||
|   unplugin-utils@0.2.4: | ||||
|     resolution: {integrity: sha512-8U/MtpkPkkk3Atewj1+RcKIjb5WBimZ/WSLhhR3w6SsIj8XJuKTacSP8g+2JhfSGw0Cb125Y+2zA/IzJZDVbhA==} | ||||
|     engines: {node: '>=18.12.0'} | ||||
| 
 | ||||
|   unplugin-vue-components@28.5.0: | ||||
|     resolution: {integrity: sha512-o7fMKU/uI8NiP+E0W62zoduuguWqB0obTfHFtbr1AP2uo2lhUPnPttWUE92yesdiYfo9/0hxIrj38FMc1eaySg==} | ||||
|     engines: {node: '>=14'} | ||||
|     peerDependencies: | ||||
|       '@babel/parser': ^7.15.8 | ||||
|       '@nuxt/kit': ^3.2.2 | ||||
|       vue: 2 || 3 | ||||
|     peerDependenciesMeta: | ||||
|       '@babel/parser': | ||||
|         optional: true | ||||
|       '@nuxt/kit': | ||||
|         optional: true | ||||
| 
 | ||||
|   unplugin@2.3.2: | ||||
|     resolution: {integrity: sha512-3n7YA46rROb3zSj8fFxtxC/PqoyvYQ0llwz9wtUPUutr9ig09C8gGo5CWCwHrUzlqC1LLR43kxp5vEIyH1ac1w==} | ||||
|     engines: {node: '>=18.12.0'} | ||||
| 
 | ||||
|   unset-value@1.0.0: | ||||
|     resolution: {integrity: sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==} | ||||
|     engines: {node: '>=0.10.0'} | ||||
| @ -3134,6 +3220,9 @@ packages: | ||||
|     peerDependencies: | ||||
|       vue: ^3.0.11 | ||||
| 
 | ||||
|   viewerjs@1.11.7: | ||||
|     resolution: {integrity: sha512-0JuVqOmL5v1jmEAlG5EBDR3XquxY8DWFQbFMprOXgaBB0F7Q/X9xWdEaQc59D8xzwkdUgXEMSSknTpriq95igg==} | ||||
| 
 | ||||
|   vite-hot-client@2.0.4: | ||||
|     resolution: {integrity: sha512-W9LOGAyGMrbGArYJN4LBCdOC5+Zwh7dHvOHC0KmGKkJhsOzaKbpo/jEjpPKVHIW0/jBWj8RZG0NUxfgA8BxgAg==} | ||||
|     peerDependencies: | ||||
| @ -3288,6 +3377,9 @@ packages: | ||||
|     resolution: {integrity: sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==} | ||||
|     engines: {node: '>=10.13.0'} | ||||
| 
 | ||||
|   webpack-virtual-modules@0.6.2: | ||||
|     resolution: {integrity: sha512-66/V2i5hQanC51vBQKPH4aI8NMAcBW59FVBs+rC7eGHupMyfn34q7rZIE+ETlJ+XTevqfUhVVBgSUNSW2flEUQ==} | ||||
| 
 | ||||
|   webpack@5.99.8: | ||||
|     resolution: {integrity: sha512-lQ3CPiSTpfOnrEGeXDwoq5hIGzSjmwD72GdfVzF7CQAI7t47rJG9eDWvcEkEn3CUQymAElVvDg3YNTlCYj+qUQ==} | ||||
|     engines: {node: '>=10.13.0'} | ||||
| @ -4028,6 +4120,8 @@ snapshots: | ||||
| 
 | ||||
|   '@unocss/reset@0.58.0': {} | ||||
| 
 | ||||
|   '@unocss/reset@66.1.1': {} | ||||
| 
 | ||||
|   '@unocss/rule-utils@0.58.0': | ||||
|     dependencies: | ||||
|       '@unocss/core': 0.58.0 | ||||
| @ -5112,6 +5206,8 @@ snapshots: | ||||
| 
 | ||||
|   escape-string-regexp@1.0.5: {} | ||||
| 
 | ||||
|   escape-string-regexp@5.0.0: {} | ||||
| 
 | ||||
|   eslint-scope@5.1.1: | ||||
|     dependencies: | ||||
|       esrecurse: 4.3.0 | ||||
| @ -5136,6 +5232,10 @@ snapshots: | ||||
| 
 | ||||
|   estree-walker@2.0.2: {} | ||||
| 
 | ||||
|   estree-walker@3.0.3: | ||||
|     dependencies: | ||||
|       '@types/estree': 1.0.7 | ||||
| 
 | ||||
|   event-emitter@0.3.5: | ||||
|     dependencies: | ||||
|       d: 1.0.2 | ||||
| @ -5237,6 +5337,10 @@ snapshots: | ||||
|     dependencies: | ||||
|       reusify: 1.1.0 | ||||
| 
 | ||||
|   fdir@6.4.4(picomatch@4.0.2): | ||||
|     optionalDependencies: | ||||
|       picomatch: 4.0.2 | ||||
| 
 | ||||
|   figures@6.1.0: | ||||
|     dependencies: | ||||
|       is-unicode-supported: 2.1.0 | ||||
| @ -5575,6 +5679,8 @@ snapshots: | ||||
| 
 | ||||
|   js-tokens@4.0.0: {} | ||||
| 
 | ||||
|   js-tokens@9.0.1: {} | ||||
| 
 | ||||
|   js-yaml@3.14.1: | ||||
|     dependencies: | ||||
|       argparse: 1.0.10 | ||||
| @ -6332,6 +6438,8 @@ snapshots: | ||||
|     dependencies: | ||||
|       compute-scroll-into-view: 1.0.20 | ||||
| 
 | ||||
|   scule@1.3.0: {} | ||||
| 
 | ||||
|   section-matter@1.0.0: | ||||
|     dependencies: | ||||
|       extend-shallow: 2.0.1 | ||||
| @ -6424,6 +6532,8 @@ snapshots: | ||||
| 
 | ||||
|   sortablejs@1.14.0: {} | ||||
| 
 | ||||
|   sortablejs@1.15.6: {} | ||||
| 
 | ||||
|   source-map-js@1.2.1: {} | ||||
| 
 | ||||
|   source-map-resolve@0.5.3: | ||||
| @ -6474,6 +6584,10 @@ snapshots: | ||||
| 
 | ||||
|   strip-final-newline@4.0.0: {} | ||||
| 
 | ||||
|   strip-literal@3.0.0: | ||||
|     dependencies: | ||||
|       js-tokens: 9.0.1 | ||||
| 
 | ||||
|   stylis@4.3.6: {} | ||||
| 
 | ||||
|   superjson@2.2.2: | ||||
| @ -6514,6 +6628,11 @@ snapshots: | ||||
| 
 | ||||
|   tinyexec@1.0.1: {} | ||||
| 
 | ||||
|   tinyglobby@0.2.13: | ||||
|     dependencies: | ||||
|       fdir: 6.4.4(picomatch@4.0.2) | ||||
|       picomatch: 4.0.2 | ||||
| 
 | ||||
|   to-object-path@0.3.0: | ||||
|     dependencies: | ||||
|       kind-of: 3.2.2 | ||||
| @ -6566,6 +6685,23 @@ snapshots: | ||||
| 
 | ||||
|   unicorn-magic@0.3.0: {} | ||||
| 
 | ||||
|   unimport@4.2.0: | ||||
|     dependencies: | ||||
|       acorn: 8.14.1 | ||||
|       escape-string-regexp: 5.0.0 | ||||
|       estree-walker: 3.0.3 | ||||
|       local-pkg: 1.1.1 | ||||
|       magic-string: 0.30.17 | ||||
|       mlly: 1.7.4 | ||||
|       pathe: 2.0.3 | ||||
|       picomatch: 4.0.2 | ||||
|       pkg-types: 2.1.0 | ||||
|       scule: 1.3.0 | ||||
|       strip-literal: 3.0.0 | ||||
|       tinyglobby: 0.2.13 | ||||
|       unplugin: 2.3.2 | ||||
|       unplugin-utils: 0.2.4 | ||||
| 
 | ||||
|   union-value@1.0.1: | ||||
|     dependencies: | ||||
|       arr-union: 3.1.0 | ||||
| @ -6610,6 +6746,44 @@ snapshots: | ||||
|       - rollup | ||||
|       - supports-color | ||||
| 
 | ||||
|   unplugin-auto-import@19.2.0(@vueuse/core@10.11.1(vue@3.5.13(typescript@5.2.2))): | ||||
|     dependencies: | ||||
|       local-pkg: 1.1.1 | ||||
|       magic-string: 0.30.17 | ||||
|       picomatch: 4.0.2 | ||||
|       unimport: 4.2.0 | ||||
|       unplugin: 2.3.2 | ||||
|       unplugin-utils: 0.2.4 | ||||
|     optionalDependencies: | ||||
|       '@vueuse/core': 10.11.1(vue@3.5.13(typescript@5.2.2)) | ||||
| 
 | ||||
|   unplugin-utils@0.2.4: | ||||
|     dependencies: | ||||
|       pathe: 2.0.3 | ||||
|       picomatch: 4.0.2 | ||||
| 
 | ||||
|   unplugin-vue-components@28.5.0(@babel/parser@7.27.2)(vue@3.5.13(typescript@5.2.2)): | ||||
|     dependencies: | ||||
|       chokidar: 3.6.0 | ||||
|       debug: 4.4.0 | ||||
|       local-pkg: 1.1.1 | ||||
|       magic-string: 0.30.17 | ||||
|       mlly: 1.7.4 | ||||
|       tinyglobby: 0.2.13 | ||||
|       unplugin: 2.3.2 | ||||
|       unplugin-utils: 0.2.4 | ||||
|       vue: 3.5.13(typescript@5.2.2) | ||||
|     optionalDependencies: | ||||
|       '@babel/parser': 7.27.2 | ||||
|     transitivePeerDependencies: | ||||
|       - supports-color | ||||
| 
 | ||||
|   unplugin@2.3.2: | ||||
|     dependencies: | ||||
|       acorn: 8.14.1 | ||||
|       picomatch: 4.0.2 | ||||
|       webpack-virtual-modules: 0.6.2 | ||||
| 
 | ||||
|   unset-value@1.0.0: | ||||
|     dependencies: | ||||
|       has-value: 0.3.1 | ||||
| @ -6648,6 +6822,8 @@ snapshots: | ||||
|       evtd: 0.2.4 | ||||
|       vue: 3.5.13(typescript@5.2.2) | ||||
| 
 | ||||
|   viewerjs@1.11.7: {} | ||||
| 
 | ||||
|   vite-hot-client@2.0.4(vite@4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0)): | ||||
|     dependencies: | ||||
|       vite: 4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0) | ||||
| @ -6822,6 +6998,8 @@ snapshots: | ||||
| 
 | ||||
|   webpack-sources@3.2.3: {} | ||||
| 
 | ||||
|   webpack-virtual-modules@0.6.2: {} | ||||
| 
 | ||||
|   webpack@5.99.8: | ||||
|     dependencies: | ||||
|       '@types/eslint-scope': 3.7.7 | ||||
|  | ||||
| @ -25,3 +25,7 @@ export const ServeRefreshToken = () => { | ||||
| export const ServeForgetPassword = (data) => { | ||||
|   return post('/api/v1/auth/forget', data) | ||||
| } | ||||
| // 获取用户信息服务
 | ||||
| export const GetUserInfo = (data) => { | ||||
|   return post('/api/v1/users/info', data) | ||||
| } | ||||
| @ -9,7 +9,10 @@ export const ServeGetTalkList = (data = {}) => { | ||||
| export const ServeCreateTalkList = (data = {}) => { | ||||
|   return post('/api/v1/talk/create', data) | ||||
| } | ||||
| 
 | ||||
| // 聊天列表创建服务接口
 | ||||
| export const voiceToText = (data = {}) => { | ||||
|   return post('/api/v1/talk/message/voice-to-text', data) | ||||
| } | ||||
| // 删除聊天列表服务接口
 | ||||
| export const ServeDeleteTalkList = (data = {}) => { | ||||
|   return post('/api/v1/talk/delete', data) | ||||
|  | ||||
| @ -45,10 +45,12 @@ export const ServeFindFriendApplyNum = () => { | ||||
| } | ||||
| 
 | ||||
| // 搜索用户信息服务接口
 | ||||
| // export const ServeSearchUser = (data) => {
 | ||||
| //   return get('/api/v1/contact/detail', data)
 | ||||
| // }
 | ||||
| export const ServeSearchUser = (data) => { | ||||
|   return get('/api/v1/contact/detail', data) | ||||
|   return post('/api/v1/users/info', data) | ||||
| } | ||||
| 
 | ||||
| // 搜索用户信息服务接口
 | ||||
| export const ServeContactGroupList = (data) => { | ||||
|   return get('/api/v1/contact/group/list', data) | ||||
|  | ||||
| @ -21,6 +21,9 @@ export const ServeFileSubareaUpload = (data = {}, options = {}) => { | ||||
| } | ||||
| 
 | ||||
| // 上传图片文件或者视频
 | ||||
| export const uploadImg = (data) => { | ||||
|   return post('/upload/img', data,{baseURL:import.meta.env.VITE_EPR_BASEURL}) | ||||
| export const uploadImg = (data, signal) => { | ||||
|   return post('/upload/img', data, { | ||||
|     baseURL: import.meta.env.VITE_EPR_BASEURL, | ||||
|     signal: signal | ||||
|   }) | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,7 @@ | ||||
| * { | ||||
|   margin: 0; | ||||
|   padding: 0; | ||||
|   box-sizing: border-box!important; | ||||
| } | ||||
| 
 | ||||
| @font-face { | ||||
|  | ||||
| @ -1,10 +1,10 @@ | ||||
| // 默认主题 | ||||
| html { | ||||
|   --im-primary-color: #1890ff; | ||||
|   --im-primary-color: #462AA0; | ||||
|   --im-bg-color: #ffffff; | ||||
|   --line-border-color: #f5f5f5; | ||||
|   --border-color: #eeeaea; | ||||
|   --im-text-color: #333; | ||||
|   --im-text-color: #BABABA; | ||||
|   --im-text-color-grey: #333; | ||||
|   --im-active-bg-color: #f5f5f5; | ||||
|   --im-hover-bg-color: #f5f5f5; | ||||
| @ -21,10 +21,10 @@ html { | ||||
|   // message | ||||
|   --im-message-bg-color: #f7f7f7; | ||||
|   --im-message-border-color: #efeff5; | ||||
|   --im-message-left-bg-color: #eff0f1; | ||||
|   --im-message-left-bg-color: #F4F4FC; | ||||
|   --im-message-left-text-color: #333; | ||||
|   --im-message-right-bg-color: #daf3fd; | ||||
|   --im-message-right-text-color: #333; | ||||
|   --im-message-right-bg-color: #46299D; | ||||
|   --im-message-right-text-color: #fff; | ||||
| } | ||||
| 
 | ||||
| // 黑色主题 | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/close.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/excel-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/faxi@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/file-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/file@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 607 B | 
| Before Width: | Height: | Size: 6.6 KiB After Width: | Height: | Size: 5.6 KiB | 
| Before Width: | Height: | Size: 5.5 KiB After Width: | Height: | Size: 5.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/pcyyb_2100100012_installer.exe
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/pdf-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/ppt-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/word-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.9 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/zu6254@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 118 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/zu6299@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/zu6300@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/zu6302@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/zu6306@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.3 KiB | 
| @ -87,8 +87,8 @@ const text_avatar = computed(() => { | ||||
|   background: linear-gradient(to right, #674bbc, #46299d); | ||||
|   flex-shrink: 0; | ||||
|   img { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     width: 42px; | ||||
|     height: 42px; | ||||
|     object-fit: cover; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -1,81 +1,117 @@ | ||||
| <script lang="ts" setup> | ||||
| // 引入Quill编辑器的样式文件 | ||||
| import '@vueup/vue-quill/dist/vue-quill.snow.css' | ||||
| // 引入图片上传插件的样式 | ||||
| import 'quill-image-uploader/dist/quill.imageUploader.min.css' | ||||
| // 引入自定义的提及功能样式 | ||||
| import '@/assets/css/editor-mention.less' | ||||
| // 引入Vue核心功能 | ||||
| import { reactive, watch, ref, markRaw, computed, onMounted, onUnmounted } from 'vue' | ||||
| // 引入Naive UI的弹出框组件 | ||||
| import { NPopover } from 'naive-ui' | ||||
| // 引入图标组件 | ||||
| import { | ||||
|   Voice as IconVoice, | ||||
|   SourceCode, | ||||
|   Local, | ||||
|   SmilingFace, | ||||
|   Pic, | ||||
|   FolderUpload, | ||||
|   Ranking, | ||||
|   History | ||||
|   Voice as IconVoice,    // 语音图标 | ||||
|   SourceCode,            // 代码图标 | ||||
|   Local,                 // 地理位置图标 | ||||
|   SmilingFace,           // 表情图标 | ||||
|   Pic,                   // 图片图标 | ||||
|   FolderUpload,          // 文件上传图标 | ||||
|   Ranking,               // 排名图标(用于投票) | ||||
|   History                // 历史记录图标 | ||||
| } from '@icon-park/vue-next' | ||||
| // 引入Quill编辑器及其核心实例 | ||||
| import { QuillEditor, Quill } from '@vueup/vue-quill' | ||||
| // 引入图片上传插件 | ||||
| import ImageUploader from 'quill-image-uploader' | ||||
| // 引入自定义表情符号格式 | ||||
| import EmojiBlot from './formats/emoji' | ||||
| // 引入自定义引用格式 | ||||
| import QuoteBlot from './formats/quote' | ||||
| // 引入提及功能 | ||||
| import 'quill-mention' | ||||
| // 引入状态管理 | ||||
| import { useDialogueStore, useEditorDraftStore } from '@/store' | ||||
| // 引入编辑器工具函数 | ||||
| import { deltaToMessage, deltaToString, isEmptyDelta } from './util' | ||||
| // 引入获取图片信息的工具函数 | ||||
| import { getImageInfo } from '@/utils/functions' | ||||
| // 引入编辑器常量定义 | ||||
| import { EditorConst } from '@/constant/event-bus' | ||||
| // 引入事件调用工具 | ||||
| import { emitCall } from '@/utils/common' | ||||
| // 引入默认头像常量 | ||||
| import { defAvatar } from '@/constant/default' | ||||
| import MeEditorVote from './MeEditorVote.vue' | ||||
| import MeEditorEmoticon from './MeEditorEmoticon.vue' | ||||
| import MeEditorCode from './MeEditorCode.vue' | ||||
| import MeEditorRecorder from './MeEditorRecorder.vue' | ||||
| // 引入编辑器各子组件 | ||||
| import MeEditorVote from './MeEditorVote.vue'            // 投票组件 | ||||
| import MeEditorEmoticon from './MeEditorEmoticon.vue'    // 表情组件 | ||||
| import MeEditorCode from './MeEditorCode.vue'            // 代码编辑组件 | ||||
| import MeEditorRecorder from './MeEditorRecorder.vue'    // 录音组件 | ||||
| // 引入上传API | ||||
| import { ServeUploadImage } from '@/api/upload' | ||||
| import { uploadImg } from '@/api/upload' | ||||
| // 引入事件总线钩子 | ||||
| import { useEventBus } from '@/hooks' | ||||
| 
 | ||||
| Quill.register('formats/emoji', EmojiBlot) | ||||
| Quill.register('formats/quote', QuoteBlot) | ||||
| Quill.register('modules/imageUploader', ImageUploader) | ||||
| // 注册Quill编辑器的自定义格式 | ||||
| Quill.register('formats/emoji', EmojiBlot)       // 注册表情格式 | ||||
| Quill.register('formats/quote', QuoteBlot)       // 注册引用格式 | ||||
| Quill.register('modules/imageUploader', ImageUploader)  // 注册图片上传模块 | ||||
| 
 | ||||
| // 定义组件的事件 | ||||
| const emit = defineEmits(['editor-event']) | ||||
| // 获取对话状态管理 | ||||
| const dialogueStore = useDialogueStore() | ||||
| // 获取编辑器草稿状态管理 | ||||
| const editorDraftStore = useEditorDraftStore() | ||||
| // 定义组件props | ||||
| const props = defineProps({ | ||||
|   vote: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|     default: false  // 是否显示投票功能 | ||||
|   }, | ||||
|   members: { | ||||
|     default: () => [] | ||||
|     default: () => []  // 聊天成员列表,用于@功能 | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| // 编辑器引用 | ||||
| const editor = ref() | ||||
| 
 | ||||
| // 获取Quill编辑器实例 | ||||
| const getQuill = () => { | ||||
|   return editor.value?.getQuill() | ||||
| } | ||||
| 
 | ||||
| // 获取当前编辑器光标位置 | ||||
| const getQuillSelectionIndex = () => { | ||||
|   let quill = getQuill() | ||||
| 
 | ||||
|   return (quill.getSelection() || {}).index || quill.getLength() | ||||
| } | ||||
| 
 | ||||
| // 计算当前对话索引名称(标识当前聊天) | ||||
| const indexName = computed(() => dialogueStore.index_name) | ||||
| // 控制是否显示编辑器的投票界面 | ||||
| const isShowEditorVote = ref(false) | ||||
| // 控制是否显示编辑器的代码界面 | ||||
| const isShowEditorCode = ref(false) | ||||
| // 控制是否显示录音界面 | ||||
| const isShowEditorRecorder = ref(false) | ||||
| // 图片文件上传DOM引用 | ||||
| const fileImageRef = ref() | ||||
| // 文件上传DOM引用 | ||||
| const uploadFileRef = ref() | ||||
| // 表情面板引用 | ||||
| const emoticonRef = ref() | ||||
| 
 | ||||
| // 编辑器配置选项 | ||||
| const editorOption = { | ||||
|   debug: false, | ||||
|   modules: { | ||||
|     toolbar: false, | ||||
|     toolbar: false,  // 禁用默认工具栏 | ||||
|     clipboard: { | ||||
|       // 粘贴版,处理粘贴时候的自带样式 | ||||
|       // 粘贴处理,去除粘贴时的自带样式 | ||||
|       matchers: [[Node.ELEMENT_NODE, onClipboardMatcher]] | ||||
|     }, | ||||
| 
 | ||||
| @ -83,19 +119,22 @@ const editorOption = { | ||||
|       bindings: { | ||||
|         enter: { | ||||
|           key: 13, | ||||
|           handler: onSendMessage | ||||
|           handler: onSendMessage  // 按Enter键发送消息 | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
| 
 | ||||
|     // 图片上传配置 | ||||
|     imageUploader: { | ||||
|       upload: onEditorUpload | ||||
|     }, | ||||
| 
 | ||||
|     // @功能配置 | ||||
|     mention: { | ||||
|       allowedChars: /^[\u4e00-\u9fa5]*$/, | ||||
|       mentionDenotationChars: ['@'], | ||||
|       positioningStrategy: 'fixed', | ||||
|       allowedChars: /^[\u4e00-\u9fa5]*$/,  // 允许中文字符 | ||||
|       mentionDenotationChars: ['@'],       // @符号触发 | ||||
|       positioningStrategy: 'fixed',        // 定位策略 | ||||
|       // 渲染@项目的函数 | ||||
|       renderItem: (data: any) => { | ||||
|         const el = document.createElement('div') | ||||
|         el.className = 'ed-member-item' | ||||
| @ -103,6 +142,7 @@ const editorOption = { | ||||
|         el.innerHTML += `<span class="nickname">${data.nickname}</span>` | ||||
|         return el | ||||
|       }, | ||||
|       // 数据源函数,过滤匹配的用户 | ||||
|       source: function (searchTerm: string, renderList: any) { | ||||
|         if (!props.members.length) { | ||||
|           return renderList([]) | ||||
| @ -123,66 +163,73 @@ const editorOption = { | ||||
|     } | ||||
|   }, | ||||
|   placeholder: '按Enter发送 / Shift+Enter 换行', | ||||
|   theme: 'snow' | ||||
|   theme: 'snow'  // 使用snow主题 | ||||
| } | ||||
| 
 | ||||
| // 底部工具栏配置 | ||||
| const navs = reactive([ | ||||
|   { | ||||
|     title: '图片', | ||||
|     icon: markRaw(Pic), | ||||
|     show: true, | ||||
|     click: () => { | ||||
|       fileImageRef.value.click() | ||||
|       fileImageRef.value.click()  // 触发图片上传 | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     title: '附件', | ||||
|     title: '文件', | ||||
|     icon: markRaw(FolderUpload), | ||||
|     show: true, | ||||
|     click: () => { | ||||
|       uploadFileRef.value.click() | ||||
|       uploadFileRef.value.click()  // 触发文件上传 | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     title: '代码', | ||||
|     icon: markRaw(SourceCode), | ||||
|     show: true, | ||||
|     click: () => { | ||||
|       isShowEditorCode.value = true | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     title: '语音消息', | ||||
|     icon: markRaw(IconVoice), | ||||
|     show: true, | ||||
|     click: () => { | ||||
|       isShowEditorRecorder.value = true | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     title: '地理位置', | ||||
|     icon: markRaw(Local), | ||||
|     show: true, | ||||
|     click: () => {} | ||||
|   }, | ||||
|   { | ||||
|     title: '群投票', | ||||
|     icon: markRaw(Ranking), | ||||
|     show: computed(() => props.vote), | ||||
|     click: () => { | ||||
|       isShowEditorVote.value = true | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     title: '历史记录', | ||||
|     icon: markRaw(History), | ||||
|     show: true, | ||||
|     click: () => { | ||||
|       emit('editor-event', emitCall('history_event')) | ||||
|     } | ||||
|   } | ||||
|   // 以下功能已被注释掉,但保留代码 | ||||
|   // { | ||||
|   //   title: '代码', | ||||
|   //   icon: markRaw(SourceCode), | ||||
|   //   show: true, | ||||
|   //   click: () => { | ||||
|   //     isShowEditorCode.value = true | ||||
|   //   } | ||||
|   // }, | ||||
|   // { | ||||
|   //   title: '语音消息', | ||||
|   //   icon: markRaw(IconVoice), | ||||
|   //   show: true, | ||||
|   //   click: () => { | ||||
|   //     isShowEditorRecorder.value = true | ||||
|   //   } | ||||
|   // }, | ||||
|   // { | ||||
|   //   title: '地理位置', | ||||
|   //   icon: markRaw(Local), | ||||
|   //   show: true, | ||||
|   //   click: () => {} | ||||
|   // }, | ||||
|   // { | ||||
|   //   title: '群投票', | ||||
|   //   icon: markRaw(Ranking), | ||||
|   //   show: computed(() => props.vote), | ||||
|   //   click: () => { | ||||
|   //     isShowEditorVote.value = true | ||||
|   //   } | ||||
|   // }, | ||||
|   // { | ||||
|   //   title: '历史记录', | ||||
|   //   icon: markRaw(History), | ||||
|   //   show: true, | ||||
|   //   click: () => { | ||||
|   //     emit('editor-event', emitCall('history_event')) | ||||
|   //   } | ||||
|   // } | ||||
| ]) | ||||
| 
 | ||||
| /** | ||||
|  * 上传图片函数 | ||||
|  * @param file 文件对象 | ||||
|  * @returns Promise,成功时返回图片URL | ||||
|  */ | ||||
| function onUploadImage(file: File) { | ||||
|   return new Promise((resolve) => { | ||||
|     let image = new Image() | ||||
| @ -190,35 +237,44 @@ function onUploadImage(file: File) { | ||||
|     image.onload = () => { | ||||
|       const form = new FormData() | ||||
|       form.append('file', file) | ||||
|       form.append("source", "fonchain-chat"); | ||||
|       // form.append('width', image.width.toString()) | ||||
|       // form.append('height', image.height.toString()) | ||||
|       form.append("source", "fonchain-chat");  // 图片来源标识 | ||||
|       // 添加图片尺寸信息作为URL参数 | ||||
|       form.append("urlParam", `width=${image.width}&height=${image.height}`); | ||||
| 
 | ||||
|       // 调用上传API | ||||
|       uploadImg(form).then(({ code, data, message }) => { | ||||
|         if (code == 0) { | ||||
|           resolve(data.ori_url) | ||||
|           resolve(data.ori_url)  // 返回原始图片URL | ||||
|         } else { | ||||
|           resolve('') | ||||
|           window['$message'].error(message) | ||||
|           window['$message'].error(message)  // 显示错误信息 | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 编辑器上传处理函数 | ||||
|  * @param file 要上传的文件 | ||||
|  * @returns Promise | ||||
|  */ | ||||
| function onEditorUpload(file: File) { | ||||
|   async function fn(file: File, resolve: Function, reject: Function) { | ||||
|     if (file.type.indexOf('image/') === 0) { | ||||
|       // 如果是图片,使用图片上传处理 | ||||
|       return resolve(await onUploadImage(file)) | ||||
|     } | ||||
| 
 | ||||
|     reject() | ||||
| 
 | ||||
|     // 非图片文件的处理 | ||||
|     if (file.type.indexOf('video/') === 0) { | ||||
|       // 视频文件 | ||||
|       let fn = emitCall('video_event', file, () => {}) | ||||
|       emit('editor-event', fn) | ||||
|     } else { | ||||
|       // 其他文件 | ||||
|       let fn = emitCall('file_event', file, () => {}) | ||||
|       emit('editor-event', fn) | ||||
|     } | ||||
| @ -229,29 +285,40 @@ function onEditorUpload(file: File) { | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 投票事件处理 | ||||
|  * @param data 投票数据 | ||||
|  */ | ||||
| function onVoteEvent(data: any) { | ||||
|   const msg = emitCall('vote_event', data, (ok: boolean) => { | ||||
|     if (ok) { | ||||
|       isShowEditorVote.value = false | ||||
|       isShowEditorVote.value = false  // 成功后关闭投票界面 | ||||
|     } | ||||
|   }) | ||||
| 
 | ||||
|   emit('editor-event', msg) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 表情事件处理 | ||||
|  * @param data 表情数据 | ||||
|  */ | ||||
| function onEmoticonEvent(data: any) { | ||||
|   emoticonRef.value.setShow(false) | ||||
|   emoticonRef.value.setShow(false)  // 关闭表情面板 | ||||
| 
 | ||||
|   if (data.type == 1) { | ||||
|     // 插入文本表情 | ||||
|     const quill = getQuill() | ||||
|     let index = getQuillSelectionIndex() | ||||
| 
 | ||||
|     // 删除编辑器中多余的换行符 | ||||
|     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { | ||||
|       quill.deleteText(0, 1) | ||||
|       index = 0 | ||||
|     } | ||||
| 
 | ||||
|     if (data.img) { | ||||
|       // 插入图片表情 | ||||
|       quill.insertEmbed(index, 'emoji', { | ||||
|         alt: data.value, | ||||
|         src: data.img, | ||||
| @ -259,40 +326,54 @@ function onEmoticonEvent(data: any) { | ||||
|         height: '24px' | ||||
|       }) | ||||
|     } else { | ||||
|       // 插入文本表情 | ||||
|       quill.insertText(index, data.value) | ||||
|     } | ||||
| 
 | ||||
|     // 设置光标位置 | ||||
|     quill.setSelection(index + 1, 0, 'user') | ||||
|   } else { | ||||
|     // 发送整个表情包 | ||||
|     let fn = emitCall('emoticon_event', data.value, () => {}) | ||||
|     emit('editor-event', fn) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 代码事件处理 | ||||
|  * @param data 代码数据 | ||||
|  */ | ||||
| function onCodeEvent(data: any) { | ||||
|   const msg = emitCall('code_event', data, (ok: boolean) => { | ||||
|     isShowEditorCode.value = false | ||||
|     isShowEditorCode.value = false  // 成功后关闭代码界面 | ||||
|   }) | ||||
| 
 | ||||
|   emit('editor-event', msg) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 文件上传处理 | ||||
|  * @param e 上传事件对象 | ||||
|  */ | ||||
| async function onUploadFile(e: any) { | ||||
|   let file = e.target.files[0] | ||||
| 
 | ||||
|   e.target.value = null | ||||
|   e.target.value = null  // 清空input,允许再次选择相同文件 | ||||
| 
 | ||||
|     console.log("文件类型"+file.type) | ||||
|   console.log("文件类型"+file.type) | ||||
|   if (file.type.indexOf('image/') === 0) { | ||||
|     console.log("进入图片") | ||||
|     // 处理图片文件 | ||||
|     const quill = getQuill() | ||||
|     let index = getQuillSelectionIndex() | ||||
| 
 | ||||
|     // 删除编辑器中多余的换行符 | ||||
|     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { | ||||
|       quill.deleteText(0, 1) | ||||
|       index = 0 | ||||
|     } | ||||
| 
 | ||||
|     // 上传图片并插入到编辑器中 | ||||
|     let src = await onUploadImage(file) | ||||
|     if (src) { | ||||
|       quill.insertEmbed(index, 'image', src) | ||||
| @ -304,29 +385,41 @@ async function onUploadFile(e: any) { | ||||
| 
 | ||||
|   if (file.type.indexOf('video/') === 0) { | ||||
|     console.log("进入视频") | ||||
|     // 处理视频文件 | ||||
|     let fn = emitCall('video_event', file, () => {}) | ||||
|     emit('editor-event', fn) | ||||
|   } else { | ||||
|     console.log("进入其他") | ||||
|     // 处理其他类型文件 | ||||
|     let fn = emitCall('file_event', file, () => {}) | ||||
|     emit('editor-event', fn) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 录音事件处理 | ||||
|  * @param file 录音文件 | ||||
|  */ | ||||
| function onRecorderEvent(file: any) { | ||||
|   emit('editor-event', emitCall('file_event', file)) | ||||
|   isShowEditorRecorder.value = false | ||||
|   isShowEditorRecorder.value = false  // 关闭录音界面 | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 粘贴内容处理,移除粘贴内容中的样式 | ||||
|  * @param node DOM节点 | ||||
|  * @param Delta Quill Delta对象 | ||||
|  * @returns 处理后的Delta | ||||
|  */ | ||||
| function onClipboardMatcher(node: any, Delta) { | ||||
|   const ops: any[] = [] | ||||
| 
 | ||||
|   Delta.ops.forEach((op) => { | ||||
|     // 如果粘贴了图片,这里会是一个对象,所以可以这样处理 | ||||
|     // 处理粘贴内容 | ||||
|     if (op.insert && typeof op.insert === 'string') { | ||||
|       ops.push({ | ||||
|         insert: op.insert, // 文字内容 | ||||
|         attributes: {} //文字样式(包括背景色和文字颜色等) | ||||
|         attributes: {} // 移除所有样式 | ||||
|       }) | ||||
|     } else { | ||||
|       ops.push(op) | ||||
| @ -337,12 +430,16 @@ function onClipboardMatcher(node: any, Delta) { | ||||
|   return Delta | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 发送消息处理 | ||||
|  * 根据编辑器内容类型发送不同类型的消息 | ||||
|  */ | ||||
| function onSendMessage() { | ||||
|   var delta = getQuill().getContents() | ||||
|   let data = deltaToMessage(delta) | ||||
|   let data = deltaToMessage(delta)  // 转换Delta为消息格式 | ||||
| 
 | ||||
|   if (data.items.length === 0) { | ||||
|     return | ||||
|     return  // 没有内容不发送 | ||||
|   } | ||||
| 
 | ||||
|   switch (data.msgType) { | ||||
| @ -351,60 +448,72 @@ function onSendMessage() { | ||||
|         return window['$message'].info('发送内容超长,请分条发送') | ||||
|       } | ||||
| 
 | ||||
|       // 发送文本消息 | ||||
|       emit( | ||||
|         'editor-event', | ||||
|         emitCall('text_event', data, (ok: any) => { | ||||
|           ok && getQuill().setContents([], Quill.sources.USER) | ||||
|           ok && getQuill().setContents([], Quill.sources.USER)  // 成功发送后清空编辑器 | ||||
|         }) | ||||
|       ) | ||||
|       break | ||||
|     case 3: // 图片消息 | ||||
|       // 发送图片消息 | ||||
|       emit( | ||||
|         'editor-event', | ||||
|         emitCall( | ||||
|           'image_event', | ||||
|           { ...getImageInfo(data.items[0].content), url: data.items[0].content, size: 10000 }, | ||||
|           (ok: any) => { | ||||
|             ok && getQuill().setContents([]) | ||||
|             ok && getQuill().setContents([])  // 成功发送后清空编辑器 | ||||
|           } | ||||
|         ) | ||||
|       ) | ||||
|       break | ||||
|     case 12: // 图文消息 | ||||
|     case 12: // 图文混合消息 | ||||
|       // 发送混合消息 | ||||
|       emit( | ||||
|         'editor-event', | ||||
|         emitCall('mixed_event', data, (ok: any) => { | ||||
|           ok && getQuill().setContents([]) | ||||
|           ok && getQuill().setContents([])  // 成功发送后清空编辑器 | ||||
|         }) | ||||
|       ) | ||||
|       break | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 编辑器内容改变时的处理 | ||||
|  * 保存草稿并触发输入事件 | ||||
|  */ | ||||
| function onEditorChange() { | ||||
|   let delta = getQuill().getContents() | ||||
| 
 | ||||
|   let text = deltaToString(delta) | ||||
|   let text = deltaToString(delta)  // 将Delta转为纯文本 | ||||
| 
 | ||||
|   if (!isEmptyDelta(delta)) { | ||||
|     // 保存草稿到store | ||||
|     editorDraftStore.items[indexName.value || ''] = JSON.stringify({ | ||||
|       text: text, | ||||
|       ops: delta.ops | ||||
|     }) | ||||
|   } else { | ||||
|     // 删除 editorDraftStore.items 下的元素 | ||||
|     // 编辑器为空时删除对应草稿 | ||||
|     delete editorDraftStore.items[indexName.value || ''] | ||||
|   } | ||||
| 
 | ||||
|   // 触发输入事件 | ||||
|   emit('editor-event', emitCall('input_event', text)) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 加载编辑器草稿内容 | ||||
|  * 当切换聊天对象时,加载对应的草稿 | ||||
|  */ | ||||
| function loadEditorDraftText() { | ||||
|   if (!editor.value) return | ||||
| 
 | ||||
|   // 这里延迟处理,不然会有问题 | ||||
|   // 延迟处理,确保DOM已渲染 | ||||
|   setTimeout(() => { | ||||
|     hideMentionDom() | ||||
|     hideMentionDom()  // 隐藏@菜单 | ||||
| 
 | ||||
|     const quill = getQuill() | ||||
| 
 | ||||
| @ -415,33 +524,47 @@ function loadEditorDraftText() { | ||||
|     if (draft) { | ||||
|       quill.setContents(JSON.parse(draft)?.ops || []) | ||||
|     } else { | ||||
|       quill.setContents([]) | ||||
|       quill.setContents([])  // 没有草稿则清空编辑器 | ||||
|     } | ||||
| 
 | ||||
|     // 设置光标位置到末尾 | ||||
|     const index = getQuillSelectionIndex() | ||||
|     quill.setSelection(index, 0, 'user') | ||||
|   }, 0) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 处理@成员事件 | ||||
|  * @param data @成员数据 | ||||
|  */ | ||||
| function onSubscribeMention(data: any) { | ||||
|   const mention = getQuill().getModule('mention') | ||||
| 
 | ||||
|   // 插入@项 | ||||
|   mention.insertItem({ id: data?.id, denotationChar: '@', value: data.value }, true) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 处理引用事件 | ||||
|  * @param data 引用数据 | ||||
|  */ | ||||
| function onSubscribeQuote(data: any) { | ||||
|   // 检查是否已有引用内容 | ||||
|   const delta = getQuill().getContents() | ||||
|   if (delta.ops?.some((item: any) => item.insert.quote)) { | ||||
|     return | ||||
|     return  // 已有引用则不再添加 | ||||
|   } | ||||
| 
 | ||||
|   const quill = getQuill() | ||||
|   const index = getQuillSelectionIndex() | ||||
| 
 | ||||
|   // 在编辑器开头插入引用 | ||||
|   quill.insertEmbed(0, 'quote', data) | ||||
|   quill.setSelection(index + 1, 0, 'user') | ||||
|   quill.setSelection(index + 1, 0, 'user')  // 设置光标到引用后 | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 隐藏@成员DOM元素 | ||||
|  */ | ||||
| function hideMentionDom() { | ||||
|   let el = document.querySelector('.ql-mention-list-container') | ||||
|   if (el) { | ||||
| @ -449,27 +572,55 @@ function hideMentionDom() { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 处理编辑消息事件 | ||||
|  * @param data 消息数据 | ||||
|  */ | ||||
| function onSubscribeEdit(data: any) { | ||||
|   console.log('data', data) | ||||
|   const quill = getQuill() | ||||
|   if (!quill) return | ||||
|    | ||||
|   // 清空当前编辑器内容 | ||||
|   quill.setContents([]) | ||||
|    | ||||
|   // 插入要编辑的文本内容 | ||||
|   quill.setText(data.content) | ||||
|    | ||||
|   // 设置光标位置到末尾 | ||||
|   const index = quill.getLength() - 1 | ||||
|   quill.setSelection(index > 0 ? index : 0, 0, 'user') | ||||
| } | ||||
| 
 | ||||
| // 监听聊天索引变化,切换聊天时加载对应草稿 | ||||
| watch(indexName, loadEditorDraftText, { immediate: true }) | ||||
| 
 | ||||
| // 组件挂载时初始化 | ||||
| onMounted(() => { | ||||
|   loadEditorDraftText() | ||||
| }) | ||||
| 
 | ||||
| // 组件卸载时清理 | ||||
| onUnmounted(() => { | ||||
|   hideMentionDom() | ||||
| }) | ||||
| 
 | ||||
| // 订阅编辑器相关事件总线事件 | ||||
| useEventBus([ | ||||
|   { name: EditorConst.Mention, event: onSubscribeMention }, | ||||
|   { name: EditorConst.Quote, event: onSubscribeQuote } | ||||
|   { name: EditorConst.Mention, event: onSubscribeMention },  // @成员事件 | ||||
|   { name: EditorConst.Quote, event: onSubscribeQuote },       // 引用事件 | ||||
|   { name: EditorConst.Edit, event: onSubscribeEdit }          // 编辑消息事件 | ||||
| ]) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <!-- 编辑器容器 --> | ||||
|   <section class="el-container editor"> | ||||
|     <section class="el-container is-vertical"> | ||||
|       <!-- 工具栏区域 --> | ||||
|       <header class="el-header toolbar bdr-t"> | ||||
|         <div class="tools"> | ||||
|           <!-- 表情选择器弹出框 --> | ||||
|           <n-popover | ||||
|             placement="top-start" | ||||
|             trigger="click" | ||||
| @ -489,6 +640,7 @@ useEventBus([ | ||||
|             <MeEditorEmoticon @on-select="onEmoticonEvent" /> | ||||
|           </n-popover> | ||||
| 
 | ||||
|           <!-- 工具栏其他功能按钮 --> | ||||
|           <div | ||||
|             class="item pointer" | ||||
|             v-for="nav in navs" | ||||
| @ -502,6 +654,7 @@ useEventBus([ | ||||
|         </div> | ||||
|       </header> | ||||
| 
 | ||||
|       <!-- 编辑器主体区域 --> | ||||
|       <main class="el-main height100"> | ||||
|         <QuillEditor | ||||
|           ref="editor" | ||||
| @ -514,11 +667,13 @@ useEventBus([ | ||||
|     </section> | ||||
|   </section> | ||||
| 
 | ||||
|   <!-- 隐藏的文件上传表单 --> | ||||
|   <form enctype="multipart/form-data" style="display: none"> | ||||
|     <input type="file" ref="fileImageRef" accept="image/*" @change="onUploadFile" /> | ||||
|     <input type="file" ref="uploadFileRef" @change="onUploadFile" /> | ||||
|   </form> | ||||
| 
 | ||||
|   <!-- 条件渲染的功能组件 --> | ||||
|   <MeEditorVote v-if="isShowEditorVote" @close="isShowEditorVote = false" @submit="onVoteEvent" /> | ||||
| 
 | ||||
|   <MeEditorCode | ||||
| @ -536,7 +691,7 @@ useEventBus([ | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .editor { | ||||
|   --tip-bg-color: rgb(241 241 241 / 90%); | ||||
|   --tip-bg-color: rgb(241 241 241 / 90%);  /* 提示背景颜色 */ | ||||
| 
 | ||||
|   height: 100%; | ||||
| 
 | ||||
| @ -559,7 +714,7 @@ useEventBus([ | ||||
|         user-select: none; | ||||
| 
 | ||||
|         .tip-title { | ||||
|           display: none; | ||||
|           display: none;  /* 默认隐藏提示文字 */ | ||||
|           position: absolute; | ||||
|           top: 40px; | ||||
|           left: 0px; | ||||
| @ -577,7 +732,7 @@ useEventBus([ | ||||
| 
 | ||||
|         &:hover { | ||||
|           .tip-title { | ||||
|             display: block; | ||||
|             display: block;  /* 悬停时显示提示文字 */ | ||||
|           } | ||||
|         } | ||||
|       } | ||||
| @ -585,6 +740,7 @@ useEventBus([ | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* 暗色模式样式调整 */ | ||||
| html[theme-mode='dark'] { | ||||
|   .editor { | ||||
|     --tip-bg-color: #48484d; | ||||
| @ -593,13 +749,16 @@ html[theme-mode='dark'] { | ||||
| </style> | ||||
| 
 | ||||
| <style lang="less"> | ||||
| /* 全局编辑器样式 */ | ||||
| #editor { | ||||
|   overflow: hidden; | ||||
| } | ||||
| 
 | ||||
| /* 编辑器主体区域样式 */ | ||||
| .ql-editor { | ||||
|   padding: 8px; | ||||
| 
 | ||||
|   /* 滚动条样式 */ | ||||
|   &::-webkit-scrollbar { | ||||
|     width: 3px; | ||||
|     height: 3px; | ||||
| @ -611,6 +770,7 @@ html[theme-mode='dark'] { | ||||
|     background-color: transparent; | ||||
|   } | ||||
| 
 | ||||
|   /* 悬停时显示滚动条 */ | ||||
|   &:hover { | ||||
|     &::-webkit-scrollbar-thumb { | ||||
|       background-color: var(--im-scrollbar-thumb); | ||||
| @ -618,6 +778,7 @@ html[theme-mode='dark'] { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* 编辑器占位符样式 */ | ||||
| .ql-editor.ql-blank::before { | ||||
|   font-family: | ||||
|     PingFang SC, | ||||
| @ -626,6 +787,7 @@ html[theme-mode='dark'] { | ||||
|   left: 8px; | ||||
| } | ||||
| 
 | ||||
| /* 编辑器中图片样式 */ | ||||
| .ql-snow .ql-editor img { | ||||
|   max-width: 100px; | ||||
|   border-radius: 3px; | ||||
| @ -633,6 +795,7 @@ html[theme-mode='dark'] { | ||||
|   margin: 0px 2px; | ||||
| } | ||||
| 
 | ||||
| /* 图片上传中样式 */ | ||||
| .image-uploading { | ||||
|   display: flex; | ||||
|   width: 100px; | ||||
| @ -646,15 +809,18 @@ html[theme-mode='dark'] { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* 表情符号样式 */ | ||||
| .ed-emoji { | ||||
|   background-color: unset !important; | ||||
| } | ||||
| 
 | ||||
| /* 编辑器占位符样式 */ | ||||
| .ql-editor.ql-blank::before { | ||||
|   font-style: unset; | ||||
|   color: #b8b3b3; | ||||
| } | ||||
| 
 | ||||
| /* 引用卡片样式 */ | ||||
| .quote-card-content { | ||||
|   display: flex; | ||||
|   background-color: #f6f6f6; | ||||
| @ -691,6 +857,7 @@ html[theme-mode='dark'] { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* 暗色模式下的样式调整 */ | ||||
| html[theme-mode='dark'] { | ||||
|   .ql-editor.ql-blank::before { | ||||
|     color: #57575a; | ||||
|  | ||||
| @ -48,10 +48,10 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | ||||
|     <input type="file" ref="fileImageRef" accept="image/*" @change="onUpload" /> | ||||
|   </form> | ||||
| 
 | ||||
|   <section class="el-container is-vertical section height100"> | ||||
|     <header class="el-header em-header bdr-b"> | ||||
|   <section class="el-container is-vertical section height100 p-10px"> | ||||
|     <!-- <header class="el-header em-header bdr-b"> | ||||
|       <span>{{ items[tabIndex].name }}</span> | ||||
|     </header> | ||||
|     </header> --> | ||||
| 
 | ||||
|     <main class="el-main em-main me-scrollbar me-scrollbar-thumb"> | ||||
|       <div class="symbol-box" v-if="tabIndex == 0"> | ||||
| @ -82,7 +82,7 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | ||||
|       </div> | ||||
|     </main> | ||||
| 
 | ||||
|     <footer class="el-footer em-footer tabs"> | ||||
|     <!-- <footer class="el-footer em-footer tabs"> | ||||
|       <div | ||||
|         class="tab pointer" | ||||
|         v-for="(item, index) in items" | ||||
| @ -93,7 +93,7 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | ||||
|         <p class="tip">{{ item.name }}</p> | ||||
|         <img width="20" height="20" :src="item.icon" /> | ||||
|       </div> | ||||
|     </footer> | ||||
|     </footer> --> | ||||
|   </section> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| @ -185,17 +185,18 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | ||||
|       display: flex; | ||||
|       flex-wrap: wrap; | ||||
| 
 | ||||
|       .option { | ||||
|         height: 32px; | ||||
|         width: 32px; | ||||
|         margin: 2px; | ||||
|         font-size: 24px; | ||||
|       .option{ | ||||
|         margin: 7px; | ||||
|         :deep(.emoji){ | ||||
|           height: 22px; | ||||
|         width: 22px; | ||||
|         user-select: none; | ||||
|         transition: all 0.5s; | ||||
| 
 | ||||
|         &:hover { | ||||
|           transform: scale(1.5); | ||||
|         } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @ -3,7 +3,7 @@ import { ref, reactive } from 'vue' | ||||
| import { PlayOne, PauseOne } from '@icon-park/vue-next' | ||||
| import { ITalkRecordExtraAudio, ITalkRecord } from '@/types/chat' | ||||
| 
 | ||||
| defineProps<{ | ||||
| const props = defineProps<{ | ||||
|   extra: ITalkRecordExtraAudio | ||||
|   data: ITalkRecord | ||||
|   maxWidth?: Boolean | ||||
| @ -18,7 +18,8 @@ const state = reactive({ | ||||
|   progress: 0, | ||||
|   duration: 0, | ||||
|   currentTime: 0, | ||||
|   loading: true | ||||
|   loading: true, | ||||
|   showText: false | ||||
| }) | ||||
| 
 | ||||
| const onPlay = () => { | ||||
| @ -40,6 +41,12 @@ const onCanplay = () => { | ||||
|   state.duration = audioRef.value.duration | ||||
|   durationDesc.value = formatTime(parseInt(audioRef.value.duration)) | ||||
|   state.loading = false | ||||
|    | ||||
|   if (props.data.is_convert_text === 1 && props.data.extra.content) { | ||||
|     setTimeout(() => { | ||||
|       state.showText = true | ||||
|     }, 300) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const onError = (e: any) => { | ||||
| @ -61,17 +68,12 @@ const formatTime = (value: number = 0) => { | ||||
|     return '-' | ||||
|   } | ||||
| 
 | ||||
|   const minutes = Math.floor(value / 60) | ||||
|   let seconds = value | ||||
|   if (minutes > 0) { | ||||
|     seconds = Math.floor(value - minutes * 60) | ||||
|   } | ||||
| 
 | ||||
|   return `${minutes}'${seconds}"` | ||||
|   return `${Math.floor(value)}"` | ||||
| } | ||||
| </script> | ||||
| <template> | ||||
|   <div class="im-message-audio"> | ||||
|   <div class="pointer w-200px bg-#f5f5f5 rounded-10px px-11px"> | ||||
|     <div class="im-message-audio h-44px"> | ||||
|     <audio | ||||
|       ref="audioRef" | ||||
|       preload="auto" | ||||
| @ -98,20 +100,27 @@ const formatTime = (value: number = 0) => { | ||||
|     </div> | ||||
|     <div class="time">{{ durationDesc }}</div> | ||||
|   </div> | ||||
|    | ||||
|   <transition name="expand"> | ||||
|     <div class="text-container py-12px border-t-2px border-t-solid border-t-#E0E0E4" v-if="data.is_convert_text===1"> | ||||
|       <div class="flex justify-center items-center" v-if="data.is_convert_text===1&&!data.extra.content"> | ||||
|         <n-spin :stroke-width="3" size="small" /> | ||||
|       </div> | ||||
|       <transition name="fade"> | ||||
|         <div class="text-content" v-if="data.extra.content">{{ data.extra.content }}</div> | ||||
|       </transition> | ||||
|     </div> | ||||
|   </transition> | ||||
|   </div> | ||||
| 
 | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| .im-message-audio { | ||||
|   --audio-bg-color: #f5f5f5; | ||||
|   --audio-btn-bg-color: #ffffff; | ||||
| 
 | ||||
|   width: 200px; | ||||
|   height: 45px; | ||||
|   border-radius: 10px; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   overflow: hidden; | ||||
|   background-color: var(--audio-bg-color); | ||||
| 
 | ||||
|   > div { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
| @ -132,6 +141,7 @@ const formatTime = (value: number = 0) => { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       transition: all 0.3s ease; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| @ -230,6 +240,7 @@ const formatTime = (value: number = 0) => { | ||||
|       height: 70%; | ||||
|       width: 1px; | ||||
|       background-color: #9b9595; | ||||
|       transition: left 0.1s linear; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| @ -241,6 +252,40 @@ const formatTime = (value: number = 0) => { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .expand-enter-active, | ||||
| .expand-leave-active { | ||||
|   transition: all 0.5s ease; | ||||
|   overflow: hidden; | ||||
| } | ||||
| 
 | ||||
| .expand-enter-from, | ||||
| .expand-leave-to { | ||||
|   max-height: 0; | ||||
|   opacity: 0; | ||||
|   padding: 0; | ||||
|   border-top-width: 0; | ||||
| } | ||||
| 
 | ||||
| .fade-enter-active, | ||||
| .fade-leave-active { | ||||
|   transition: opacity 0.2s ease; | ||||
| } | ||||
| 
 | ||||
| .fade-enter-from, | ||||
| .fade-leave-to { | ||||
|   opacity: 0; | ||||
| } | ||||
| 
 | ||||
| .text-container { | ||||
|   overflow: hidden; | ||||
|   transition: all 0.2s ease; | ||||
| } | ||||
| 
 | ||||
| .text-content { | ||||
|   line-height: 1.5; | ||||
|   transition: all 0.2s ease; | ||||
| } | ||||
| 
 | ||||
| html[theme-mode='dark'] { | ||||
|   .im-message-audio { | ||||
|     --audio-bg-color: #2c2c32; | ||||
|  | ||||
| @ -1,118 +1,222 @@ | ||||
| <script lang="ts" setup> | ||||
| <script setup> | ||||
| import { fileFormatSize } from '@/utils/strings' | ||||
| import { download, getFileNameSuffix } from '@/utils/functions' | ||||
| import { ITalkRecordExtraFile, ITalkRecord } from '@/types/chat' | ||||
| import { ref, computed } from 'vue' | ||||
| import { useUploadsStore } from '@/store' | ||||
| import pptText from '@/assets/image/ppt-text.png' | ||||
| import excelText from '@/assets/image/excel-text.png' | ||||
| import wordText from '@/assets/image/word-text.png' | ||||
| import pdfText from '@/assets/image/pdf-text.png' | ||||
| import fileText from '@/assets/image/file-text.png' | ||||
| 
 | ||||
| defineProps<{ | ||||
|   extra: ITalkRecordExtraFile | ||||
|   data: ITalkRecord | ||||
|   maxWidth?: Boolean | ||||
| }>() | ||||
| // 定义组件属性 | ||||
| const props = defineProps({ | ||||
|   // 文件的额外信息 | ||||
|   extra: { | ||||
|     type: Object, | ||||
|     required: true | ||||
|   }, | ||||
|   // 聊天记录数据 | ||||
|   data: { | ||||
|     type: Object, | ||||
|     required: true | ||||
|   }, | ||||
|   // 是否使用最大宽度 | ||||
|   maxWidth: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const uploadsStore = useUploadsStore() | ||||
| const isPlaying = ref(false) | ||||
| 
 | ||||
| // 文件类型配置 | ||||
| const fileTypes = { | ||||
|   PDF: { icon: pdfText, color: '#DE4E4E', type: 'PDF' }, | ||||
|   PPT: { icon: pptText, color: '#B74B2B', type: 'PPT' }, | ||||
|   EXCEL: { icon: excelText, color: '#3C7F4B', type: 'EXCEL' }, | ||||
|   WORD: { icon: wordText, color: '#2750B2', type: 'WORD' }, | ||||
|   DEFAULT: { icon: fileText, color: '#747474', type: '文件' } | ||||
| } | ||||
| 
 | ||||
| // Excel文件扩展名映射 | ||||
| const EXCEL_EXTENSIONS = ['XLS', 'XLSX', 'CSV'] | ||||
| // Word文件扩展名映射 | ||||
| const WORD_EXTENSIONS = ['DOC', 'DOCX', 'RTF', 'DOT', 'DOTX'] | ||||
| // PPT文件扩展名映射 | ||||
| const PPT_EXTENSIONS = ['PPT', 'PPTX', 'PPS', 'PPSX'] | ||||
| 
 | ||||
| // 获取文件类型信息 | ||||
| const fileInfo = computed(() => { | ||||
|   const extension = getFileExtension(props.extra.name) | ||||
|   if (EXCEL_EXTENSIONS.includes(extension)) { | ||||
|     return fileTypes.EXCEL | ||||
|   } | ||||
|   if (WORD_EXTENSIONS.includes(extension)) { | ||||
|     return fileTypes.WORD | ||||
|   } | ||||
|   if (PPT_EXTENSIONS.includes(extension)) { | ||||
|     return fileTypes.PPT | ||||
|   } | ||||
|   return fileTypes[extension] || fileTypes.DEFAULT | ||||
| }) | ||||
| 
 | ||||
| // 获取文件扩展名 | ||||
| function getFileExtension(filename) { | ||||
|   const parts = filename.split('.') | ||||
|   return parts.length > 1 ? parts.pop().toUpperCase() : '' | ||||
| } | ||||
| 
 | ||||
| // 切换播放状态 | ||||
| const togglePlay = () => { | ||||
|   isPlaying.value = !isPlaying.value | ||||
|    | ||||
|   if (props.extra.is_uploading && props.extra.upload_id) { | ||||
|     const action = isPlaying.value ? 'resumeUpload' : 'pauseUpload' | ||||
|     uploadsStore[action](props.extra.upload_id) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 计算SVG圆环进度条的参数 | ||||
| const radius = 9 | ||||
| const circumference = computed(() => 2 * Math.PI * radius) | ||||
| const strokeDashoffset = computed(() =>  | ||||
|   circumference.value * (1 - (props.extra.percentage || 0) / 100) | ||||
| ) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <section class="file-message"> | ||||
|     <div class="main"> | ||||
|       <div class="ext">{{ getFileNameSuffix(extra.name) }}</div> | ||||
|       <div class="file-box"> | ||||
|         <p class="info"> | ||||
|           <span class="name">{{ extra.name }}</span> | ||||
|           <span class="size">({{ fileFormatSize(extra.size) }})</span> | ||||
|         </p> | ||||
|         <p class="notice">文件已成功发送, 文件助手永久保存</p> | ||||
|   <div class="file-message"> | ||||
|     <!-- 文件头部信息 --> | ||||
|     <div class="file-header"> | ||||
|       <!-- 文件名 --> | ||||
|       <div class="file-name">{{ extra.name }}</div> | ||||
|       <!-- 文件图标区域 --> | ||||
|       <div  class="file-icon-container"> | ||||
|         <img class="file-icon" :src="fileInfo.icon" alt="文件图标"> | ||||
|              | ||||
|         <!-- 上传进度圆环 - 上传状态 --> | ||||
|         <div v-if="extra.is_uploading" class="progress-overlay"> | ||||
|           <div class="circle-progress-container" @click="togglePlay"> | ||||
|             <svg class="circle-progress" width="20" height="20" viewBox="0 0 20 20"> | ||||
|               <!-- 底色圆环 --> | ||||
|               <circle  | ||||
|                 cx="10"  | ||||
|                 cy="10"  | ||||
|                 r="9" | ||||
|                 fill="transparent" | ||||
|                 stroke="#EEEEEE" | ||||
|                 stroke-width="2" | ||||
|               /> | ||||
|               <!-- 进度圆环 --> | ||||
|               <circle  | ||||
|                 cx="10"  | ||||
|                 cy="10"  | ||||
|                 r="9" | ||||
|                 fill="transparent" | ||||
|                 :stroke="fileInfo.color" | ||||
|                 stroke-width="2" | ||||
|                 :stroke-dasharray="circumference" | ||||
|                 :stroke-dashoffset="strokeDashoffset" | ||||
|                 transform="rotate(-90 10 10)" | ||||
|                 class="progress-circle" | ||||
|               /> | ||||
|                | ||||
|               <!-- 暂停/播放图标 --> | ||||
|               <g v-if="isPlaying" class="pause-icon"> | ||||
|                 <rect x="7" y="5" width="2" height="10" :fill="fileInfo.color" /> | ||||
|                 <rect x="11" y="5" width="2" height="10" :fill="fileInfo.color" /> | ||||
|               </g> | ||||
|               <g v-else class="play-icon"> | ||||
|                 <rect x="6" y="6" width="8" height="8" :fill="fileInfo.color" /> | ||||
|               </g> | ||||
|             </svg> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="footer"> | ||||
|       <a @click="download(data.msg_id)">下载</a> | ||||
|       <a>在线预览</a> | ||||
|     </div> | ||||
|   </section> | ||||
|     <!-- 文件大小信息 --> | ||||
|     <div class="file-size">{{ fileFormatSize(extra.size) }}</div> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .file-message { | ||||
|   width: 250px; | ||||
|   min-height: 85px; | ||||
|   padding: 10px; | ||||
|   border-radius: 10px; | ||||
|   border: 1px solid var(--im-message-border-color); | ||||
|   width: 243px; | ||||
|   background-color: #fff; | ||||
|   border-radius: 8px; | ||||
|   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); | ||||
|   padding: 0 14px; | ||||
|   cursor: pointer; | ||||
| } | ||||
| 
 | ||||
|   .main { | ||||
|     height: 45px; | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     margin-top: 5px; | ||||
| .file-header { | ||||
|   display: flex; | ||||
|   padding: 14px 5px 14px 0; | ||||
|   justify-content: space-between; | ||||
|   width: 100%; | ||||
|   border-bottom: 1px solid #EEEEEE; | ||||
| } | ||||
| 
 | ||||
|     .ext { | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|       align-items: center; | ||||
|       width: 45px; | ||||
|       height: 45px; | ||||
|       color: #ffffff; | ||||
|       background: #49a4ff; | ||||
|       border-radius: 5px; | ||||
|       font-size: 12px; | ||||
|     } | ||||
| .file-name { | ||||
|   color: #1A1A1A; | ||||
|   font-size: 14px; | ||||
|   word-break: break-word; | ||||
|   overflow: hidden; | ||||
|   text-overflow: ellipsis; | ||||
|   display: -webkit-box; | ||||
|   -webkit-line-clamp: 2; | ||||
|   -webkit-box-orient: vertical; | ||||
| } | ||||
| 
 | ||||
|     .file-box { | ||||
|       flex: 1 1; | ||||
|       height: 45px; | ||||
|       margin-left: 10px; | ||||
|       overflow: hidden; | ||||
| .file-icon-container { | ||||
|   position: relative; | ||||
| } | ||||
| 
 | ||||
|       .info { | ||||
|         display: flex; | ||||
|         justify-content: space-between; | ||||
|         align-items: center; | ||||
|         overflow: hidden; | ||||
|         height: 24px; | ||||
|         font-size: 14px; | ||||
| .file-icon { | ||||
|   width: 48px; | ||||
|   height: 48px; | ||||
| } | ||||
| 
 | ||||
|         .name { | ||||
|           flex: 1 auto; | ||||
|           white-space: nowrap; | ||||
|           overflow: hidden; | ||||
|           text-overflow: ellipsis; | ||||
|         } | ||||
| .progress-overlay { | ||||
|   background-color: #fff; | ||||
|   position: absolute; | ||||
|   top: 6px; | ||||
|   left: 11px; | ||||
|   width: 30px; | ||||
|   height: 30px; | ||||
|   display: flex; | ||||
|   justify-content: center; | ||||
|   align-items: center; | ||||
| } | ||||
| 
 | ||||
|         .size { | ||||
|           font-size: 12px; | ||||
|           color: #cac6c6; | ||||
|           flex-shrink: 0; | ||||
|         } | ||||
|       } | ||||
| .file-size { | ||||
|   color: #747474; | ||||
|   font-size: 12px; | ||||
|   padding: 5px 0 11px; | ||||
| } | ||||
| 
 | ||||
|       .notice { | ||||
|         height: 25px; | ||||
|         line-height: 25px; | ||||
|         font-size: 12px; | ||||
|         color: #929191; | ||||
|         white-space: nowrap; | ||||
|         overflow: hidden; | ||||
|         text-overflow: ellipsis; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| .circle-progress-container { | ||||
|   width: 20px; | ||||
|   height: 20px; | ||||
|   position: relative; | ||||
|   cursor: pointer; | ||||
| } | ||||
| 
 | ||||
|   .footer { | ||||
|     height: 30px; | ||||
|     line-height: 37px; | ||||
|     text-align: right; | ||||
|     font-size: 12px; | ||||
|     border-top: 1px solid var(--border-color); | ||||
|     margin-top: 10px; | ||||
| .circle-progress { | ||||
|   transform: rotate(-90deg); | ||||
|   transform-origin: center; | ||||
| } | ||||
| 
 | ||||
|     a { | ||||
|       margin: 0 3px; | ||||
|       user-select: none; | ||||
|       cursor: pointer; | ||||
|       color: var(--im-text-color); | ||||
| .progress-circle { | ||||
|   transition: stroke-dashoffset 0.3s ease; | ||||
| } | ||||
| 
 | ||||
|       &:hover { | ||||
|         color: royalblue; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| .pause-icon, .play-icon { | ||||
|   transform-origin: center; | ||||
| } | ||||
| 
 | ||||
| .pause-icon { | ||||
|   transform: rotate(90deg); | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -46,7 +46,8 @@ const img = (src: string, width = 200) => { | ||||
|   background: var(--im-message-left-bg-color); | ||||
|   min-width: 30px; | ||||
|   min-height: 30px; | ||||
| 
 | ||||
|   max-width:240px; | ||||
|   height:149px | ||||
|   &.left { | ||||
|     background: var(--im-message-right-bg-color); | ||||
|   } | ||||
|  | ||||
| @ -1,7 +1,9 @@ | ||||
| <script setup> | ||||
| import { formatTime } from '@/utils/datetime' | ||||
| import { bus } from '@/utils/event-bus' | ||||
| import { EditorConst } from '@/constant/event-bus' | ||||
| 
 | ||||
| defineProps({ | ||||
| const props = defineProps({ | ||||
|   login_uid: { | ||||
|     type: Number, | ||||
|     default: 0 | ||||
| @ -21,13 +23,30 @@ defineProps({ | ||||
|   datetime: { | ||||
|     type: String, | ||||
|     default: '' | ||||
|   }, | ||||
|   data: { | ||||
|     type: Object, | ||||
|     default: () => {} | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const onRevoke = () => { | ||||
|   // 只处理文本消息 | ||||
|   if (props.data.msg_type === 1 && props.data.extra?.content) { | ||||
|     // 通过事件总线发送编辑消息事件 | ||||
|     bus.emit(EditorConst.Edit, { | ||||
|       content: props.data.extra.content | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| <template> | ||||
|   <div class="im-message-revoke"> | ||||
|     <div class="content"> | ||||
|       <span v-if="login_uid == user_id"> 你撤回了一条消息 | {{ formatTime(datetime) }} </span> | ||||
|       <div v-if="login_uid === user_id"> | ||||
|         <span> 你撤回了一条消息 | {{ formatTime(datetime) }} </span> | ||||
|         <n-button @click="onRevoke" v-if="data.msg_type === 1&&data.extra?.content" text class="text-#46299D text-11px">重新编辑</n-button> | ||||
|       </div> | ||||
|       <span v-else-if="talk_type == 1"> 对方撤回了一条消息 | {{ formatTime(datetime) }} </span> | ||||
|       <span v-else> | ||||
|         "{{ nickname }}" 撤回了一条消息 | | ||||
|  | ||||
| @ -45,7 +45,7 @@ textContent = textReplaceEmoji(textContent) | ||||
|   color: var(--im-message-left-text-color); | ||||
|   background: var(--im-message-left-bg-color); | ||||
|   border-radius: 0px 10px 10px 10px; | ||||
| 
 | ||||
|   font-size: 14px; | ||||
|   &.right { | ||||
|     background-color: var(--im-message-right-bg-color); | ||||
|     color: var(--im-message-right-text-color); | ||||
|  | ||||
| @ -1,11 +1,17 @@ | ||||
| <script lang="ts" setup> | ||||
| import 'xgplayer/dist/index.min.css' | ||||
| import { ref, nextTick } from 'vue' | ||||
| import { NImage, NModal, NCard } from 'naive-ui' | ||||
| import { Play, Close } from '@icon-park/vue-next' | ||||
| import { ref, nextTick, watch } from 'vue' | ||||
| import { NImage, NModal, NCard, NProgress, NPopconfirm } from 'naive-ui' | ||||
| import { Play, Close, Pause, Right, Attention } from '@icon-park/vue-next' | ||||
| import { getImageInfo } from '@/utils/functions' | ||||
| import {PauseOutline} from '@vicons/ionicons5' | ||||
| import Player from 'xgplayer' | ||||
| import { ITalkRecordExtraVideo, ITalkRecord } from '@/types/chat' | ||||
| import { useUploadsStore } from '@/store' | ||||
| // @ts-ignore | ||||
| const message = window.$message | ||||
| 
 | ||||
| const uploadsStore = useUploadsStore() | ||||
| 
 | ||||
| const props = defineProps<{ | ||||
|   extra: ITalkRecordExtraVideo | ||||
| @ -40,8 +46,43 @@ const img = (src: string, width = 200) => { | ||||
| } | ||||
| 
 | ||||
| const open = ref(false) | ||||
| const isPaused = ref(false) | ||||
| const uploadFailed = ref(false) | ||||
| 
 | ||||
| // 查找上传项并检查状态 | ||||
| const updatePauseStatus = () => { | ||||
|   if (props.extra.is_uploading && props.extra.upload_id) { | ||||
|     // 使用新的查找方法 | ||||
|     const item = uploadsStore.findItemByClientId(props.extra.upload_id) | ||||
|      | ||||
|     if (item && item.is_paused !== undefined) { | ||||
|       isPaused.value = item.is_paused | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 初始化时检查状态 | ||||
| updatePauseStatus() | ||||
| 
 | ||||
| // 监听关键道具变化 | ||||
| watch(() => props.extra.percentage, (newVal: number | undefined) => { | ||||
|   // 确保进度更新时 UI 也实时更新   | ||||
|   // 检测上传失败状态 (-1表示上传失败) | ||||
|   if (newVal === -1) { | ||||
|     uploadFailed.value = true | ||||
|     // 显示上传失败提示 | ||||
|     message.error('视频发送失败,请点击红色感叹号重试') | ||||
|   } else if (newVal !== undefined && newVal > 0) { | ||||
|     uploadFailed.value = false | ||||
|   } | ||||
| }, { immediate: true }) | ||||
| 
 | ||||
| async function onPlay() { | ||||
|   // 如果视频正在上传,不执行播放操作 | ||||
|   if (props.extra.is_uploading) { | ||||
|     return | ||||
|   } | ||||
|    | ||||
|   open.value = true | ||||
| 
 | ||||
|   await nextTick() | ||||
| @ -54,18 +95,85 @@ async function onPlay() { | ||||
|     lang: 'zh-cn' | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 暂停上传 | ||||
| function pauseUpload(e) { | ||||
|   e.stopPropagation() | ||||
|   if (props.extra.is_uploading && props.extra.upload_id) { | ||||
|     uploadsStore.pauseUpload(props.extra.upload_id) | ||||
|     isPaused.value = true | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 继续上传 | ||||
| function resumeUpload(e) { | ||||
|   console.log('resumeUpload') | ||||
|   e.stopPropagation() | ||||
|   if (props.extra.is_uploading && props.extra.upload_id) { | ||||
|     uploadsStore.resumeUpload(props.extra.upload_id) | ||||
|     isPaused.value = false | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 重新上传视频 | ||||
| function retryUpload(e) { | ||||
|   e.stopPropagation() | ||||
|   if (props.extra.upload_id) { | ||||
|     // 重置失败状态 | ||||
|     uploadFailed.value = false | ||||
|      | ||||
|     // 恢复上传 | ||||
|     uploadsStore.resumeUpload(props.extra.upload_id) | ||||
|     message.success('正在重新上传视频...') | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| <template> | ||||
|   <section | ||||
|     class="im-message-video" | ||||
|     :class="{ left: data.float === 'left' }" | ||||
|     :style="img(extra.cover, 350)" | ||||
|     @click="onPlay" | ||||
|   > | ||||
|     <n-image :src="extra.cover" preview-disabled /> | ||||
|    | ||||
|     <div class="btn-video"> | ||||
|       <n-icon :component="Play" size="36" /> | ||||
|     <!-- <n-image :src="extra.cover" preview-disabled /> --> | ||||
|     <video :src="props.extra.url" :controls="false"></video> | ||||
|      | ||||
|     <!-- 上传进度显示 --> | ||||
|     <div v-if="extra.is_uploading && !uploadFailed" class="upload-progress"> | ||||
|       <n-progress | ||||
|    | ||||
|         type="circle" | ||||
|         :percentage="Math.round(extra.percentage || 0)" | ||||
|         :show-indicator="false" | ||||
|         :stroke-width="6" | ||||
|         color="#fff" | ||||
|         rail-color="#E3E3E3" | ||||
|       /> | ||||
|        | ||||
|       <!-- 暂停/继续按钮移到圆圈内部 --> | ||||
|       <div class="upload-control" @click.stop> | ||||
|         <n-icon  | ||||
|           v-if="!isPaused"  | ||||
|           class="control-btn"  | ||||
|           :component="PauseOutline"  | ||||
|           size="20"  | ||||
|           @click="pauseUpload"  | ||||
|         /> | ||||
|         <div v-else class="w-15px h-15px bg-#fff rounded-4px"  @click="resumeUpload" > | ||||
| 
 | ||||
|         </div> | ||||
|         <!-- <n-icon  | ||||
|           v-else  | ||||
|           class="control-btn"  | ||||
|           :component="Right"  | ||||
|           size="20"  | ||||
|           @click="resumeUpload"  | ||||
|         /> --> | ||||
|       </div> | ||||
|     </div> | ||||
|     <!-- 播放按钮,仅在视频不是上传状态且未失败时显示 --> | ||||
|     <div v-if="!extra.is_uploading && !uploadFailed" class="btn-video"> | ||||
|       <n-icon :component="Play" size="40" /> | ||||
|     </div> | ||||
| 
 | ||||
|     <n-modal v-model:show="open"> | ||||
| @ -92,23 +200,25 @@ async function onPlay() { | ||||
|   min-height: 30px; | ||||
|   display: inline-flex; | ||||
|   position: relative; | ||||
| 
 | ||||
|   height:149px; | ||||
|   width: 225px; | ||||
|   &.left { | ||||
|     background: var(--im-message-right-bg-color); | ||||
|   } | ||||
| 
 | ||||
|   :deep(.n-image img) { | ||||
|   video { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     border-radius: 5px; | ||||
|     object-fit: cover; | ||||
|     background-color: #333; /* 添加背景色,避免默认显示为灰色 */ | ||||
|   } | ||||
| 
 | ||||
|   .btn-video { | ||||
|     width: 30px; | ||||
|     height: 20px; | ||||
|     left: 50%; | ||||
|     top: 50%; | ||||
|     transform: translate(-50%, -50%); | ||||
|     position: absolute; | ||||
|     left: calc(50% - 15px); | ||||
|     top: calc(50% - 10px); | ||||
|     cursor: pointer; | ||||
|     color: #ffffff; | ||||
|   } | ||||
| @ -134,4 +244,54 @@ async function onPlay() { | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
| } | ||||
| 
 | ||||
| .upload-progress { | ||||
|   position: absolute; | ||||
|   left: 50%; | ||||
|   top: 50%; | ||||
|   transform: translate(-50%, -50%); | ||||
|   width: 40px; | ||||
|   height: 40px; | ||||
|   display: flex; | ||||
|   justify-content: center; | ||||
|   align-items: center; | ||||
|    | ||||
|   .upload-control { | ||||
|     position: absolute; | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
|     align-items: center; | ||||
|     cursor: pointer; | ||||
|      | ||||
|     .control-btn { | ||||
|       color: white; | ||||
|       z-index: 2; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /* 上传失败样式 */ | ||||
| .upload-failed { | ||||
|   position: absolute; | ||||
|   left: 10px; | ||||
|   bottom: 10px; | ||||
|   z-index: 2; | ||||
|    | ||||
|   .failed-icon { | ||||
|     width: 30px; | ||||
|     height: 30px; | ||||
|     background-color: rgba(0, 0, 0, 0.7); | ||||
|     border-radius: 50%; | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
|     align-items: center; | ||||
|     cursor: pointer; | ||||
|      | ||||
|     &:hover { | ||||
|       background-color: rgba(0, 0, 0, 0.9); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -1,12 +1,10 @@ | ||||
| <script lang="ts" setup> | ||||
| import { ref, computed } from 'vue' | ||||
| import { NModal, NInput, NScrollbar, NCheckbox, NTabs, NTab } from 'naive-ui' | ||||
| import { Search, Delete } from '@icon-park/vue-next' | ||||
| import { ref, computed, onMounted, watch } from 'vue' | ||||
| import { ServeGetContacts } from '@/api/contact' | ||||
| import { ServeGetGroups } from '@/api/group' | ||||
| 
 | ||||
| import XNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | ||||
| const emit = defineEmits(['close', 'on-submit']) | ||||
| 
 | ||||
| import { CloseCircle } from '@vicons/ionicons5' | ||||
| interface Item { | ||||
|   id: number | ||||
|   type: number | ||||
| @ -17,16 +15,18 @@ interface Item { | ||||
|   keyword: string | ||||
| } | ||||
| 
 | ||||
| const tabsIndex = ref<number>(1) | ||||
| const isShowBox = ref(true) | ||||
| const loading = ref(true) | ||||
| const items = ref<Item[]>([]) | ||||
| const keywords = ref('') | ||||
| const loadGroupStatus = ref(false) | ||||
| 
 | ||||
|  defineProps<{ | ||||
|   forwardMode: number | ||||
| }>() | ||||
| // 搜索过滤器:不再按类型过滤,将好友和群组融合在一起 | ||||
| const searchFilter = computed(() => { | ||||
|   return items.value.filter((item: Item) => { | ||||
|     return tabsIndex.value == item.type && item.keyword.match(keywords.value) != null | ||||
|     return item.keyword.toLowerCase().includes(keywords.value.toLowerCase()) | ||||
|   }) | ||||
| }) | ||||
| 
 | ||||
| @ -40,6 +40,7 @@ const isCanSubmit = computed(() => { | ||||
| 
 | ||||
| const onLoad = () => { | ||||
|   onLoadContact() | ||||
|   onLoadGroup() | ||||
| } | ||||
| 
 | ||||
| const onLoadContact = () => { | ||||
| @ -55,7 +56,7 @@ const onLoadContact = () => { | ||||
|             avatar: item.avatar, | ||||
|             type: 1, | ||||
|             name: item.remark || item.nickname, | ||||
|             keyword: item.remark + item.nickname, | ||||
|             keyword: (item.remark || '') + item.nickname, | ||||
|             remark: item.remark, | ||||
|             checked: false | ||||
|           } | ||||
| @ -75,6 +76,7 @@ const onLoadGroup = async () => { | ||||
|   loading.value = true | ||||
|   let { code, data } = await ServeGetGroups() | ||||
|   if (code != 200) { | ||||
|     loading.value = false | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
| @ -101,6 +103,13 @@ const onMaskClick = () => { | ||||
| } | ||||
| 
 | ||||
| const onTriggerContact = (item: any) => { | ||||
|   // 如果是单选模式,先取消所有选中 | ||||
|   if (selectType.value === 1) { | ||||
|     items.value.forEach(contact => { | ||||
|       contact.checked = false | ||||
|     }) | ||||
|   } | ||||
|    | ||||
|   let data = items.value.find((val: any) => val.id === item.id) | ||||
| 
 | ||||
|   if (data) { | ||||
| @ -108,6 +117,18 @@ const onTriggerContact = (item: any) => { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const onRemoveContact = (item: any) => { | ||||
|   let data = items.value.find((val: any) => val.id === item.id) | ||||
|    | ||||
|   if (data) { | ||||
|     data.checked = false | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const onCancel = () => { | ||||
|   isShowBox.value = false | ||||
| } | ||||
| 
 | ||||
| const onSubmit = () => { | ||||
|   let data = checkedFilter.value.map((item: any) => { | ||||
|     return { | ||||
| @ -119,219 +140,110 @@ const onSubmit = () => { | ||||
|   emit('on-submit', data) | ||||
| } | ||||
| 
 | ||||
| const onTabs = (value: number) => { | ||||
|   tabsIndex.value = value | ||||
|   if (value == 2) { | ||||
|     onLoadGroup() | ||||
|   } | ||||
| // 1 单选 2 多选 | ||||
| const selectType = ref(1) | ||||
| const changeSelectType = () => { | ||||
|   selectType.value = selectType.value == 1 ? 2 : 1 | ||||
|    | ||||
|   // 切换选择模式时清空已选择的联系人 | ||||
|   items.value.forEach(item => { | ||||
|     item.checked = false | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| onLoad() | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <n-modal | ||||
|     v-model:show="isShowBox" | ||||
|     preset="card" | ||||
|     title="选择联系人" | ||||
|     class="modal-radius" | ||||
|     style="max-width: 650px; height: 550px" | ||||
|     :on-after-leave="onMaskClick" | ||||
|     :segmented="{ | ||||
|       content: true, | ||||
|       footer: true | ||||
|     }" | ||||
|     :content-style="{ | ||||
|       padding: 0 | ||||
|     }" | ||||
|   > | ||||
|     <section class="el-container launch-box"> | ||||
|       <aside class="el-aside bdr-r" style="width: 240px"> | ||||
|         <section class="el-container is-vertical height100"> | ||||
|           <header class="el-header tabs"> | ||||
|             <n-tabs type="line" justify-content="space-around" @update:value="onTabs"> | ||||
|               <n-tab name="1"> 好友 </n-tab> | ||||
|               <n-tab name="2"> 群聊 </n-tab> | ||||
|               <!-- <n-tab name="企业"> 企业 </n-tab> --> | ||||
|             </n-tabs> | ||||
|           </header> | ||||
|   <x-n-modal v-model:show="isShowBox" :title="forwardMode === 2 ? '合并转发' : '逐条转发'" style="width: 997px; height: 740px;background-color: #F9F9FD" | ||||
|     :on-after-leave="onMaskClick" content-style="display: flex; justify-content: center; align-items: center;"> | ||||
|     <div class="w-927px h-627px bg-#fff rounded-3px px-35px py-20px"> | ||||
|       <div class="flex items-center justify-between mb-28px"> | ||||
|         <div class="text-#333639">搜索</div> | ||||
|         <div class="w-779px h-34px"> | ||||
|           <n-input v-model:value="keywords" type="text" clearable placeholder="请输入"> | ||||
|           | ||||
|           <header class="el-header sub-header"> | ||||
|             <n-input placeholder="搜索" v-model:value="keywords" clearable size="small"> | ||||
|               <template #prefix> | ||||
|                 <n-icon :component="Search" /> | ||||
|               </template> | ||||
|             </n-input> | ||||
|           </header> | ||||
| 
 | ||||
|           <main class="el-main" v-loading="loading" loading-text="加载中..."> | ||||
|             <n-scrollbar> | ||||
|               <div class="friend-items"> | ||||
|                 <div | ||||
|                   class="friend-item pointer" | ||||
|                   v-for="item in searchFilter" | ||||
|                   :key="item.id" | ||||
|                   @click="onTriggerContact(item)" | ||||
|                 > | ||||
|                   <div class="avatar"> | ||||
|                     <im-avatar | ||||
|                       class="pointer" | ||||
|                       :src="item.avatar" | ||||
|                       :size="25" | ||||
|                       :username="item.remark || item.name" | ||||
|                     /> | ||||
|                   </div> | ||||
| 
 | ||||
|                   <div class="content"> | ||||
|                     <span class="text-ellipsis">{{ item.remark || item.name }}</span> | ||||
|                   </div> | ||||
| 
 | ||||
|                   <div class="checkbox"> | ||||
|                     <n-checkbox size="small" :checked="item.checked" /> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </n-scrollbar> | ||||
|           </main> | ||||
|         </section> | ||||
|       </aside> | ||||
| 
 | ||||
|       <main class="el-main"> | ||||
|         <section class="el-container is-vertical height100"> | ||||
|           <main class="el-main o-hidden"> | ||||
|             <n-scrollbar class="friend-items"> | ||||
|               <div class="friend-items"> | ||||
|                 <div v-show="!checkedFilter.length" style="padding-top: 100px"> | ||||
|                   <n-empty size="200" description="暂无数据"> | ||||
|                     <template #icon> | ||||
|                       <img src="@/assets/image/no-data.svg" alt="" /> | ||||
|                     </template> | ||||
|                   </n-empty> | ||||
|                 </div> | ||||
| 
 | ||||
|                 <div | ||||
|                   class="friend-item pointer" | ||||
|                   v-for="item in checkedFilter" | ||||
|                   :key="item.id" | ||||
|                   @click="onTriggerContact(item)" | ||||
|                 > | ||||
|                   <div class="avatar"> | ||||
|                     <im-avatar | ||||
|                       class="pointer" | ||||
|                       :src="item.avatar" | ||||
|                       :size="25" | ||||
|                       :username="item.remark || item.name" | ||||
|                     /> | ||||
|                   </div> | ||||
| 
 | ||||
|                   <div class="content"> | ||||
|                     <span class="text-ellipsis"> | ||||
|                       {{ item.remark || item.name }} | ||||
|                     </span> | ||||
|                     <span v-if="item.type == 2" class="badge group">群</span> | ||||
|                   </div> | ||||
| 
 | ||||
|                   <div class="checkbox"> | ||||
|                     <n-icon :size="16" :component="Delete" /> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </n-scrollbar> | ||||
|           </main> | ||||
|         </section> | ||||
|       </main> | ||||
|     </section> | ||||
| 
 | ||||
|     <template #footer> | ||||
|       <div class="footer"> | ||||
|         <div> | ||||
|           <span>已选择({{ checkedFilter.length }})</span> | ||||
|         </div> | ||||
| 
 | ||||
|         <div> | ||||
|           <n-button type="tertiary" @click="isShowBox = false"> 取消 </n-button> | ||||
|           <n-button type="primary" class="mt-l15" @click="onSubmit" :disabled="isCanSubmit"> | ||||
|             确定 | ||||
|           </n-button> | ||||
|           </n-input> | ||||
|         </div> | ||||
|       </div> | ||||
|     </template> | ||||
|   </n-modal> | ||||
|       <div class="flex justify-between"> | ||||
|         <div class="w-260px h-517px rounded-4px border-1px border-solid border-#E5E5E5 px-12px"> | ||||
|           <div class="border-b-2px border-b-solid border-b-#FBFBFB h-35px flex items-center justify-end"> | ||||
|             <n-button text color="#46299D" class="text-14px" @click="changeSelectType"> | ||||
|               {{ selectType === 1 ? '多选' : '单选' }} | ||||
|             </n-button> | ||||
|           </div> | ||||
|           <div> | ||||
|             <n-virtual-list v-if="!loading" style="max-height: 470px" :item-size="65" :items="searchFilter"> | ||||
|               <template #default="{ item }"> | ||||
|                 <div class="flex items-center border-b-2px border-b-solid h-65px border-b-#FBFBFB" | ||||
|                   @click="onTriggerContact(item)"> | ||||
|                   <div class="mr-22px"> | ||||
|                     <n-radio v-if="selectType === 1" :checked="item.checked" /> | ||||
|                     <n-checkbox v-else :checked="item.checked" /> | ||||
|                   </div> | ||||
|                   <div class="mr-10px"> | ||||
|                     <n-image class="w-42px h-42px rounded-full" :src="item.avatar" /> | ||||
|                   </div> | ||||
|                   <div class="flex items-center"> | ||||
|                     <span class="text-ellipsis">{{ item.name }}</span> | ||||
|                     <span v-if="item.type == 2" class="badge group ml-2">群</span> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </template> | ||||
|             </n-virtual-list> | ||||
|             <div v-else class="flex-center h-470px"> | ||||
|               <span>加载中...</span> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="w-578px h-517px rounded-4px border-1px border-solid border-#E5E5E5 px-12px"> | ||||
|           <div class="border-b-2px border-b-solid border-b-#FBFBFB h-35px flex items-center text-14px text-#000"> | ||||
|             发送给 | ||||
|           </div> | ||||
|           <div class="h-350px border-b-2px border-b-solid border-b-#FBFBFB"> | ||||
|             <div v-if="checkedFilter.length > 0"> | ||||
|               <n-virtual-list style="max-height: 350px" :item-size="65" :items="checkedFilter"> | ||||
|                 <template #default="{ item }"> | ||||
|                   <div class="flex items-center border-b-2px border-b-solid h-65px border-b-#FBFBFB pr-20px"> | ||||
|                     <div class="mr-10px"> | ||||
|                       <n-image class="w-42px h-42px rounded-full" :src="item.avatar" /> | ||||
|                     </div> | ||||
|                     <div class="flex items-center"> | ||||
|                       <span class="text-ellipsis">{{ item.name }}</span> | ||||
|                       <span v-if="item.type == 2" class="badge group ml-2">群</span> | ||||
|                     </div> | ||||
|                     <n-button class="ml-auto" text color="#C7C7C9" @click="onRemoveContact(item)"> | ||||
|                       <n-icon :component="CloseCircle" size="18" /> | ||||
|                     </n-button> | ||||
|                   </div> | ||||
|                 </template> | ||||
|               </n-virtual-list> | ||||
|             </div> | ||||
|             <div v-else class="flex-center h-350px"> | ||||
|               <n-empty size="medium" description="暂无选择联系人"> | ||||
|                 <template #icon> | ||||
|                   <img src="@/assets/image/no-data.svg" alt="" /> | ||||
|                 </template> | ||||
|               </n-empty> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="flex flex-col items-center justify-center h-120px"> | ||||
|             <div class="text-14px text-#999999 mb-23px"> | ||||
|               <span>[{{ forwardMode === 2 ? '合并转发' : '逐条转发' }}]</span>  | ||||
|               <span v-if="checkedFilter.length > 0"> | ||||
|                 {{ checkedFilter.map(item => item.name).join('、') }}的会话记录 | ||||
|               </span> | ||||
|               <span v-else>请选择联系人</span> | ||||
|             </div> | ||||
|             <div class="flex justify-center items-center"> | ||||
|               <n-button color="#C7C7C9" class="w-250px h-34px text-14px text-#fff mr-10px" @click="onCancel">取消</n-button> | ||||
|               <n-button color="#46299D" class="w-250px h-34px text-14px text-#fff"  | ||||
|                 @click="onSubmit" :disabled="isCanSubmit">发送</n-button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </x-n-modal> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| :deep(.n-divider__title) { | ||||
|   font-weight: unset; | ||||
| } | ||||
| 
 | ||||
| .launch-box { | ||||
|   height: 410px; | ||||
|   width: 100%; | ||||
|   overflow: hidden; | ||||
| 
 | ||||
|   .sub-header { | ||||
|     height: 50px; | ||||
|     padding: 10px 15px; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     align-items: center; | ||||
|   } | ||||
| 
 | ||||
|   .friend-items { | ||||
|     height: 100%; | ||||
|     overflow-y: auto; | ||||
|     padding: 0 15px; | ||||
| 
 | ||||
|     .friend-item { | ||||
|       height: 40px; | ||||
|       box-sizing: border-box; | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       margin: 5px 0; | ||||
| 
 | ||||
|       > div { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|         justify-content: center; | ||||
|       } | ||||
| 
 | ||||
|       .avatar { | ||||
|         width: 30px; | ||||
|         justify-content: flex-start; | ||||
|       } | ||||
| 
 | ||||
|       .content { | ||||
|         flex: 1 auto; | ||||
|         padding-left: 8px; | ||||
|         overflow: hidden; | ||||
|         font-size: 14px; | ||||
|         font-weight: 400; | ||||
|         justify-content: flex-start; | ||||
|         &:hover { | ||||
|           color: #409eff; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       .checkbox { | ||||
|         flex-shrink: 0; | ||||
|         width: 30px; | ||||
|         justify-content: flex-end; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .badge { | ||||
|   &.group { | ||||
|     color: #3370ff !important; | ||||
|     background-color: #e1eaff !important; | ||||
|   } | ||||
|   margin: 0 3px; | ||||
| } | ||||
| 
 | ||||
| .footer { | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: space-between; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -1,18 +1,16 @@ | ||||
| <script lang="ts" setup> | ||||
| import { ref, computed, reactive } from 'vue' | ||||
| import { NIcon, NModal, NButton, NInput, NDropdown, NPopover } from 'naive-ui' | ||||
| import { CloseOne, Male, Female, SendOne } from '@icon-park/vue-next' | ||||
| import { ServeSearchUser } from '@/api/contact' | ||||
| import { ServeCreateContact } from '@/api/contact' | ||||
| import { ServeContactGroupList, ServeContactMoveGroup, ServeEditContactRemark } from '@/api/contact' | ||||
| import { useTalkStore } from '@/store' | ||||
| import { useRouter } from 'vue-router' | ||||
| 
 | ||||
| import xNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | ||||
| import { NSkeleton } from 'naive-ui' | ||||
| const router = useRouter() | ||||
| const talkStore = useTalkStore() | ||||
| 
 | ||||
| const emit = defineEmits(['update:show', 'update:uid', 'updateRemark']) | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   show: { | ||||
|     type: Boolean, | ||||
| @ -26,7 +24,7 @@ const props = defineProps({ | ||||
| 
 | ||||
| const loading = ref(true) | ||||
| const isOpenFrom = ref(false) | ||||
| const state: any = reactive({ | ||||
| const userInfo: any = ref({ | ||||
|   id: 0, | ||||
|   avatar: '', | ||||
|   gender: 0, | ||||
| @ -43,26 +41,26 @@ const editCardPopover: any = ref(false) | ||||
| const modelRemark = ref('') | ||||
| 
 | ||||
| const options = ref<any>([]) | ||||
| const groupName = computed(() => { | ||||
|   const item = options.value.find((item: any) => { | ||||
|     return item.key == state.group_id | ||||
|   }) | ||||
| // const groupName = computed(() => { | ||||
| //   const item = options.value.find((item: any) => { | ||||
| //     return item.key == state.group_id | ||||
| //   }) | ||||
| 
 | ||||
|   if (item) { | ||||
|     return item.label | ||||
|   } | ||||
| //   if (item) { | ||||
| //     return item.label | ||||
| //   } | ||||
| 
 | ||||
|   return '未设置分组' | ||||
| }) | ||||
| //   return '未设置分组' | ||||
| // }) | ||||
| 
 | ||||
| const onLoadData = () => { | ||||
|   ServeSearchUser({ | ||||
|     user_id: props.uid | ||||
|     erp_user_id: props.uid | ||||
|   }).then(({ code, data }) => { | ||||
|     if (code == 200) { | ||||
|       Object.assign(state, data) | ||||
|       userInfo.value = data | ||||
| 
 | ||||
|       modelRemark.value = state.remark | ||||
|       // modelRemark.value = state.remark | ||||
| 
 | ||||
|       loading.value = false | ||||
|     } else { | ||||
| @ -70,15 +68,15 @@ const onLoadData = () => { | ||||
|     } | ||||
|   }) | ||||
| 
 | ||||
|   ServeContactGroupList().then((res) => { | ||||
|     if (res.code == 200) { | ||||
|       let items = res.data.items || [] | ||||
|       options.value = [] | ||||
|       for (const iter of items) { | ||||
|         options.value.push({ label: iter.name, key: iter.id }) | ||||
|       } | ||||
|     } | ||||
|   }) | ||||
|   // ServeContactGroupList().then((res) => { | ||||
|   //   if (res.code == 200) { | ||||
|   //     let items = res.data.items || [] | ||||
|   //     options.value = [] | ||||
|   //     for (const iter of items) { | ||||
|   //       options.value.push({ label: iter.name, key: iter.id }) | ||||
|   //     } | ||||
|   //   } | ||||
|   // }) | ||||
| } | ||||
| 
 | ||||
| const onToTalk = () => { | ||||
| @ -86,84 +84,84 @@ const onToTalk = () => { | ||||
|   emit('update:show', false) | ||||
| } | ||||
| 
 | ||||
| const onJoinContact = () => { | ||||
|   if (!state.text.length) { | ||||
|     return window['$message'].info('备注信息不能为空') | ||||
|   } | ||||
| // const onJoinContact = () => { | ||||
| //   if (!state.text.length) { | ||||
| //     return window['$message'].info('备注信息不能为空') | ||||
| //   } | ||||
| 
 | ||||
|   ServeCreateContact({ | ||||
|     friend_id: props.uid, | ||||
|     remark: state.text | ||||
|   }).then((res) => { | ||||
|     if (res.code == 200) { | ||||
|       isOpenFrom.value = false | ||||
|       window['$message'].success('申请发送成功') | ||||
|     } else { | ||||
|       window['$message'].error(res.message) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| //   ServeCreateContact({ | ||||
| //     friend_id: props.uid, | ||||
| //     remark: state.text | ||||
| //   }).then((res) => { | ||||
| //     if (res.code == 200) { | ||||
| //       isOpenFrom.value = false | ||||
| //       window['$message'].success('申请发送成功') | ||||
| //     } else { | ||||
| //       window['$message'].error(res.message) | ||||
| //     } | ||||
| //   }) | ||||
| // } | ||||
| 
 | ||||
| const onChangeRemark = () => { | ||||
|   ServeEditContactRemark({ | ||||
|     friend_id: props.uid, | ||||
|     remark: modelRemark.value | ||||
|   }).then(({ code, message }) => { | ||||
|     if (code == 200) { | ||||
|       editCardPopover.value.setShow(false) | ||||
|       window['$message'].success('备注成功') | ||||
|       state.remark = modelRemark.value | ||||
| // const onChangeRemark = () => { | ||||
| //   ServeEditContactRemark({ | ||||
| //     friend_id: props.uid, | ||||
| //     remark: modelRemark.value | ||||
| //   }).then(({ code, message }) => { | ||||
| //     if (code == 200) { | ||||
| //       editCardPopover.value.setShow(false) | ||||
| //       window['$message'].success('备注成功') | ||||
| //       state.remark = modelRemark.value | ||||
| 
 | ||||
|       emit('updateRemark', { | ||||
|         user_id: props.uid, | ||||
|         remark: modelRemark.value | ||||
|       }) | ||||
|     } else { | ||||
|       window['$message'].error(message) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| //       emit('updateRemark', { | ||||
| //         user_id: props.uid, | ||||
| //         remark: modelRemark.value | ||||
| //       }) | ||||
| //     } else { | ||||
| //       window['$message'].error(message) | ||||
| //     } | ||||
| //   }) | ||||
| // } | ||||
| 
 | ||||
| const handleSelectGroup = (value) => { | ||||
|   ServeContactMoveGroup({ | ||||
|     user_id: props.uid, | ||||
|     group_id: value | ||||
|   }).then(({ code, message }) => { | ||||
|     if (code == 200) { | ||||
|       state.group_id = value | ||||
|       window['$message'].success('分组修改成功') | ||||
|     } else { | ||||
|       window['$message'].error(message) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| // const handleSelectGroup = (value) => { | ||||
| //   ServeContactMoveGroup({ | ||||
| //     user_id: props.uid, | ||||
| //     group_id: value | ||||
| //   }).then(({ code, message }) => { | ||||
| //     if (code == 200) { | ||||
| //       state.group_id = value | ||||
| //       window['$message'].success('分组修改成功') | ||||
| //     } else { | ||||
| //       window['$message'].error(message) | ||||
| //     } | ||||
| //   }) | ||||
| // } | ||||
| 
 | ||||
| const reset = () => { | ||||
|   loading.value = true | ||||
| // const reset = () => { | ||||
| //   loading.value = true | ||||
| 
 | ||||
|   Object.assign(state, { | ||||
|     id: 0, | ||||
|     avatar: '', | ||||
|     gender: 0, | ||||
|     mobile: '', | ||||
|     motto: '', | ||||
|     nickname: '', | ||||
|     remark: '', | ||||
|     email: '', | ||||
|     status: 1, | ||||
|     text: '' | ||||
|   }) | ||||
| //   Object.assign(state, { | ||||
| //     id: 0, | ||||
| //     avatar: '', | ||||
| //     gender: 0, | ||||
| //     mobile: '', | ||||
| //     motto: '', | ||||
| //     nickname: '', | ||||
| //     remark: '', | ||||
| //     email: '', | ||||
| //     status: 1, | ||||
| //     text: '' | ||||
| //   }) | ||||
| 
 | ||||
|   isOpenFrom.value = false | ||||
| } | ||||
| //   isOpenFrom.value = false | ||||
| // } | ||||
| 
 | ||||
| const onUpdate = (value) => { | ||||
|   if (!value) { | ||||
|     setTimeout(reset, 100) | ||||
|   } | ||||
| // const onUpdate = (value) => { | ||||
| //   if (!value) { | ||||
| //     setTimeout(reset, 100) | ||||
| //   } | ||||
| 
 | ||||
|   emit('update:show', value) | ||||
| } | ||||
| //   emit('update:show', value) | ||||
| // } | ||||
| 
 | ||||
| const onAfterEnter = () => { | ||||
|   onLoadData() | ||||
| @ -171,162 +169,90 @@ const onAfterEnter = () => { | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <n-modal :show="show" :on-update:show="onUpdate" :on-after-enter="onAfterEnter"> | ||||
|     <div class="section" v-loading="loading"> | ||||
|       <section class="el-container container is-vertical"> | ||||
|         <header class="el-header header"> | ||||
|           <im-avatar | ||||
|             class="avatar" | ||||
|             :size="100" | ||||
|             :src="state.avatar" | ||||
|             :username="state.remark || state.nickname" | ||||
|             :font-size="30" | ||||
|           /> | ||||
|   <x-n-modal content-style="padding:0;" :closable="false" class="w-311px min-h-445px" style="border-radius: 10px;overflow:hidden;" :show="show"  :on-after-enter="onAfterEnter"> | ||||
|     <div class="section relative px-7px pt-82px pb-20px"> | ||||
|       <div class="absolute top-9px right-7px pointer z-10" @click="emit('update:show', false)"> | ||||
|         <img class="w-20px h-20px" src="@/assets/image/close.png" alt=""> | ||||
|       </div> | ||||
|        | ||||
|           <div class="gender" v-show="state.gender > 0"> | ||||
|             <n-icon v-if="state.gender == 1" :component="Male" color="#508afe" /> | ||||
|             <n-icon v-if="state.gender == 2" :component="Female" color="#ff5722" /> | ||||
|       <template v-if="loading"> | ||||
|         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> | ||||
|           <div class="w-59px h-59px rounded-8px mr-12px overflow-hidden"> | ||||
|             <n-skeleton circle height="59px" width="59px" /> | ||||
|           </div> | ||||
| 
 | ||||
|           <div class="close" @click="onUpdate(false)"> | ||||
|             <close-one theme="outline" size="22" fill="#fff" :strokeWidth="2" /> | ||||
|           <div class="w-full"> | ||||
|             <n-skeleton text style="width: 80%; margin-bottom: 5px;" /> | ||||
|             <n-skeleton text style="width: 60%;" /> | ||||
|           </div> | ||||
| 
 | ||||
|           <div class="nickname text-ellipsis"> | ||||
|             {{ state.remark || state.nickname || '未设置昵称' }} | ||||
|         </div> | ||||
|         <div class="bg-#fff rounded-4px mb-20px"> | ||||
|           <div class="flex px-15px py-9px" v-for="i in 6" :key="i"> | ||||
|             <n-skeleton text style="width: 30%; margin-right: 10px;" /> | ||||
|             <n-skeleton text style="width: 60%;" /> | ||||
|           </div> | ||||
|         </header> | ||||
|         </div> | ||||
|         <div> | ||||
|           <n-skeleton text style="width: 100%; height: 42px; border-radius: 4px;" /> | ||||
|         </div> | ||||
|       </template> | ||||
|        | ||||
|         <main class="el-main main me-scrollbar me-scrollbar-thumb"> | ||||
|           <div class="motto"> | ||||
|             {{ state.motto || '编辑个签,展示我的独特态度。' }} | ||||
|       <template v-else> | ||||
|         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> | ||||
|           <div class="w-59px h-59px  rounded-8px mr-12px overflow-hidden"> | ||||
|            <n-image width="59" :src="userInfo.avatar" > | ||||
| 
 | ||||
|            </n-image> | ||||
|           </div> | ||||
| 
 | ||||
|           <div class="infos"> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">工号 :</span> | ||||
|               <span class="text">{{ state.job_num}}</span> | ||||
|             </div> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">手机 :</span> | ||||
|               <span class="text">{{ state.mobile }}</span> | ||||
|             </div> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">昵称 :</span> | ||||
|               <span class="text text-ellipsis">{{ state.nickname || '-' }} </span> | ||||
|             </div> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">性别 :</span> | ||||
|               <span class="text">{{ | ||||
|                 state.gender == 1 ? '男' : state.gender == 2 ? '女' : '未知' | ||||
|               }}</span> | ||||
|             </div> | ||||
|             <div class="info-item" v-if="state.friend_status == 2"> | ||||
|               <span class="name">备注 :</span> | ||||
|               <n-popover trigger="click" placement="top-start" ref="editCardPopover"> | ||||
|                 <template #trigger> | ||||
|                   <span class="text edit pointer text-ellipsis"> | ||||
|                     {{ state.remark || '未设置' }}   | ||||
|                   </span> | ||||
|                 </template> | ||||
| 
 | ||||
|                 <template #header> 设置备注 </template> | ||||
| 
 | ||||
|                 <div style="display: flex"> | ||||
|                   <n-input | ||||
|                     type="text" | ||||
|                     placeholder="请填写备注" | ||||
|                     :autofocus="true" | ||||
|                     maxlength="10" | ||||
|                     v-model:value="modelRemark" | ||||
|                     @keydown.enter="onChangeRemark" | ||||
|                   /> | ||||
|                   <n-button | ||||
|                     type="primary" | ||||
|                     text-color="#ffffff" | ||||
|                     class="mt-l5" | ||||
|                     @click="onChangeRemark" | ||||
|                   > | ||||
|                     确定 | ||||
|                   </n-button> | ||||
|                 </div> | ||||
|               </n-popover> | ||||
|             </div> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">邮箱 :</span> | ||||
|               <span class="text">{{ state.email || '-' }}</span> | ||||
|             </div> | ||||
|             <div class="info-item" v-if="state.friend_status == 2"> | ||||
|               <span class="name">分组 :</span> | ||||
|               <n-dropdown | ||||
|                 trigger="click" | ||||
|                 placement="top-start" | ||||
|                 :show-arrow="true" | ||||
|                 :options="options" | ||||
|                 @select="handleSelectGroup" | ||||
|               > | ||||
|                 <span class="text edit pointer">{{ groupName }}</span> | ||||
|               </n-dropdown> | ||||
|             </div> | ||||
|             <div class="info-item"> | ||||
|               <span class="name">入职时间 :</span> | ||||
|               <span class="text">{{ state.enter_date}}</span> | ||||
|             </div> | ||||
|           <div> | ||||
|             <div class="text-#000 text-16px mb-5px">{{ userInfo.nickname }}</div> | ||||
|             <div class="text-#ACACAC text-12px">工号:{{ userInfo.job_num }}</div> | ||||
|           </div> | ||||
|         </main> | ||||
| 
 | ||||
|         <footer v-if="state.friend_status == 2" class="el-footer footer bdr-t flex-center"> | ||||
|           <n-button | ||||
|             round | ||||
|             block | ||||
|             type="primary" | ||||
|             text-color="#ffffff" | ||||
|             @click="onToTalk" | ||||
|             style="width: 91%" | ||||
|           > | ||||
|             <template #icon> | ||||
|               <n-icon :component="SendOne" /> | ||||
|             </template> | ||||
|             发送消息 | ||||
|         </div> | ||||
|         <div class="bg-#fff rounded-4px mb-20px"> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">公司别</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.company_name }}</div> | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">主管</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.leaders?.map(x=>x.user_name)?.join(',') }}</div> | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">部门</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.erp_dept_position?.map(x=>x.department_name)?.join(',') }}</div> | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">手机号</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.tel_num }}</div> | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">岗位</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.erp_dept_position?.map(x=>x.position_name)?.join(',') }}</div> | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">入职日期</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.enter_date }}</div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div> | ||||
|           <n-button block color="#EEE9F8" text-color="#46299D"     @click="onToTalk"> | ||||
|               <div class="flex items-center justify-center py-11px"> | ||||
|                 <img class="w-19.8px h-20px mr-15px" src="@/assets/image/faxi@2x.png" alt=""> | ||||
|                 <span>发送消息</span> | ||||
|               </div> | ||||
|           </n-button> | ||||
|         </footer> | ||||
| 
 | ||||
|         <footer v-else-if="state.friend_status == 1" class="el-footer footer bdr-t flex-center"> | ||||
|           <template v-if="isOpenFrom"> | ||||
|             <n-input | ||||
|               type="text" | ||||
|               placeholder="请填写申请备注" | ||||
|               v-model:value="state.text" | ||||
|               @keydown.enter="onJoinContact" | ||||
|             /> | ||||
| 
 | ||||
|             <n-button type="primary" text-color="#ffffff" class="mt-l5" @click="onJoinContact"> | ||||
|               确定 | ||||
|             </n-button> | ||||
|           </template> | ||||
|           <template v-else> | ||||
|             <n-button | ||||
|               type="primary" | ||||
|               text-color="#ffffff" | ||||
|               block | ||||
|               round | ||||
|               style="width: 91%" | ||||
|               @click="isOpenFrom = true" | ||||
|             > | ||||
|               添加好友 | ||||
|             </n-button> | ||||
|           </template> | ||||
|         </footer> | ||||
|       </section> | ||||
|         </div> | ||||
|       </template> | ||||
|     </div> | ||||
|   </n-modal> | ||||
|   </x-n-modal> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .section { | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   position: relative; | ||||
|   width: 360px; | ||||
|   height: 600px; | ||||
|   background-image: url('@/assets/image/zu6254@2x.png'); | ||||
|   border-radius: 10px; | ||||
|   overflow: hidden; | ||||
|   background-color: var(--im-bg-color); | ||||
| @ -336,7 +262,6 @@ const onAfterEnter = () => { | ||||
|     height: 230px; | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|     background: linear-gradient(to right, rgb(137, 104, 255), rgb(175, 152, 255)); | ||||
|     display: flex; | ||||
|     padding: 20px; | ||||
|     position: relative; | ||||
| @ -346,7 +271,6 @@ const onAfterEnter = () => { | ||||
|       width: 150px; | ||||
|       height: 150px; | ||||
|       content: ''; | ||||
|       background: linear-gradient(to right, rgb(142, 110, 255), rgb(208, 195, 255)); | ||||
|       position: absolute; | ||||
|       z-index: 1; | ||||
|       border-radius: 50%; | ||||
|  | ||||
| @ -1,144 +0,0 @@ | ||||
| import { createApp, h, ref } from 'vue' | ||||
| import { NImage, NImageGroup } from 'naive-ui' | ||||
| 
 | ||||
| interface PreviewOptions { | ||||
|   onStart?: () => void | ||||
|   onError?: (e: Event) => void | ||||
|   showToolbar?: boolean | ||||
| } | ||||
| 
 | ||||
| class ImagePreview { | ||||
|   private static instance: { | ||||
|     app: any | ||||
|     container: HTMLElement | ||||
|   } | null = null | ||||
| 
 | ||||
|   static async preview( | ||||
|     sources: string | File | (string | File)[],  | ||||
|     index = 0, | ||||
|     options: PreviewOptions = {}, | ||||
|   ) { | ||||
|     try { | ||||
|       const urls = await this.normalizeImageSources( | ||||
|         Array.isArray(sources) ? sources : [sources] | ||||
|       ) | ||||
| 
 | ||||
|       if (!urls.length) { | ||||
|         console.warn('[ImagePreview] No valid image sources') | ||||
|         return | ||||
|       } | ||||
| 
 | ||||
|       this.destroy() | ||||
|       options.onStart?.() | ||||
| 
 | ||||
|       const container = document.createElement('div') | ||||
|       container.style.display = 'none' | ||||
|       document.body.appendChild(container) | ||||
| 
 | ||||
|       const app = createApp({ | ||||
|         setup() { | ||||
|           const imageRef = ref<InstanceType<typeof NImage> | null>(null) | ||||
| 
 | ||||
|           return () => { | ||||
|             if (urls.length === 1) { | ||||
|               return h(NImage, { | ||||
|                 ref: imageRef, | ||||
|                 src: urls[0], | ||||
|                 previewDisabled: false, | ||||
|                 preview: true, | ||||
|                 showToolbar: options.showToolbar ?? true, | ||||
|                 style: { | ||||
|                   display: 'none' | ||||
|                 }, | ||||
|                 onLoad: () => { | ||||
|                   imageRef.value?.click() | ||||
|                 } | ||||
|               }) | ||||
|             } else { | ||||
|               return h(NImageGroup, { | ||||
|                 showToolbar: options.showToolbar ?? true, | ||||
|                 currentIndex: index | ||||
|               }, { | ||||
|                 default: () => urls.map((url, i) => { | ||||
|                   const imgRef = ref<InstanceType<typeof NImage> | null>(null) | ||||
|                   return h(NImage, { | ||||
|                     ref: i === index ? imgRef : undefined, | ||||
|                     src: url, | ||||
|                     previewDisabled: false, | ||||
|                     preview: true, | ||||
|                     style: { | ||||
|                       display: 'none' | ||||
|                     }, | ||||
|                     onLoad: i === index ? () => { | ||||
|                       imgRef.value?.click() | ||||
|                     } : undefined | ||||
|                   }) | ||||
|                 }) | ||||
|               }) | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
| 
 | ||||
|       app.mount(container) | ||||
|       this.instance = { app, container } | ||||
| 
 | ||||
|     } catch (error) { | ||||
|       console.error('[ImagePreview] Error:', error) | ||||
|       options.onError?.(error as Event) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private static async normalizeImageSources(sources: (string | File)[]): Promise<string[]> { | ||||
|     const urls: string[] = [] | ||||
| 
 | ||||
|     for (const source of sources) { | ||||
|       try { | ||||
|         if (typeof source === 'string') { | ||||
|           if (source.startsWith('data:') || source.startsWith('http')) { | ||||
|             urls.push(source) | ||||
|           } else { | ||||
|             console.warn('[ImagePreview] Invalid image source:', source) | ||||
|           } | ||||
|         } else if (source instanceof File) { | ||||
|           const url = await this.fileToUrl(source) | ||||
|           urls.push(url) | ||||
|         } | ||||
|       } catch (error) { | ||||
|         console.warn('[ImagePreview] Failed to process source:', source, error) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return urls | ||||
|   } | ||||
| 
 | ||||
|   private static fileToUrl(file: File): Promise<string> { | ||||
|     return new Promise((resolve, reject) => { | ||||
|       if (!file.type.startsWith('image/')) { | ||||
|         reject(new Error('Not an image file')) | ||||
|         return | ||||
|       } | ||||
| 
 | ||||
|       const reader = new FileReader() | ||||
|       reader.onload = () => resolve(reader.result as string) | ||||
|       reader.onerror = reject | ||||
|       reader.readAsDataURL(file) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   private static destroy() { | ||||
|     if (this.instance) { | ||||
|       const { app, container } = this.instance | ||||
|       app.unmount() | ||||
|       container.remove() | ||||
|       this.instance = null | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   static close() { | ||||
|     this.destroy() | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export const previewImage = ImagePreview.preview.bind(ImagePreview) | ||||
| export const closePreview = ImagePreview.close.bind(ImagePreview) | ||||
| @ -4,5 +4,6 @@ export const enum ContactConst { | ||||
| 
 | ||||
| export const enum EditorConst { | ||||
|   Mention = 'editor:mention', | ||||
|   Quote = 'editor:quote' | ||||
|   Quote = 'editor:quote', | ||||
|   Edit = 'editor:edit' | ||||
| } | ||||
|  | ||||
| @ -1,14 +1,28 @@ | ||||
| // 主题配置
 | ||||
| const primaryColor='#46299D' | ||||
| export const overrides = { | ||||
|   common: { | ||||
|     primaryColor: '#46299D', | ||||
|     primaryColorHover: '#46299D', | ||||
|     primaryColorPressed: '#46299D', | ||||
|     primaryColorSuppl: '#46299D', | ||||
|     bodyColor: '#ffffff' | ||||
|   DataTable: { | ||||
|     sorterIconColor:'#fff', | ||||
|     thColorHover: primaryColor, | ||||
|     thTextColor: "#fff", | ||||
|     thColor: primaryColor, | ||||
|     thBackgroundColor: primaryColor, | ||||
|     itemColorActive:primaryColor, | ||||
|   }, | ||||
| 
 | ||||
|   Dialog: { | ||||
|     borderRadius: '10px' | ||||
|   Button: { | ||||
|     textColor: primaryColor, | ||||
|   }, | ||||
|   Dropdown:{ | ||||
|     optionTextColorHover:'#46299D', | ||||
|     optionColorHover:'#EEE9F8' | ||||
|   }, | ||||
|   common: { | ||||
|     primaryColorPressed: primaryColor, | ||||
|     primaryHover:primaryColor, | ||||
|     primaryDefault: primaryColor, | ||||
|     primaryActive: primaryColor, | ||||
|     primarySuppl: primaryColor, | ||||
|     primaryColor: primaryColor, | ||||
|     primaryColorHover: primaryColor | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -7,21 +7,21 @@ const settingsStore = useSettingsStore() | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <section class="container flex-center"> | ||||
|   <section class="top-container flex-center"> | ||||
|     <section | ||||
|       class="el-container im-container" | ||||
|       :class="{ | ||||
|         'small-screen': !settingsStore.isFullScreen | ||||
|       }" | ||||
|     > | ||||
|       <aside | ||||
|       <!-- <aside | ||||
|         class="el-aside" | ||||
|         :class="{ | ||||
|           'pd-t15': isElectronMode() | ||||
|         }" | ||||
|       > | ||||
|         <Menu /> | ||||
|       </aside> | ||||
|       </aside> --> | ||||
|       <main class="el-main"> | ||||
|         <router-view /> | ||||
|       </main> | ||||
| @ -30,8 +30,8 @@ const settingsStore = useSettingsStore() | ||||
| 
 | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| .container { | ||||
|   | ||||
| .top-container { | ||||
|  width: 100%; | ||||
|   background: url(@/assets/image/background.jpeg); | ||||
|   background-position: center center; | ||||
|   background-repeat: no-repeat; | ||||
| @ -39,8 +39,8 @@ const settingsStore = useSettingsStore() | ||||
|   background-size: cover; | ||||
| 
 | ||||
|   .im-container { | ||||
|     height: 80vh; | ||||
|     width: 100vw; | ||||
|     height: 85vh; | ||||
|  width: 100%; | ||||
|     overflow: hidden; | ||||
|     background-color: #fff; | ||||
| 
 | ||||
|  | ||||
| @ -226,6 +226,30 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|         useEditorStore().loadUserEmoticon() | ||||
|         window['$message'] && window['$message'].success('收藏成功') | ||||
|       }) | ||||
|     }, | ||||
| 
 | ||||
|     // 更新视频上传进度
 | ||||
|     updateUploadProgress(uploadId, percentage) { | ||||
|       const record = this.records.find(item =>  | ||||
|         item.extra && item.extra.is_uploading && item.extra.upload_id === uploadId | ||||
|       ) | ||||
|        | ||||
|       if (record) { | ||||
|         record.extra.percentage = percentage | ||||
|       } | ||||
|     }, | ||||
|      | ||||
|     // 视频上传完成后更新消息
 | ||||
|     completeUpload(uploadId, videoInfo) { | ||||
|       const record = this.records.find(item =>  | ||||
|         item.extra && item.extra.is_uploading && item.extra.upload_id === uploadId | ||||
|       ) | ||||
|        | ||||
|       if (record) { | ||||
|         record.extra.is_uploading = false | ||||
|         record.extra.url = videoInfo.url | ||||
|         // record.extra.cover = videoInfo.cover
 | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|  | ||||
| @ -1,10 +1,34 @@ | ||||
| import { defineStore } from 'pinia' | ||||
| import { ServeFindFileSplitInfo, ServeFileSubareaUpload } from '@/api/upload' | ||||
| import { ServeSendTalkFile } from '@/api/chat' | ||||
| 
 | ||||
| import { uploadImg } from '@/api/upload' | ||||
| import { | ||||
|   useDialogueStore | ||||
| } from '@/store' | ||||
| // @ts-ignore
 | ||||
| const message = window.$message | ||||
| 
 | ||||
| // 定义上传项接口
 | ||||
| interface UploadItem { | ||||
|   file: File; | ||||
|   talk_type: number; | ||||
|   receiver_id: number; | ||||
|   upload_id: string; | ||||
|   client_upload_id?: string; // 上传时的客户端ID
 | ||||
|   uploadIndex: number; | ||||
|   percentage: number; | ||||
|   status: number; // 文件上传状态 0:等待上传 1:上传中 2:上传完成 3:网络异常
 | ||||
|   files: FormData[]; | ||||
|   avatar: string; | ||||
|   username: string; | ||||
|   is_paused?: boolean; // 是否暂停上传
 | ||||
|   form?: FormData; | ||||
|   progress_interval?: any; | ||||
|   upload_controller?: AbortController; | ||||
|   onProgress?: (percentage: number) => void; | ||||
|   onComplete?: (data: any) => void; | ||||
| } | ||||
| 
 | ||||
| // 处理拆分上传文件
 | ||||
| function fileSlice(file: File, uploadId: string, eachSize: number) { | ||||
|   const splitNum = Math.ceil(file.size / eachSize) // 分片总数
 | ||||
| @ -31,7 +55,8 @@ export const useUploadsStore = defineStore('uploads', { | ||||
|   state: () => { | ||||
|     return { | ||||
|       isShow: false, | ||||
|       items: [] | ||||
|       items: [] as UploadItem[], | ||||
|       dialogueStore: useDialogueStore() | ||||
|     } | ||||
|   }, | ||||
|   getters: { | ||||
| @ -45,81 +70,282 @@ export const useUploadsStore = defineStore('uploads', { | ||||
|     close() { | ||||
|       this.isShow = false | ||||
|     }, | ||||
|     // 获取分片文件数组索引
 | ||||
|     findItem(uploadId: string): UploadItem | undefined { | ||||
|       return this.items.find((item) => item.upload_id === uploadId) | ||||
|     }, | ||||
| 
 | ||||
|     // 通过客户端ID查找上传项
 | ||||
|     findItemByClientId(clientUploadId: string): UploadItem | undefined { | ||||
|       return this.items.find((item) => item.client_upload_id === clientUploadId) | ||||
|     }, | ||||
| 
 | ||||
|     // // 暂停文件上传
 | ||||
|     // pauseUpload(uploadId: string) {
 | ||||
|     //   const item = this.findItem(uploadId)
 | ||||
|     //   if (!item) return
 | ||||
|        | ||||
|     //   item.is_paused = true
 | ||||
|     //   console.log(`暂停上传: ${uploadId}`)
 | ||||
|     // },
 | ||||
| 
 | ||||
|     // 恢复文件上传
 | ||||
|     // resumeUpload(uploadId: string) {
 | ||||
|     //   const item = this.findItem(uploadId)
 | ||||
|     //   if (!item) return
 | ||||
|        | ||||
|     //   item.is_paused = false
 | ||||
|     //   console.log(`恢复上传: ${uploadId}`)
 | ||||
|        | ||||
|     //   // 继续上传
 | ||||
|     //   this.triggerUpload(uploadId)
 | ||||
|     // },
 | ||||
| 
 | ||||
|     // 发送上传消息
 | ||||
|     async sendUploadMessage(item: any) { | ||||
|       try { | ||||
|         await ServeSendTalkFile({ | ||||
|           upload_id: item.upload_id, | ||||
|           receiver_id: item.receiver_id, | ||||
|           talk_type: item.talk_type | ||||
|         }) | ||||
|       } catch (error) { | ||||
|         console.error("发送上传消息失败:", error) | ||||
|       } | ||||
|     }, | ||||
| 
 | ||||
|     // 初始化上传(使用分片上传方式)
 | ||||
|     async initUploadFile( | ||||
|       file: File,  | ||||
|       talkType: number,  | ||||
|       receiverId: number,  | ||||
|       username: string, | ||||
|       uploadId: string, | ||||
|       onProgress: (percentage: number) => void, | ||||
|       onComplete: (data: any) => void | ||||
|     ) { | ||||
|       // 使用分片上传机制,先获取分片信息
 | ||||
|       try { | ||||
|         const res = await ServeFindFileSplitInfo({ | ||||
|           file_name: file.name, | ||||
|           file_size: file.size | ||||
|         }) | ||||
|          | ||||
|     // 初始化上传
 | ||||
|     initUploadFile(file: File, talkType: number, receiverId: number, username: string) { | ||||
|       ServeFindFileSplitInfo({ | ||||
|         file_name: file.name, | ||||
|         file_size: file.size | ||||
|       }).then((res) => { | ||||
|         if (res.code == 200) { | ||||
|           const { upload_id, split_size } = res.data | ||||
|            | ||||
|           // 使用较小的分片大小,以获得更细粒度的进度控制
 | ||||
|           // 将分片大小减半,增加分片数量
 | ||||
|           const actualSplitSize = Math.min(split_size, 512 * 1024); // 使用更小的分片,如512KB
 | ||||
|            | ||||
|           // 创建分片数组
 | ||||
|           const fileChunks = fileSlice(file, upload_id, actualSplitSize) | ||||
|            | ||||
|           // @ts-ignore
 | ||||
|           this.items.unshift({ | ||||
|             file: file, | ||||
|             talk_type: talkType, | ||||
|             receiver_id: receiverId, | ||||
|             upload_id: upload_id, | ||||
|             client_upload_id: uploadId, // 客户端生成的上传ID,用于前端标识
 | ||||
|             uploadIndex: 0, | ||||
|             percentage: 0, | ||||
|             status: 0, // 文件上传状态 0:等待上传 1:上传中 2:上传完成 3:网络异常
 | ||||
|             files: fileSlice(file, upload_id, split_size), | ||||
|             files: fileChunks, | ||||
|             avatar: '', | ||||
|             username: username | ||||
|             username: username, | ||||
|             is_paused: false, | ||||
|             onProgress: onProgress, | ||||
|             onComplete: onComplete, | ||||
|           }) | ||||
|            | ||||
|           this.triggerUpload(upload_id) | ||||
|           this.isShow = true | ||||
|           this.isShow = false // 不显示上传管理抽屉
 | ||||
|            | ||||
|           // 开始上传分片
 | ||||
|           this.triggerUpload(upload_id, uploadId) | ||||
|         } else { | ||||
|           message.error(res.message) | ||||
|           onProgress(-1) // 通知上传失败
 | ||||
|         } | ||||
|       }) | ||||
|       } catch (error) { | ||||
|         console.error("初始化分片上传失败:", error); | ||||
|         message.error("初始化上传失败,请重试") | ||||
|         onProgress(-1) | ||||
|       } | ||||
|     }, | ||||
|      | ||||
|     // 获取分片文件数组索引
 | ||||
|     findItem(uploadId: string): any { | ||||
|       return this.items.find((item: any) => item.upload_id === uploadId) | ||||
|     }, | ||||
|     // 触发分片上传
 | ||||
|     async triggerUpload(uploadId: string, clientUploadId?: string) { | ||||
|       const currentItem = this.findItem(uploadId) | ||||
|       if (!currentItem) return | ||||
|        | ||||
|     // 触发上传
 | ||||
|     triggerUpload(uploadId: string) { | ||||
|       const item = this.findItem(uploadId) | ||||
|       // 如果已暂停,不继续上传
 | ||||
|       if (currentItem.is_paused) return | ||||
|        | ||||
|       const form = item.files[item.uploadIndex] | ||||
|       // 如果已上传完成,不继续上传
 | ||||
|       if (currentItem.uploadIndex >= currentItem.files.length) { | ||||
|         if (clientUploadId) { | ||||
|           this.completeUpload(currentItem, clientUploadId) | ||||
|         } | ||||
|         return | ||||
|       } | ||||
|        | ||||
|       item.status = 1 | ||||
|       // 获取当前要上传的分片
 | ||||
|       const form = currentItem.files[currentItem.uploadIndex] | ||||
|        | ||||
|       ServeFileSubareaUpload(form) | ||||
|         .then((res) => { | ||||
|           if (res.code == 200) { | ||||
|             item.uploadIndex++ | ||||
|       // 更新状态为上传中
 | ||||
|       currentItem.status = 1 | ||||
|        | ||||
|             if (item.uploadIndex === item.files.length) { | ||||
|               item.status = 2 | ||||
|               item.percentage = 100 | ||||
|               this.sendUploadMessage(item) | ||||
|             } else { | ||||
|               const percentage = (item.uploadIndex / item.files.length) * 100 | ||||
|               item.percentage = percentage.toFixed(1) | ||||
|               this.triggerUpload(uploadId) | ||||
|       // 上传当前分片
 | ||||
|       try { | ||||
|         const res = await ServeFileSubareaUpload(form) | ||||
|          | ||||
|         // 获取最新的项目状态,确保仍然存在且没有被暂停
 | ||||
|         const updatedItem = this.findItem(uploadId) | ||||
|         if (!updatedItem || updatedItem.is_paused) return | ||||
|          | ||||
|         if (res.code == 200) { | ||||
|           // 当前分片上传成功,增加索引
 | ||||
|           updatedItem.uploadIndex++ | ||||
|            | ||||
|           // 计算上传进度
 | ||||
|           const percentage = (updatedItem.uploadIndex / updatedItem.files.length) * 100 | ||||
|           updatedItem.percentage = parseFloat(percentage.toFixed(1)) | ||||
|            | ||||
|           // 回调进度
 | ||||
|           if (updatedItem.onProgress) { | ||||
|             updatedItem.onProgress(updatedItem.percentage) | ||||
|           } | ||||
|           // if (clientUploadId) {
 | ||||
|           //   this.dialogueStore.updateUploadProgress(clientUploadId, percentage)
 | ||||
|           // }
 | ||||
|           // 检查是否全部上传完成
 | ||||
|           if (updatedItem.uploadIndex === updatedItem.files.length) { | ||||
|             // 所有分片上传完成
 | ||||
|             if (clientUploadId) { | ||||
|               this.completeUpload(updatedItem, clientUploadId) | ||||
|             } | ||||
|           } else { | ||||
|             item.status = 3 | ||||
|             // 继续上传下一个分片
 | ||||
|             this.triggerUpload(uploadId, clientUploadId) | ||||
|           } | ||||
|         }) | ||||
|         .catch(() => { | ||||
|           item.status = 3 | ||||
|         }) | ||||
|         } else { | ||||
|           // 上传失败处理
 | ||||
|           console.error(`分片上传失败,错误码: ${res.code},错误信息: ${res.message || '未知错误'}`); | ||||
|           updatedItem.status = 3 | ||||
|            | ||||
|           // 尝试重试当前分片
 | ||||
|           this.retryUpload(uploadId, clientUploadId, res.message || '上传失败,请重试') | ||||
|         } | ||||
|       } catch (error) { | ||||
|         console.error("分片上传错误:", error); | ||||
|          | ||||
|         // 获取最新的项目状态
 | ||||
|         const updatedItem = this.findItem(uploadId) | ||||
|         if (!updatedItem) return | ||||
|          | ||||
|         // 如果是暂停导致的错误,不改变状态
 | ||||
|         if (updatedItem.is_paused) return | ||||
|          | ||||
|         updatedItem.status = 3 | ||||
|          | ||||
|         // 尝试重试当前分片
 | ||||
|         this.retryUpload(uploadId, clientUploadId, '网络错误,正在重试') | ||||
|       } | ||||
|     }, | ||||
|      | ||||
|     // 发送上传消息
 | ||||
|     sendUploadMessage(item: any) { | ||||
|       ServeSendTalkFile({ | ||||
|         upload_id: item.upload_id, | ||||
|         receiver_id: item.receiver_id, | ||||
|         talk_type: item.talk_type | ||||
|       }) | ||||
|     } | ||||
|     // 重试上传
 | ||||
|     retryUpload(uploadId: string, clientUploadId?: string, errorMessage?: string) { | ||||
|       const item = this.findItem(uploadId) | ||||
|       if (!item) return | ||||
|        | ||||
|       // 如果有暂停/恢复按钮,先告知用户上传出错
 | ||||
|       if (item.onProgress) { | ||||
|         item.onProgress(-1) | ||||
|       } | ||||
|        | ||||
|       // 显示错误提示
 | ||||
|       message.warning(errorMessage) | ||||
|        | ||||
|       // 创建一个5秒后自动重试的机制
 | ||||
|       setTimeout(() => { | ||||
|         const currentItem = this.findItem(uploadId) | ||||
|         if (!currentItem) return | ||||
|          | ||||
|         // 如果用户没有手动暂停,则自动重试
 | ||||
|         if (!currentItem.is_paused) { | ||||
|           console.log('正在重试上传分片...'); | ||||
|           this.triggerUpload(uploadId, clientUploadId) | ||||
|         } | ||||
|       }, 5000) | ||||
|     }, | ||||
|      | ||||
|     // 完成上传
 | ||||
|     async completeUpload(item: UploadItem, clientUploadId: string) { | ||||
|       if (!item) return; | ||||
|        | ||||
|       item.status = 2 | ||||
|       item.percentage = 100 | ||||
|       if (item.onProgress) { | ||||
|         item.onProgress(100) | ||||
|       } | ||||
|        | ||||
|       // 获取最终URL并回调
 | ||||
|       try { | ||||
|         await ServeSendTalkFile({ | ||||
|           upload_id: item.upload_id, | ||||
|           receiver_id: item.receiver_id, | ||||
|           talk_type: item.talk_type | ||||
|         }) | ||||
|          | ||||
|         if (item.onComplete) { | ||||
|           item.onComplete(item) | ||||
|         } | ||||
|       } catch (error) { | ||||
|         console.error("发送文件消息失败:", error); | ||||
|       } | ||||
|     }, | ||||
|      | ||||
|      | ||||
|     pauseUpload(clientUploadId: string) { | ||||
|       const item = this.findItemByClientId(clientUploadId) | ||||
|       if (!item) return | ||||
|        | ||||
|       item.is_paused = true | ||||
|     }, | ||||
|      | ||||
|     // 恢复上传
 | ||||
|     resumeUpload(clientUploadId: string) { | ||||
|       const item = this.findItemByClientId(clientUploadId) | ||||
|       if (!item) return | ||||
|        | ||||
|       item.is_paused = false | ||||
|        | ||||
|       // 继续上传
 | ||||
|       if (item.upload_id) { | ||||
|         this.triggerUpload(item.upload_id, clientUploadId) | ||||
|       } | ||||
|     }, | ||||
|      | ||||
|     // 重试文件上传
 | ||||
|     retryCommonUpload(uploadId: string, errorMessage: string) { | ||||
|       const item = this.findItem(uploadId) | ||||
|       if (!item) return | ||||
|        | ||||
|       // 显示错误提示
 | ||||
|       message.warning(errorMessage) | ||||
|        | ||||
|       // 创建一个5秒后自动重试的机制
 | ||||
|       setTimeout(() => { | ||||
|         const currentItem = this.findItem(uploadId) | ||||
|         if (!currentItem) return | ||||
|          | ||||
|         // 如果用户没有手动暂停,则自动重试
 | ||||
|         if (!currentItem.is_paused) { | ||||
|           console.log('正在重试上传分片...'); | ||||
|           this.triggerUpload(uploadId) | ||||
|         } | ||||
|       }, 5000) | ||||
|     }, | ||||
|   } | ||||
| }) | ||||
|  | ||||
| @ -4,7 +4,7 @@ import { ServeFindFriendApplyNum } from '@/api/contact' | ||||
| import { ServeGroupApplyUnread } from '@/api/group' | ||||
| import { delAccessToken } from '@/utils/auth' | ||||
| import { storage } from '@/utils/storage' | ||||
| 
 | ||||
| import { GetUserInfo } from '@/api/auth' | ||||
| interface UserStoreState { | ||||
|   uid: number | ||||
|   nickname: string | ||||
| @ -35,7 +35,7 @@ export const useUserStore = defineStore('user', { | ||||
|       online: false, // 在线状态
 | ||||
|       isQiye: false, | ||||
|       isContactApply: false, | ||||
|       isGroupApply: false | ||||
|       isGroupApply: false, | ||||
|     } | ||||
|   }, | ||||
|   getters: {}, | ||||
|  | ||||
| @ -47,7 +47,10 @@ export interface ITalkRecord { | ||||
|   extra: any | ||||
|   isCheck: boolean | ||||
|   send_status: number | ||||
|   float: string | ||||
|   float: string, | ||||
|   is_convert_text?:number//语音记录的 是否是在展示转文本状态 1:是 0:否,
 | ||||
|   erp_user_id:number | ||||
|    | ||||
| } | ||||
| 
 | ||||
| export interface ITalkRecordExtraText { | ||||
| @ -65,6 +68,7 @@ export interface ITalkRecordExtraFile { | ||||
|   name: string | ||||
|   path: string | ||||
|   size: number | ||||
|   percentage: number | ||||
| } | ||||
| 
 | ||||
| export interface ITalkRecordExtraForward { | ||||
| @ -90,6 +94,9 @@ export interface ITalkRecordExtraVideo { | ||||
|   url: string | ||||
|   duration: number | ||||
|   size: number | ||||
|   is_uploading?: boolean | ||||
|   upload_id?: string | ||||
|   percentage?: number | ||||
| } | ||||
| 
 | ||||
| export interface ITalkRecordExtraMixed { | ||||
|  | ||||
| @ -10,29 +10,36 @@ defineProps({ | ||||
|   username: String, | ||||
|   active: Boolean | ||||
| }) | ||||
| //群类型:1=普通群;2=部门群;3=项目群;4=总群/公司群 | ||||
| const labelColor=[ | ||||
|   {group_type:2,color:'#377EC6',label:'部门'}, | ||||
|   {group_type:3,color:'#C1691C',label:'项目'}, | ||||
|   {group_type:4,color:'#7A58DE',label:'公司'}, | ||||
| ] | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="talk pointer" :class="{ actived: active }" @click="emit('tab-talk', data)"> | ||||
|     <div class="avatar-box"> | ||||
|     <div class="avatar-box relative"> | ||||
|        | ||||
|       <avatarModule    :mode="data?.group_type === 0 ? 1 : 2" | ||||
|               :avatar="data?.avatar" | ||||
|               :groupType="data?.group_type" | ||||
|               :userName="data?.name" :customStyle="{width:'42px',height:'42px'}"></avatarModule> | ||||
|               <div v-if="[2,3,4].includes(data.group_type)" class="absolute w-32px h-18px border-2px border-solid rounded-3px top-28px bg-#fff text-10px flex justify-center items-center leading-none" :style="`color:${labelColor.find(x=>x.group_type===data.group_type)?.color};border-color:${labelColor.find(x=>x.group_type===data.group_type)?.color}`">{{ labelColor.find(x=>x.group_type===data.group_type)?.label }}</div> | ||||
|       <!-- <im-avatar :src="avatar" :size="34" :username="data.name" /> --> | ||||
|       <div class="top-mask" @click.stop="emit('top-talk', data)"> | ||||
|       <!-- <div class="top-mask" @click.stop="emit('top-talk', data)"> | ||||
|         <n-icon :component="data.is_top == 1 ? ArrowDown : ArrowUp" /> | ||||
|       </div> | ||||
|       </div> --> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="content-box"> | ||||
|       <div class="header"> | ||||
|         <div class="title"> | ||||
|           <span class="nickname">{{ username }}</span> | ||||
|           <span class="badge top" v-show="data.is_top">顶</span> | ||||
|           <!-- <span class="badge top" v-show="data.is_top">顶</span> | ||||
|           <span class="badge roboot" v-show="data.is_robot">助</span> | ||||
|           <span class="badge group" v-show="data.talk_type == 2">群</span> | ||||
|           <span class="badge group" v-show="data.talk_type == 2">群</span> --> | ||||
|         </div> | ||||
|         <div class="datetime"><Xtime :time="data.updated_at" /></div> | ||||
|       </div> | ||||
| @ -80,8 +87,8 @@ defineProps({ | ||||
|   border-radius: 5px; | ||||
| 
 | ||||
|   .avatar-box { | ||||
|     height: 34px; | ||||
|     width: 34px; | ||||
|     height: 42px; | ||||
|     width: 42px; | ||||
|     border-radius: 50%; | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
| @ -90,7 +97,7 @@ defineProps({ | ||||
|     user-select: none; | ||||
|     transition: ease 1s; | ||||
|     position: relative; | ||||
|     overflow: hidden; | ||||
| 
 | ||||
| 
 | ||||
|     .top-mask { | ||||
|       width: 100%; | ||||
| @ -193,9 +200,9 @@ defineProps({ | ||||
|           user-select: none; | ||||
| 
 | ||||
|           .badge { | ||||
|             background-color: #f44336; | ||||
|             background-color: #D03050; | ||||
|             color: #ffffff; | ||||
|             border-radius: 3px; | ||||
|             border-radius: 50%; | ||||
|             transform-origin: right; | ||||
|           } | ||||
|         } | ||||
|  | ||||
| @ -69,38 +69,49 @@ const onContactModal = (data: { id: number; type: number }[]) => { | ||||
|     <div class="multi-title"> | ||||
|       <span>已选中:{{ dialogueStore.selectItems.length }} 条消息</span> | ||||
|     </div> | ||||
|     <div class="multi-groups"> | ||||
|       <div class="btn-group"> | ||||
|     <div class="flex items-center relative"> | ||||
|       <div class="multi-groups"> | ||||
|       <div class="btn-group mr-156px"> | ||||
|         <div class="multi-icon pointer flex-center" @click="onMergeForward"> | ||||
|           <n-icon :size="22" :component="Share" /> | ||||
|           <!-- <n-icon :size="22" :component="Share" /> --> | ||||
|            <img src="@/assets/image/zu6299@2x.png" class="w-72px h-72px" alt=""> | ||||
|         </div> | ||||
|         <p>合并转发</p> | ||||
|       </div> | ||||
|       <div class="btn-group"> | ||||
|       <div class="btn-group mr-156px"> | ||||
|         <div class="multi-icon pointer flex-center" @click="onSingleForward"> | ||||
|           <n-icon :size="22" :component="ShareThree" /> | ||||
|           <!-- <n-icon :size="22" :component="ShareThree" /> --> | ||||
|            <img class="w-72px h-72px" src="@/assets/image/zu6300@2x.png"> | ||||
|         </div> | ||||
|         <p>逐条转发</p> | ||||
|       </div> | ||||
|       <div class="btn-group"> | ||||
|       <div class="btn-group "> | ||||
|         <div class="multi-icon pointer flex-center" @click="onMultiDelete"> | ||||
|           <n-icon :size="22" :component="Delete" /> | ||||
|           <!-- <n-icon :size="22" :component="Delete" /> --> | ||||
|           <img class="w-72px h-72px" src="@/assets/image/zu6302@2x.png"> | ||||
|         </div> | ||||
|         <p>批量删除</p> | ||||
|       </div> | ||||
|       <div class="btn-group"> | ||||
|       <!-- <div class="btn-group"> | ||||
|         <div class="multi-icon pointer flex-center" @click="onClose"> | ||||
|           <n-icon :size="22" :component="Close" /> | ||||
|           | ||||
|         </div> | ||||
|         <p>关闭</p> | ||||
|       </div> | ||||
|       </div> --> | ||||
|    | ||||
|     </div> | ||||
|     <div class="pointer absolute right-150px top-50% translate-y-[-50%]"  @click="onClose"> | ||||
|       <img class="w-30px h-30px" src="@/assets/image/zu6306@2x.png" alt=""> | ||||
|     </div> | ||||
|     </div> | ||||
|   | ||||
|   </section> | ||||
| 
 | ||||
|   <ContactModal | ||||
|     v-if="isShowContactModal" | ||||
|     v-on:close="isShowContactModal = false" | ||||
|     v-on:on-submit="onContactModal" | ||||
|     :forward-mode="forwardMode" | ||||
|   /> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| @ -126,13 +137,9 @@ const onContactModal = (data: { id: number; type: number }[]) => { | ||||
|     justify-content: center; | ||||
| 
 | ||||
|     .btn-group { | ||||
|       width: 50px; | ||||
|       height: 80px; | ||||
|       margin: 0 15px; | ||||
| 
 | ||||
|       .multi-icon { | ||||
|         width: 50px; | ||||
|         height: 50px; | ||||
|         width: 72px; | ||||
|         height: 72px; | ||||
|         background-color: var(--im-active-bg-color); | ||||
|         border-radius: 50%; | ||||
| 
 | ||||
|  | ||||
| @ -13,7 +13,10 @@ import SkipBottom from './SkipBottom.vue' | ||||
| import { ITalkRecord } from '@/types/chat' | ||||
| import { EditorConst } from '@/constant/event-bus' | ||||
| import { useInject, useTalkRecord, useUtil } from '@/hooks' | ||||
| 
 | ||||
| import { ExclamationCircleFilled } from '@ant-design/icons-vue'; | ||||
| import { useUserStore } from '@/store' | ||||
| import RevokeMessage from '@/components/talk/message/RevokeMessage.vue' | ||||
| import { voiceToText } from '@/api/chat.js' | ||||
| const props = defineProps({ | ||||
|   uid: { | ||||
|     type: Number, | ||||
| @ -39,15 +42,12 @@ const { useMessage } = useUtil() | ||||
| const { dropdown, showDropdownMenu, closeDropdownMenu } = useMenu() | ||||
| const { showUserInfoModal } = useInject() | ||||
| const dialogueStore = useDialogueStore() | ||||
| 
 | ||||
| watch(() => records, (newValue, oldValue) => { | ||||
|   console.log(newValue); | ||||
| 
 | ||||
| },{deep:true,immediate:true}) | ||||
| 
 | ||||
| const userStore = useUserStore() | ||||
| // const showUserInfoModal = (uid: number) => { | ||||
| //   userStore.getUserInfo(uid) | ||||
| // } | ||||
| // 置底按钮 | ||||
| const skipBottom = ref(false) | ||||
| 
 | ||||
| // 是否显示消息时间 | ||||
| const isShowTalkTime = (index: number, datetime: string) => { | ||||
|   if (datetime == undefined) { | ||||
| @ -236,6 +236,17 @@ const onContextMenu = (e: any, item: ITalkRecord) => { | ||||
|   e.preventDefault() | ||||
| } | ||||
| 
 | ||||
| const onConvertText =async (data: ITalkRecord) => { | ||||
|   console.log('data',data) | ||||
|   data.is_convert_text = 1 | ||||
|   const res = await voiceToText({msgId:data.msg_id,voiceUrl:data.extra.url}) | ||||
|   if(res.code == 200){ | ||||
|     data.extra.content = res.data.convText | ||||
|   } | ||||
| } | ||||
| const onloseConvertText=(data: ITalkRecord)=>{ | ||||
|   data.is_convert_text = 0 | ||||
| } | ||||
| const evnets = { | ||||
|   copy: onCopyText, | ||||
|   revoke: onRevokeTalk, | ||||
| @ -243,7 +254,9 @@ const evnets = { | ||||
|   multiSelect: onMultiSelect, | ||||
|   download: onDownloadFile, | ||||
|   quote: onQuoteMessage, | ||||
|   collect: onCollectImage | ||||
|   collect: onCollectImage, | ||||
|   convertText: onConvertText, | ||||
|   closeConvertText:onloseConvertText | ||||
| } | ||||
| 
 | ||||
| // 会话列表右键菜单回调事件 | ||||
| @ -274,11 +287,7 @@ onMounted(() => { | ||||
| 
 | ||||
| <template> | ||||
|   <section class="section"> | ||||
|     <div | ||||
|       id="imChatPanel" | ||||
|       class="me-scrollbar me-scrollbar-thumb talk-container" | ||||
|       @scroll="onPanelScroll($event)" | ||||
|     > | ||||
|     <div id="imChatPanel" class="me-scrollbar me-scrollbar-thumb talk-container" @scroll="onPanelScroll($event)"> | ||||
|       <!-- 数据加载状态栏 --> | ||||
|       <div class="load-toolbar pointer"> | ||||
|         <span v-if="loadConfig.status == 0"> 正在加载数据中 ... </span> | ||||
| @ -286,89 +295,54 @@ onMounted(() => { | ||||
|         <span v-else class="no-more"> 没有更多消息了 </span> | ||||
|       </div> | ||||
| 
 | ||||
|       <div | ||||
|         class="message-item" | ||||
|         v-for="(item, index) in records" | ||||
|         :key="item.msg_id" | ||||
|         :id="item.msg_id" | ||||
|       > | ||||
|       <div class="message-item" v-for="(item, index) in records" :key="item.msg_id" :id="item.msg_id"> | ||||
|         <!-- 系统消息 --> | ||||
|         <div v-if="item.msg_type >= 1000" class="message-box"> | ||||
|           <component | ||||
|             :is="MessageComponents[item.msg_type] || 'unknown-message'" | ||||
|             :extra="item.extra" | ||||
|             :data="item" | ||||
|           /> | ||||
|           <component :is="MessageComponents[item.msg_type] || 'unknown-message'" :extra="item.extra" :data="item" /> | ||||
|         </div> | ||||
| 
 | ||||
|         <!-- 撤回消息 --> | ||||
|         <div v-else-if="item.is_revoke == 1" class="message-box"> | ||||
|           <revoke-message | ||||
|             :login_uid="uid" | ||||
|             :user_id="item.user_id" | ||||
|             :nickname="item.nickname" | ||||
|             :talk_type="item.talk_type" | ||||
|             :datetime="item.created_at" | ||||
|           /> | ||||
|           <revoke-message :login_uid="uid" :data="item" :user_id="item.user_id" :nickname="item.nickname" :talk_type="item.talk_type" | ||||
|             :datetime="item.created_at" /> | ||||
|         </div> | ||||
| 
 | ||||
|         <div | ||||
|           v-else | ||||
|           class="message-box record-box" | ||||
|           :class="{ | ||||
|             'direction-rt': item.float == 'right', | ||||
|             'multi-select': dialogueStore.isOpenMultiSelect, | ||||
|             'multi-select-check': item.isCheck | ||||
|           }" | ||||
|         > | ||||
|         <div v-else class="message-box record-box" :class="{ | ||||
|           'direction-rt': item.float == 'right', | ||||
|           'multi-select': dialogueStore.isOpenMultiSelect, | ||||
|           'multi-select-check': item.isCheck | ||||
|         }"> | ||||
|           <!-- 多选按钮 --> | ||||
|           <aside v-if="dialogueStore.isOpenMultiSelect" class="checkbox-column"> | ||||
|             <n-checkbox | ||||
|               size="small" | ||||
|               :checked="item.isCheck" | ||||
|               @update:checked="item.isCheck = !item.isCheck" | ||||
|             /> | ||||
|             <n-checkbox size="small" :checked="item.isCheck" @update:checked="item.isCheck = !item.isCheck" /> | ||||
|           </aside> | ||||
| 
 | ||||
|           <!-- 头像信息 --> | ||||
|             | ||||
|           <aside class="avatar-column"> | ||||
|             <im-avatar | ||||
|               class="pointer" | ||||
|               :src="item.avatar" | ||||
|               :size="30" | ||||
|               :username="item.nickname" | ||||
|               @click="showUserInfoModal(item.user_id)" | ||||
|             /> | ||||
|             <im-avatar class="pointer" :src="item.avatar" :size="42" :username="item.nickname" | ||||
|               @click="showUserInfoModal(item.erp_user_id)" /> | ||||
|           </aside> | ||||
| 
 | ||||
|           <!-- 主体信息 --> | ||||
|           <main class="main-column"> | ||||
|             <div class="talk-title"> | ||||
|               <span | ||||
|                 class="nickname pointer" | ||||
|                 v-show="talk_type == 2 && item.float == 'left'" | ||||
|                 @click="onClickNickname(item)" | ||||
|               > | ||||
|               <span class="nickname pointer" v-show="talk_type == 2 && item.float == 'left'" | ||||
|                 @click="onClickNickname(item)"> | ||||
|                 <span class="at">@</span>{{ item.nickname }} | ||||
|               </span> | ||||
|               <span>{{ parseTime(item.created_at, '{m}/{d} {h}:{i}') }}</span> | ||||
|               <span>{{ parseTime(item.created_at, '{y}/{m}/{d} {h}:{i}') }}</span> | ||||
|             </div> | ||||
| 
 | ||||
|             <div | ||||
|               class="talk-content" | ||||
|               :class="{ pointer: dialogueStore.isOpenMultiSelect }" | ||||
|               @click="onRowClick(item)" | ||||
|             > | ||||
|               <component | ||||
|                 :is="MessageComponents[item.msg_type] || 'unknown-message'" | ||||
|                 :extra="item.extra" | ||||
|                 :data="item" | ||||
|                 :max-width="true" | ||||
|                 :source="'panel'" | ||||
|                 @contextmenu.prevent="onContextMenu($event, item)" | ||||
|               /> | ||||
|             <div class="talk-content" :class="{ pointer: dialogueStore.isOpenMultiSelect }" @click="onRowClick(item)"> | ||||
| 
 | ||||
|               <div class="talk-tools"> | ||||
|               <component :is="MessageComponents[item.msg_type] || 'unknown-message'" :extra="item.extra" :data="item" | ||||
|                 :max-width="true" :source="'panel'" @contextmenu.prevent="onContextMenu($event, item)" /> | ||||
|               <div v-if="item.float==='right'&&item.extra.percentage===-1&&item.extra.is_uploading" class="mr-10px"> <n-button text style="font-size: 20px"> | ||||
|                   <n-icon color="#CF3050"> | ||||
|                     <ExclamationCircleFilled /> | ||||
|                   </n-icon> | ||||
|                 </n-button></div> | ||||
|               <!-- <div class="talk-tools"> | ||||
|                 <template v-if="talk_type == 1 && item.float == 'right'"> | ||||
|                   <loading | ||||
|                     theme="outline" | ||||
| @ -380,22 +354,14 @@ onMounted(() => { | ||||
|                   /> | ||||
| 
 | ||||
|                   <span v-show="item.send_status == 1"> 正在发送... </span> | ||||
|                   <!-- <span v-show="item.send_status != 1"> 已送达 </span> --> | ||||
|                   <span v-show="item.send_status != 1"> 已送达 </span> | ||||
|                 </template> | ||||
| 
 | ||||
|                 <n-icon | ||||
|                   class="more-tools pointer" | ||||
|                   :component="MoreThree" | ||||
|                   @click="onContextMenu($event, item)" | ||||
|                 /> | ||||
|               </div> | ||||
| <n-icon class="more-tools pointer" :component="MoreThree" @click="onContextMenu($event, item)" /> | ||||
| </div> --> | ||||
|             </div> | ||||
| 
 | ||||
|             <div | ||||
|               v-if="item.extra.reply" | ||||
|               class="talk-reply pointer" | ||||
|               @click="onJumpMessage(item.extra?.reply?.msg_id)" | ||||
|             > | ||||
|             <div v-if="item.extra.reply" class="talk-reply pointer" @click="onJumpMessage(item.extra?.reply?.msg_id)"> | ||||
|               <n-icon :component="ToTop" size="14" class="icon-top" /> | ||||
|               <span class="ellipsis"> | ||||
|                 回复 {{ item.extra?.reply?.nickname }}: | ||||
| @ -416,14 +382,8 @@ onMounted(() => { | ||||
|   </section> | ||||
| 
 | ||||
|   <!-- 右键菜单 --> | ||||
|   <n-dropdown | ||||
|     :show="dropdown.show" | ||||
|     :x="dropdown.x" | ||||
|     :y="dropdown.y" | ||||
|     :options="dropdown.options" | ||||
|     @select="onContextMenuHandle" | ||||
|     @clickoutside="closeDropdownMenu" | ||||
|   /> | ||||
|   <n-dropdown :show="dropdown.show" :x="dropdown.x" :y="dropdown.y" style="width: 142px;" :options="dropdown.options" | ||||
|     @select="onContextMenuHandle" @clickoutside="closeDropdownMenu" /> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| @ -448,6 +408,7 @@ onMounted(() => { | ||||
|     text-align: center; | ||||
|     line-height: 38px; | ||||
|     font-size: 13px; | ||||
| 
 | ||||
|     .no-more { | ||||
|       color: #b9b3b3; | ||||
|     } | ||||
| @ -489,7 +450,7 @@ onMounted(() => { | ||||
|     } | ||||
| 
 | ||||
|     .avatar-column { | ||||
|       width: 35px; | ||||
|       width: 47px; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       order: 2; | ||||
| @ -524,6 +485,7 @@ onMounted(() => { | ||||
|         .nickname { | ||||
|           color: var(--im-text-color); | ||||
|           margin-right: 5px; | ||||
|           font-size: 12px; | ||||
| 
 | ||||
|           .at { | ||||
|             display: none; | ||||
|  | ||||
| @ -5,16 +5,18 @@ import { | ||||
|   useDialogueStore, | ||||
|   useSettingsStore, | ||||
|   useUploadsStore, | ||||
|   useEditorStore | ||||
|   useEditorStore, | ||||
|   useUserStore | ||||
| } from '@/store' | ||||
| import ws from '@/connect' | ||||
| import { ServePublishMessage, ServeSendVote } from '@/api/chat' | ||||
| import { throttle, getVideoImage } from '@/utils/common' | ||||
| import { parseTime } from '@/utils/datetime' | ||||
| import Editor from '@/components/editor/Editor.vue' | ||||
| import MultiSelectFooter from './MultiSelectFooter.vue' | ||||
| import HistoryRecord from '@/components/talk/HistoryRecord.vue' | ||||
| import { uploadImg } from '@/api/upload' | ||||
| 
 | ||||
| const userStore = useUserStore() | ||||
| const talkStore = useTalkStore() | ||||
| const editorStore = useEditorStore() | ||||
| const settingsStore = useSettingsStore() | ||||
| @ -98,26 +100,78 @@ const onSendImageEvent = ({ data, callBack }) => { | ||||
| 
 | ||||
| // 发送视频消息 | ||||
| const onSendVideoEvent = async ({ data }) => { | ||||
|   let resp = await getVideoImage(data) | ||||
|   const form = new FormData() | ||||
|   form.append('file', data) | ||||
|   form.append("source", "fonchain-chat"); | ||||
|   form.append("type", "video"); | ||||
|   form.append("urlParam", `width=${resp.width}&height=${resp.height}`); | ||||
|   console.log('onSendVideoEvent') | ||||
|    | ||||
|   console.log(form.get('file')); | ||||
|   let video = await uploadImg(form) | ||||
|   if (video.code != 0) return | ||||
|   // 获取视频首帧作为封面图 | ||||
|   // let resp = await getVideoImage(data) | ||||
|    | ||||
|   let message = { | ||||
|     type: 'video', | ||||
|     url: video.data.ori_url, | ||||
|     cover: video.data.cover_url, | ||||
|     duration: parseInt(resp.duration), | ||||
|     size: data.size | ||||
|   // 先创建一个带有上传ID的临时消息对象,用于显示进度 | ||||
|   const uploadId = `video-${Date.now()}-${Math.floor(Math.random() * 1000)}` | ||||
|    | ||||
|   // 创建临时消息记录 | ||||
|   const tempMessage = { | ||||
|     msg_id: uploadId, | ||||
|     sequence: Date.now(), | ||||
|     talk_type: props.talk_type, | ||||
|     msg_type: 5, // 视频消息类型 | ||||
|     user_id: props.uid, | ||||
|     receiver_id: props.receiver_id, | ||||
|     nickname: '我', // 本地显示 | ||||
|     avatar: userStore.avatar, // 本地显示可能不需要 | ||||
|     is_revoke: 0, | ||||
|     is_mark: 0, | ||||
|     is_read: 1, | ||||
|     content: '', | ||||
|     created_at: parseTime(new Date(), '{y}-{m}-{d} {h}:{i}'), | ||||
|     extra: { | ||||
|       url: '', // 上传完成后会更新 | ||||
|       size: data.size, | ||||
|       is_uploading: true, | ||||
|       upload_id: uploadId, | ||||
|       percentage: 0 | ||||
|     }, | ||||
|     isCheck: false, | ||||
|     send_status: 1, | ||||
|     float: 'right' // 我发送的消息显示在右侧 | ||||
|   } | ||||
|    | ||||
|   onSendMessage(message, () => {}) | ||||
|   // 直接添加到对话记录中 | ||||
|   dialogueStore.addDialogueRecord(tempMessage)   | ||||
|   uploadsStore.initUploadFile( | ||||
|     data,  | ||||
|     props.talk_type,  | ||||
|     props.receiver_id,  | ||||
|     dialogueStore.talk.username,  | ||||
|     uploadId,  | ||||
|     async (percentage) => { | ||||
|       dialogueStore.updateUploadProgress(uploadId, percentage) | ||||
|     }, | ||||
|     async () => { | ||||
|       dialogueStore.batchDelDialogueRecord([uploadId]) | ||||
|       // console.log('videoData', videoData) | ||||
|       // // 上传完成后的回调 | ||||
|      | ||||
|       // // 更新临时消息为最终消息 | ||||
|       // dialogueStore.completeUpload(uploadId, { | ||||
|       //   url: videoData.data.ori_url, | ||||
|       //   cover: videoData.data.cover_url | ||||
|       // }) | ||||
|        | ||||
|       // // 上传成功后,发送正式消息给服务端 | ||||
|       // let finalMessage = { | ||||
|       //   type: 'video', | ||||
|       //   url: videoData.data.ori_url, | ||||
| 
 | ||||
|       //   size: data.size | ||||
|       // } | ||||
|        | ||||
|       // // 发送真实消息到服务端 | ||||
|       // onSendMessage(finalMessage, () => { | ||||
|       //   // 上传成功且消息发送成功后,删除临时消息 | ||||
|       //   dialogueStore.batchDelDialogueRecord([uploadId]) | ||||
|       // }) | ||||
|     } | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| // 发送代码消息 | ||||
| @ -131,8 +185,41 @@ const onSendFileEvent = ({ data }) => { | ||||
|   if (data.size > maxsize) { | ||||
|     return window['$message'].warning('上传文件不能超过100M!') | ||||
|   } | ||||
|   const uploadId = `file-${Date.now()}-${Math.floor(Math.random() * 1000)}` | ||||
| 
 | ||||
|   uploadsStore.initUploadFile(data, props.talk_type, props.receiver_id, dialogueStore.talk.username) | ||||
|   const tempMessage = { | ||||
|     msg_id: uploadId, | ||||
|     sequence: Date.now(), | ||||
|     talk_type: props.talk_type, | ||||
|     msg_type: 6, | ||||
|     user_id: props.uid, | ||||
|     receiver_id: props.receiver_id, | ||||
|     nickname: dialogueStore.talk.username, | ||||
|     avatar: userStore.avatar, | ||||
|     is_revoke: 0, | ||||
|     is_read: 0, | ||||
|     created_at: parseTime(new Date(), '{y}-{m}-{d} {h}:{i}'), | ||||
|     extra: { | ||||
|       name: data.name, | ||||
|       url: '',  | ||||
|       size: data.size, | ||||
|       is_uploading: true, | ||||
|       upload_id: uploadId, | ||||
|       percentage: 0 | ||||
|     }, | ||||
|     erp_user_id: 4692, | ||||
|     float: 'right' | ||||
|   } | ||||
|   dialogueStore.addDialogueRecord(tempMessage) | ||||
| 
 | ||||
|   uploadsStore.initUploadFile(data, props.talk_type, props.receiver_id, dialogueStore.talk.username,uploadId, | ||||
|   async (percentage) => { | ||||
|       dialogueStore.updateUploadProgress(uploadId, percentage) | ||||
|     }, | ||||
|     async () => { | ||||
|       dialogueStore.batchDelDialogueRecord([uploadId]) | ||||
|     } | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| // 发送投票消息 | ||||
|  | ||||
| @ -30,21 +30,29 @@ export function useMenu() { | ||||
|   }) | ||||
| 
 | ||||
|   const showDropdownMenu = (e: any, uid: number, item: any) => { | ||||
|     dropdown.item = Object.assign({}, item) | ||||
| 
 | ||||
|   //  dropdown.item = Object.assign({}, item)
 | ||||
|   dropdown.item = item | ||||
|     dropdown.options = [] | ||||
|     if ([4].includes(item.msg_type)) { | ||||
|       if(item.is_convert_text === 1){ | ||||
|         dropdown.options.push({ label: '关闭转文字', key: 'closeConvertText' }) | ||||
|       }else{ | ||||
|         dropdown.options.push({ label: '转文字', key: 'convertText' }) | ||||
|       } | ||||
|      | ||||
|     } | ||||
|     if ([1, 3].includes(item.msg_type)) { | ||||
|       dropdown.options.push({ label: '复制', key: 'copy' }) | ||||
|     } | ||||
| 
 | ||||
|     dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||
|     dropdown.options.push({ label: '引用', key: 'quote' }) | ||||
|     if (isRevoke(uid, item)) { | ||||
|       dropdown.options.push({ label: `撤回`, key: 'revoke' }) | ||||
|     } | ||||
| 
 | ||||
|     dropdown.options.push({ label: '回复', key: 'quote' }) | ||||
|     dropdown.options.push({ label: '删除', key: 'delete' }) | ||||
| 
 | ||||
|     dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||
| 
 | ||||
| 
 | ||||
|     if ([3, 4, 5].includes(item.msg_type)) { | ||||
|       dropdown.options.push({ label: '下载', key: 'download' }) | ||||
| @ -54,6 +62,7 @@ export function useMenu() { | ||||
|       dropdown.options.push({ label: '收藏', key: 'collect' }) | ||||
|     } | ||||
|     | ||||
| 
 | ||||
|     dropdown.x = e.clientX | ||||
|     dropdown.y = e.clientY | ||||
|     dropdown.show = true | ||||
|  | ||||
| @ -1,17 +1,22 @@ | ||||
| import { defineConfig, presetUno, presetAttributify, presetIcons } from 'unocss' | ||||
| import { defineConfig } from 'unocss' | ||||
| import { presetUno, presetAttributify, presetIcons } from 'unocss' | ||||
| 
 | ||||
| export default defineConfig({ | ||||
|   // 预设
 | ||||
|   presets: [ | ||||
|     presetUno(), // 添加核心预设
 | ||||
|     presetUno(),  | ||||
|     presetAttributify(), // 启用属性模式
 | ||||
|     presetIcons(), // 启用图标
 | ||||
|   ], | ||||
|   // 自定义规则
 | ||||
|   rules: [ | ||||
|     // 通过自定义规则覆盖默认的 container 样式
 | ||||
|     ['container', { 'max-width': 'none' }], // 或者根据需要设置其他样式
 | ||||
|   ], | ||||
|   safelist: [ | ||||
|     'container' // 确保 container 在 safelist 中,以便 UnoCSS 忽略它
 | ||||
|   ], | ||||
|   // 快捷方式
 | ||||
|   shortcuts: { | ||||
|     'btn': 'px-4 py-2 rounded-lg bg-blue-500 text-white hover:bg-blue-600', | ||||
|     'uno-container': 'container' // 创建 container 的别名
 | ||||
|   }, | ||||
| })  | ||||
| @ -4,6 +4,9 @@ import vue from '@vitejs/plugin-vue' | ||||
| import vueJsx from '@vitejs/plugin-vue-jsx' | ||||
| import compressPlugin from 'vite-plugin-compression' | ||||
| import UnoCSS from 'unocss/vite' | ||||
| import AutoImport from 'unplugin-auto-import/vite' | ||||
| import { NaiveUiResolver } from 'unplugin-vue-components/resolvers' | ||||
| import Components from 'unplugin-vue-components/vite' | ||||
| import vueDevTools from 'vite-plugin-vue-devtools'; | ||||
| // https://vitejs.dev/config/
 | ||||
| export default defineConfig(({ mode }) => { | ||||
| @ -24,12 +27,28 @@ export default defineConfig(({ mode }) => { | ||||
|     assetsInclude: ['./src/assets'], | ||||
|     plugins: [ | ||||
|       vue(),  | ||||
|       AutoImport({ | ||||
|         imports: [ | ||||
|           'vue', | ||||
|           { | ||||
|             'naive-ui': [ | ||||
|               'useDialog', | ||||
|               'useMessage', | ||||
|               'useNotification', | ||||
|               'useLoadingBar' | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }), | ||||
|       Components({ | ||||
|         resolvers: [NaiveUiResolver()] | ||||
|       }), | ||||
|       vueJsx({}),  | ||||
|       compressPlugin(),  | ||||
|       UnoCSS(), | ||||
|       // vueDevTools({
 | ||||
|       //   launchEditor: 'cursor',
 | ||||
|       // })
 | ||||
|       vueDevTools({ | ||||
|         launchEditor: 'cursor', | ||||
|       }) | ||||
|     ], | ||||
|     define: { | ||||
|       __APP_ENV__: env.APP_ENV | ||||
|  | ||||