Compare commits
	
		
			No commits in common. "3eaac91ba82ca48258098997263963443f8419a3" and "9360ecaaf98e43c00af8a8654e386a8cf945224d" have entirely different histories.
		
	
	
		
			3eaac91ba8
			...
			9360ecaaf9
		
	
		
							
								
								
									
										75
									
								
								auto-imports.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -1,75 +0,0 @@ | |||||||
| /* 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
									
									
								
							
							
						
						| @ -1,96 +0,0 @@ | |||||||
| /* 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,7 +2,9 @@ ENV = 'development' | |||||||
| 
 | 
 | ||||||
| VITE_BASE=/ | VITE_BASE=/ | ||||||
| VUE_APP_PREVIEW=false | VUE_APP_PREVIEW=false | ||||||
| VITE_BASE_API=http://114.218.158.24:8503 | VITE_BASE_API=http://172.16.100.93:8503 | ||||||
|  | # VITE_BASE_API=http://192.168.88.21:9503 | ||||||
| VITE_EPR_BASEURL=http://114.218.158.24:9020 | VITE_EPR_BASEURL=http://114.218.158.24:9020 | ||||||
| VITE_SOCKET_API=ws://114.218.158.24:8504 | VITE_SOCKET_API=ws://172.16.100.93:8504 | ||||||
|  | # VITE_SOCKET_API=ws://192.168.88.21:9504 | ||||||
| VUE_APP_WEBSITE_NAME="Lumen IM" | VUE_APP_WEBSITE_NAME="Lumen IM" | ||||||
| @ -32,8 +32,6 @@ | |||||||
|     "quill": "^1.3.7", |     "quill": "^1.3.7", | ||||||
|     "quill-image-uploader": "^1.3.0", |     "quill-image-uploader": "^1.3.0", | ||||||
|     "quill-mention": "^4.1.0", |     "quill-mention": "^4.1.0", | ||||||
|     "sortablejs": "^1.15.6", |  | ||||||
|     "viewerjs": "^1.11.7", |  | ||||||
|     "vue": "^3.3.11", |     "vue": "^3.3.11", | ||||||
|     "vue-cropper": "^1.1.1", |     "vue-cropper": "^1.1.1", | ||||||
|     "vue-router": "^4.2.5", |     "vue-router": "^4.2.5", | ||||||
| @ -46,7 +44,6 @@ | |||||||
|     "@tsconfig/node18": "^18.2.2", |     "@tsconfig/node18": "^18.2.2", | ||||||
|     "@types/node": "^18.18.5", |     "@types/node": "^18.18.5", | ||||||
|     "@types/vue": "^2.0.0", |     "@types/vue": "^2.0.0", | ||||||
|     "@unocss/reset": "^66.1.1", |  | ||||||
|     "@vitejs/plugin-vue": "^4.4.0", |     "@vitejs/plugin-vue": "^4.4.0", | ||||||
|     "@vitejs/plugin-vue-jsx": "^3.0.2", |     "@vitejs/plugin-vue-jsx": "^3.0.2", | ||||||
|     "@vue/tsconfig": "^0.4.0", |     "@vue/tsconfig": "^0.4.0", | ||||||
| @ -60,8 +57,6 @@ | |||||||
|     "sass": "^1.88.0", |     "sass": "^1.88.0", | ||||||
|     "typescript": "~5.2.0", |     "typescript": "~5.2.0", | ||||||
|     "unocss": "0.58.0", |     "unocss": "0.58.0", | ||||||
|     "unplugin-auto-import": "^19.2.0", |  | ||||||
|     "unplugin-vue-components": "^28.5.0", |  | ||||||
|     "vite": "^4.5.1", |     "vite": "^4.5.1", | ||||||
|     "vite-plugin-compression": "^0.5.1", |     "vite-plugin-compression": "^0.5.1", | ||||||
|     "vite-plugin-vue-devtools": "^7.7.6", |     "vite-plugin-vue-devtools": "^7.7.6", | ||||||
|  | |||||||
							
								
								
									
										178
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
							
						
						| @ -62,12 +62,6 @@ importers: | |||||||
|       quill-mention: |       quill-mention: | ||||||
|         specifier: ^4.1.0 |         specifier: ^4.1.0 | ||||||
|         version: 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: |       vue: | ||||||
|         specifier: ^3.3.11 |         specifier: ^3.3.11 | ||||||
|         version: 3.5.13(typescript@5.2.2) |         version: 3.5.13(typescript@5.2.2) | ||||||
| @ -99,9 +93,6 @@ importers: | |||||||
|       '@types/vue': |       '@types/vue': | ||||||
|         specifier: ^2.0.0 |         specifier: ^2.0.0 | ||||||
|         version: 2.0.0(typescript@5.2.2) |         version: 2.0.0(typescript@5.2.2) | ||||||
|       '@unocss/reset': |  | ||||||
|         specifier: ^66.1.1 |  | ||||||
|         version: 66.1.1 |  | ||||||
|       '@vitejs/plugin-vue': |       '@vitejs/plugin-vue': | ||||||
|         specifier: ^4.4.0 |         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)) |         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)) | ||||||
| @ -141,12 +132,6 @@ importers: | |||||||
|       unocss: |       unocss: | ||||||
|         specifier: 0.58.0 |         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)) |         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: |       vite: | ||||||
|         specifier: ^4.5.1 |         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) |         version: 4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0) | ||||||
| @ -818,9 +803,6 @@ packages: | |||||||
|   '@unocss/reset@0.58.0': |   '@unocss/reset@0.58.0': | ||||||
|     resolution: {integrity: sha512-UVZ5kz37JGbwAA06k/gjKYcekcTwi6oIhev1EpTtCvHLL6XYcYqcwb/u4Wjzprd3L3lxDGYXvGdjREGm2u7vbQ==} |     resolution: {integrity: sha512-UVZ5kz37JGbwAA06k/gjKYcekcTwi6oIhev1EpTtCvHLL6XYcYqcwb/u4Wjzprd3L3lxDGYXvGdjREGm2u7vbQ==} | ||||||
| 
 | 
 | ||||||
|   '@unocss/reset@66.1.1': |  | ||||||
|     resolution: {integrity: sha512-WrI3sStMd/EXTcb3SaTVH10Wc9NKutW4+/HktQy470wEpncXdvihrXgCYwJH6LEEL4KOto3o+KKSD5xenWE7Aw==} |  | ||||||
| 
 |  | ||||||
|   '@unocss/rule-utils@0.58.0': |   '@unocss/rule-utils@0.58.0': | ||||||
|     resolution: {integrity: sha512-LBJ9dJ/j5UIMzJF7pmIig55MtJAYtG+tn/zQRveZuPRVahzP+KqwlyB7u3uCUnQhdgo/MJODMcqyr0jl6+kTuA==} |     resolution: {integrity: sha512-LBJ9dJ/j5UIMzJF7pmIig55MtJAYtG+tn/zQRveZuPRVahzP+KqwlyB7u3uCUnQhdgo/MJODMcqyr0jl6+kTuA==} | ||||||
|     engines: {node: '>=14'} |     engines: {node: '>=14'} | ||||||
| @ -1709,10 +1691,6 @@ packages: | |||||||
|     resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} |     resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} | ||||||
|     engines: {node: '>=0.8.0'} |     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: |   eslint-scope@5.1.1: | ||||||
|     resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} |     resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} | ||||||
|     engines: {node: '>=8.0.0'} |     engines: {node: '>=8.0.0'} | ||||||
| @ -1741,9 +1719,6 @@ packages: | |||||||
|   estree-walker@2.0.2: |   estree-walker@2.0.2: | ||||||
|     resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==} |     resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==} | ||||||
| 
 | 
 | ||||||
|   estree-walker@3.0.3: |  | ||||||
|     resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} |  | ||||||
| 
 |  | ||||||
|   event-emitter@0.3.5: |   event-emitter@0.3.5: | ||||||
|     resolution: {integrity: sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==} |     resolution: {integrity: sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==} | ||||||
| 
 | 
 | ||||||
| @ -1812,14 +1787,6 @@ packages: | |||||||
|   fastq@1.19.1: |   fastq@1.19.1: | ||||||
|     resolution: {integrity: sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==} |     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: |   figures@6.1.0: | ||||||
|     resolution: {integrity: sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg==} |     resolution: {integrity: sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg==} | ||||||
|     engines: {node: '>=18'} |     engines: {node: '>=18'} | ||||||
| @ -2186,9 +2153,6 @@ packages: | |||||||
|   js-tokens@4.0.0: |   js-tokens@4.0.0: | ||||||
|     resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} |     resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} | ||||||
| 
 | 
 | ||||||
|   js-tokens@9.0.1: |  | ||||||
|     resolution: {integrity: sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==} |  | ||||||
| 
 |  | ||||||
|   js-yaml@3.14.1: |   js-yaml@3.14.1: | ||||||
|     resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} |     resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} | ||||||
|     hasBin: true |     hasBin: true | ||||||
| @ -2849,9 +2813,6 @@ packages: | |||||||
|   scroll-into-view-if-needed@2.2.31: |   scroll-into-view-if-needed@2.2.31: | ||||||
|     resolution: {integrity: sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA==} |     resolution: {integrity: sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA==} | ||||||
| 
 | 
 | ||||||
|   scule@1.3.0: |  | ||||||
|     resolution: {integrity: sha512-6FtHJEvt+pVMIB9IBY+IcCJ6Z5f1iQnytgyfKMhDKgmzYG+TeH/wx1y3l27rshSbLiSanrR9ffZDrEsmjlQF2g==} |  | ||||||
| 
 |  | ||||||
|   section-matter@1.0.0: |   section-matter@1.0.0: | ||||||
|     resolution: {integrity: sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==} |     resolution: {integrity: sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==} | ||||||
|     engines: {node: '>=4'} |     engines: {node: '>=4'} | ||||||
| @ -2933,9 +2894,6 @@ packages: | |||||||
|   sortablejs@1.14.0: |   sortablejs@1.14.0: | ||||||
|     resolution: {integrity: sha512-pBXvQCs5/33fdN1/39pPL0NZF20LeRbLQ5jtnheIPN9JQAaufGjKdWduZn4U7wCtVuzKhmRkI0DFYHYRbB2H1w==} |     resolution: {integrity: sha512-pBXvQCs5/33fdN1/39pPL0NZF20LeRbLQ5jtnheIPN9JQAaufGjKdWduZn4U7wCtVuzKhmRkI0DFYHYRbB2H1w==} | ||||||
| 
 | 
 | ||||||
|   sortablejs@1.15.6: |  | ||||||
|     resolution: {integrity: sha512-aNfiuwMEpfBM/CN6LY0ibyhxPfPbyFeBTYJKCvzkJ2GkUpazIt3H+QIPAMHwqQ7tMKaHz1Qj+rJJCqljnf4p3A==} |  | ||||||
| 
 |  | ||||||
|   source-map-js@1.2.1: |   source-map-js@1.2.1: | ||||||
|     resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} |     resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} | ||||||
|     engines: {node: '>=0.10.0'} |     engines: {node: '>=0.10.0'} | ||||||
| @ -2993,9 +2951,6 @@ packages: | |||||||
|     resolution: {integrity: sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw==} |     resolution: {integrity: sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw==} | ||||||
|     engines: {node: '>=18'} |     engines: {node: '>=18'} | ||||||
| 
 | 
 | ||||||
|   strip-literal@3.0.0: |  | ||||||
|     resolution: {integrity: sha512-TcccoMhJOM3OebGhSBEmp3UZ2SfDMZUEBdRA/9ynfLi8yYajyWX3JiXArcJt4Umh4vISpspkQIY8ZZoCqjbviA==} |  | ||||||
| 
 |  | ||||||
|   stylis@4.3.6: |   stylis@4.3.6: | ||||||
|     resolution: {integrity: sha512-yQ3rwFWRfwNUY7H5vpU0wfdkNSnvnJinhF9830Swlaxl03zsOjCfmX0ugac+3LtK0lYSgwL/KXc8oYL3mG4YFQ==} |     resolution: {integrity: sha512-yQ3rwFWRfwNUY7H5vpU0wfdkNSnvnJinhF9830Swlaxl03zsOjCfmX0ugac+3LtK0lYSgwL/KXc8oYL3mG4YFQ==} | ||||||
| 
 | 
 | ||||||
| @ -3047,10 +3002,6 @@ packages: | |||||||
|   tinyexec@1.0.1: |   tinyexec@1.0.1: | ||||||
|     resolution: {integrity: sha512-5uC6DDlmeqiOwCPmK9jMSdOuZTh8bU39Ys6yidB+UTt5hfZUPGAypSgFRiEp+jbi9qH40BLDvy85jIU88wKSqw==} |     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: |   to-object-path@0.3.0: | ||||||
|     resolution: {integrity: sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==} |     resolution: {integrity: sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==} | ||||||
|     engines: {node: '>=0.10.0'} |     engines: {node: '>=0.10.0'} | ||||||
| @ -3115,10 +3066,6 @@ packages: | |||||||
|     resolution: {integrity: sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==} |     resolution: {integrity: sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==} | ||||||
|     engines: {node: '>=18'} |     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: |   union-value@1.0.1: | ||||||
|     resolution: {integrity: sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==} |     resolution: {integrity: sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==} | ||||||
|     engines: {node: '>=0.10.0'} |     engines: {node: '>=0.10.0'} | ||||||
| @ -3146,39 +3093,6 @@ packages: | |||||||
|       vite: |       vite: | ||||||
|         optional: true |         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: |   unset-value@1.0.0: | ||||||
|     resolution: {integrity: sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==} |     resolution: {integrity: sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==} | ||||||
|     engines: {node: '>=0.10.0'} |     engines: {node: '>=0.10.0'} | ||||||
| @ -3220,9 +3134,6 @@ packages: | |||||||
|     peerDependencies: |     peerDependencies: | ||||||
|       vue: ^3.0.11 |       vue: ^3.0.11 | ||||||
| 
 | 
 | ||||||
|   viewerjs@1.11.7: |  | ||||||
|     resolution: {integrity: sha512-0JuVqOmL5v1jmEAlG5EBDR3XquxY8DWFQbFMprOXgaBB0F7Q/X9xWdEaQc59D8xzwkdUgXEMSSknTpriq95igg==} |  | ||||||
| 
 |  | ||||||
|   vite-hot-client@2.0.4: |   vite-hot-client@2.0.4: | ||||||
|     resolution: {integrity: sha512-W9LOGAyGMrbGArYJN4LBCdOC5+Zwh7dHvOHC0KmGKkJhsOzaKbpo/jEjpPKVHIW0/jBWj8RZG0NUxfgA8BxgAg==} |     resolution: {integrity: sha512-W9LOGAyGMrbGArYJN4LBCdOC5+Zwh7dHvOHC0KmGKkJhsOzaKbpo/jEjpPKVHIW0/jBWj8RZG0NUxfgA8BxgAg==} | ||||||
|     peerDependencies: |     peerDependencies: | ||||||
| @ -3377,9 +3288,6 @@ packages: | |||||||
|     resolution: {integrity: sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==} |     resolution: {integrity: sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==} | ||||||
|     engines: {node: '>=10.13.0'} |     engines: {node: '>=10.13.0'} | ||||||
| 
 | 
 | ||||||
|   webpack-virtual-modules@0.6.2: |  | ||||||
|     resolution: {integrity: sha512-66/V2i5hQanC51vBQKPH4aI8NMAcBW59FVBs+rC7eGHupMyfn34q7rZIE+ETlJ+XTevqfUhVVBgSUNSW2flEUQ==} |  | ||||||
| 
 |  | ||||||
|   webpack@5.99.8: |   webpack@5.99.8: | ||||||
|     resolution: {integrity: sha512-lQ3CPiSTpfOnrEGeXDwoq5hIGzSjmwD72GdfVzF7CQAI7t47rJG9eDWvcEkEn3CUQymAElVvDg3YNTlCYj+qUQ==} |     resolution: {integrity: sha512-lQ3CPiSTpfOnrEGeXDwoq5hIGzSjmwD72GdfVzF7CQAI7t47rJG9eDWvcEkEn3CUQymAElVvDg3YNTlCYj+qUQ==} | ||||||
|     engines: {node: '>=10.13.0'} |     engines: {node: '>=10.13.0'} | ||||||
| @ -4120,8 +4028,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   '@unocss/reset@0.58.0': {} |   '@unocss/reset@0.58.0': {} | ||||||
| 
 | 
 | ||||||
|   '@unocss/reset@66.1.1': {} |  | ||||||
| 
 |  | ||||||
|   '@unocss/rule-utils@0.58.0': |   '@unocss/rule-utils@0.58.0': | ||||||
|     dependencies: |     dependencies: | ||||||
|       '@unocss/core': 0.58.0 |       '@unocss/core': 0.58.0 | ||||||
| @ -5206,8 +5112,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   escape-string-regexp@1.0.5: {} |   escape-string-regexp@1.0.5: {} | ||||||
| 
 | 
 | ||||||
|   escape-string-regexp@5.0.0: {} |  | ||||||
| 
 |  | ||||||
|   eslint-scope@5.1.1: |   eslint-scope@5.1.1: | ||||||
|     dependencies: |     dependencies: | ||||||
|       esrecurse: 4.3.0 |       esrecurse: 4.3.0 | ||||||
| @ -5232,10 +5136,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   estree-walker@2.0.2: {} |   estree-walker@2.0.2: {} | ||||||
| 
 | 
 | ||||||
|   estree-walker@3.0.3: |  | ||||||
|     dependencies: |  | ||||||
|       '@types/estree': 1.0.7 |  | ||||||
| 
 |  | ||||||
|   event-emitter@0.3.5: |   event-emitter@0.3.5: | ||||||
|     dependencies: |     dependencies: | ||||||
|       d: 1.0.2 |       d: 1.0.2 | ||||||
| @ -5337,10 +5237,6 @@ snapshots: | |||||||
|     dependencies: |     dependencies: | ||||||
|       reusify: 1.1.0 |       reusify: 1.1.0 | ||||||
| 
 | 
 | ||||||
|   fdir@6.4.4(picomatch@4.0.2): |  | ||||||
|     optionalDependencies: |  | ||||||
|       picomatch: 4.0.2 |  | ||||||
| 
 |  | ||||||
|   figures@6.1.0: |   figures@6.1.0: | ||||||
|     dependencies: |     dependencies: | ||||||
|       is-unicode-supported: 2.1.0 |       is-unicode-supported: 2.1.0 | ||||||
| @ -5679,8 +5575,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   js-tokens@4.0.0: {} |   js-tokens@4.0.0: {} | ||||||
| 
 | 
 | ||||||
|   js-tokens@9.0.1: {} |  | ||||||
| 
 |  | ||||||
|   js-yaml@3.14.1: |   js-yaml@3.14.1: | ||||||
|     dependencies: |     dependencies: | ||||||
|       argparse: 1.0.10 |       argparse: 1.0.10 | ||||||
| @ -6438,8 +6332,6 @@ snapshots: | |||||||
|     dependencies: |     dependencies: | ||||||
|       compute-scroll-into-view: 1.0.20 |       compute-scroll-into-view: 1.0.20 | ||||||
| 
 | 
 | ||||||
|   scule@1.3.0: {} |  | ||||||
| 
 |  | ||||||
|   section-matter@1.0.0: |   section-matter@1.0.0: | ||||||
|     dependencies: |     dependencies: | ||||||
|       extend-shallow: 2.0.1 |       extend-shallow: 2.0.1 | ||||||
| @ -6532,8 +6424,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   sortablejs@1.14.0: {} |   sortablejs@1.14.0: {} | ||||||
| 
 | 
 | ||||||
|   sortablejs@1.15.6: {} |  | ||||||
| 
 |  | ||||||
|   source-map-js@1.2.1: {} |   source-map-js@1.2.1: {} | ||||||
| 
 | 
 | ||||||
|   source-map-resolve@0.5.3: |   source-map-resolve@0.5.3: | ||||||
| @ -6584,10 +6474,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   strip-final-newline@4.0.0: {} |   strip-final-newline@4.0.0: {} | ||||||
| 
 | 
 | ||||||
|   strip-literal@3.0.0: |  | ||||||
|     dependencies: |  | ||||||
|       js-tokens: 9.0.1 |  | ||||||
| 
 |  | ||||||
|   stylis@4.3.6: {} |   stylis@4.3.6: {} | ||||||
| 
 | 
 | ||||||
|   superjson@2.2.2: |   superjson@2.2.2: | ||||||
| @ -6628,11 +6514,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   tinyexec@1.0.1: {} |   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: |   to-object-path@0.3.0: | ||||||
|     dependencies: |     dependencies: | ||||||
|       kind-of: 3.2.2 |       kind-of: 3.2.2 | ||||||
| @ -6685,23 +6566,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   unicorn-magic@0.3.0: {} |   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: |   union-value@1.0.1: | ||||||
|     dependencies: |     dependencies: | ||||||
|       arr-union: 3.1.0 |       arr-union: 3.1.0 | ||||||
| @ -6746,44 +6610,6 @@ snapshots: | |||||||
|       - rollup |       - rollup | ||||||
|       - supports-color |       - 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: |   unset-value@1.0.0: | ||||||
|     dependencies: |     dependencies: | ||||||
|       has-value: 0.3.1 |       has-value: 0.3.1 | ||||||
| @ -6822,8 +6648,6 @@ snapshots: | |||||||
|       evtd: 0.2.4 |       evtd: 0.2.4 | ||||||
|       vue: 3.5.13(typescript@5.2.2) |       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)): |   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: |     dependencies: | ||||||
|       vite: 4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0) |       vite: 4.5.14(@types/node@18.19.99)(less@4.3.0)(sass@1.88.0)(terser@5.39.0) | ||||||
| @ -6998,8 +6822,6 @@ snapshots: | |||||||
| 
 | 
 | ||||||
|   webpack-sources@3.2.3: {} |   webpack-sources@3.2.3: {} | ||||||
| 
 | 
 | ||||||
|   webpack-virtual-modules@0.6.2: {} |  | ||||||
| 
 |  | ||||||
|   webpack@5.99.8: |   webpack@5.99.8: | ||||||
|     dependencies: |     dependencies: | ||||||
|       '@types/eslint-scope': 3.7.7 |       '@types/eslint-scope': 3.7.7 | ||||||
|  | |||||||
| @ -25,7 +25,3 @@ export const ServeRefreshToken = () => { | |||||||
| export const ServeForgetPassword = (data) => { | export const ServeForgetPassword = (data) => { | ||||||
|   return post('/api/v1/auth/forget', data) |   return post('/api/v1/auth/forget', data) | ||||||
| } | } | ||||||
| // 获取用户信息服务
 |  | ||||||
| export const GetUserInfo = (data) => { |  | ||||||
|   return post('/api/v1/users/info', data) |  | ||||||
| } |  | ||||||
| @ -9,10 +9,7 @@ export const ServeGetTalkList = (data = {}) => { | |||||||
| export const ServeCreateTalkList = (data = {}) => { | export const ServeCreateTalkList = (data = {}) => { | ||||||
|   return post('/api/v1/talk/create', 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 = {}) => { | export const ServeDeleteTalkList = (data = {}) => { | ||||||
|   return post('/api/v1/talk/delete', data) |   return post('/api/v1/talk/delete', data) | ||||||
|  | |||||||
| @ -45,12 +45,10 @@ export const ServeFindFriendApplyNum = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 搜索用户信息服务接口
 | // 搜索用户信息服务接口
 | ||||||
| // export const ServeSearchUser = (data) => {
 |  | ||||||
| //   return get('/api/v1/contact/detail', data)
 |  | ||||||
| // }
 |  | ||||||
| export const ServeSearchUser = (data) => { | export const ServeSearchUser = (data) => { | ||||||
|   return post('/api/v1/users/info', data) |   return get('/api/v1/contact/detail', data) | ||||||
| } | } | ||||||
|  | 
 | ||||||
| // 搜索用户信息服务接口
 | // 搜索用户信息服务接口
 | ||||||
| export const ServeContactGroupList = (data) => { | export const ServeContactGroupList = (data) => { | ||||||
|   return get('/api/v1/contact/group/list', data) |   return get('/api/v1/contact/group/list', data) | ||||||
|  | |||||||
| @ -21,9 +21,6 @@ export const ServeFileSubareaUpload = (data = {}, options = {}) => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 上传图片文件或者视频
 | // 上传图片文件或者视频
 | ||||||
| export const uploadImg = (data, signal) => { | export const uploadImg = (data) => { | ||||||
|   return post('/upload/img', data, { |   return post('/upload/img', data,{baseURL:import.meta.env.VITE_EPR_BASEURL}) | ||||||
|     baseURL: import.meta.env.VITE_EPR_BASEURL, |  | ||||||
|     signal: signal |  | ||||||
|   }) |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,7 +1,6 @@ | |||||||
| * { | * { | ||||||
|   margin: 0; |   margin: 0; | ||||||
|   padding: 0; |   padding: 0; | ||||||
|   box-sizing: border-box!important; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @font-face { | @font-face { | ||||||
|  | |||||||
| @ -1,10 +1,10 @@ | |||||||
| // 默认主题 | // 默认主题 | ||||||
| html { | html { | ||||||
|   --im-primary-color: #462AA0; |   --im-primary-color: #1890ff; | ||||||
|   --im-bg-color: #ffffff; |   --im-bg-color: #ffffff; | ||||||
|   --line-border-color: #f5f5f5; |   --line-border-color: #f5f5f5; | ||||||
|   --border-color: #eeeaea; |   --border-color: #eeeaea; | ||||||
|   --im-text-color: #BABABA; |   --im-text-color: #333; | ||||||
|   --im-text-color-grey: #333; |   --im-text-color-grey: #333; | ||||||
|   --im-active-bg-color: #f5f5f5; |   --im-active-bg-color: #f5f5f5; | ||||||
|   --im-hover-bg-color: #f5f5f5; |   --im-hover-bg-color: #f5f5f5; | ||||||
| @ -21,10 +21,10 @@ html { | |||||||
|   // message |   // message | ||||||
|   --im-message-bg-color: #f7f7f7; |   --im-message-bg-color: #f7f7f7; | ||||||
|   --im-message-border-color: #efeff5; |   --im-message-border-color: #efeff5; | ||||||
|   --im-message-left-bg-color: #F4F4FC; |   --im-message-left-bg-color: #eff0f1; | ||||||
|   --im-message-left-text-color: #333; |   --im-message-left-text-color: #333; | ||||||
|   --im-message-right-bg-color: #46299D; |   --im-message-right-bg-color: #daf3fd; | ||||||
|   --im-message-right-text-color: #fff; |   --im-message-right-text-color: #333; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 黑色主题 | // 黑色主题 | ||||||
|  | |||||||
| Before Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 2.3 KiB | 
| Before Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 2.2 KiB | 
| Before Width: | Height: | Size: 607 B | 
| Before Width: | Height: | Size: 5.6 KiB After Width: | Height: | Size: 6.6 KiB | 
| Before Width: | Height: | Size: 5.1 KiB After Width: | Height: | Size: 5.5 KiB | 
| Before Width: | Height: | Size: 1.8 KiB | 
| Before Width: | Height: | Size: 1.5 KiB | 
| Before Width: | Height: | Size: 2.9 KiB | 
| Before Width: | Height: | Size: 118 KiB | 
| Before Width: | Height: | Size: 4.3 KiB | 
| Before Width: | Height: | Size: 4.5 KiB | 
| Before Width: | Height: | Size: 4.5 KiB | 
| Before Width: | Height: | Size: 1.3 KiB | 
| @ -87,8 +87,8 @@ const text_avatar = computed(() => { | |||||||
|   background: linear-gradient(to right, #674bbc, #46299d); |   background: linear-gradient(to right, #674bbc, #46299d); | ||||||
|   flex-shrink: 0; |   flex-shrink: 0; | ||||||
|   img { |   img { | ||||||
|     width: 42px; |     width: 100%; | ||||||
|     height: 42px; |     height: 100%; | ||||||
|     object-fit: cover; |     object-fit: cover; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,117 +1,81 @@ | |||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| // 引入Quill编辑器的样式文件 |  | ||||||
| import '@vueup/vue-quill/dist/vue-quill.snow.css' | import '@vueup/vue-quill/dist/vue-quill.snow.css' | ||||||
| // 引入图片上传插件的样式 |  | ||||||
| import 'quill-image-uploader/dist/quill.imageUploader.min.css' | import 'quill-image-uploader/dist/quill.imageUploader.min.css' | ||||||
| // 引入自定义的提及功能样式 |  | ||||||
| import '@/assets/css/editor-mention.less' | import '@/assets/css/editor-mention.less' | ||||||
| // 引入Vue核心功能 |  | ||||||
| import { reactive, watch, ref, markRaw, computed, onMounted, onUnmounted } from 'vue' | import { reactive, watch, ref, markRaw, computed, onMounted, onUnmounted } from 'vue' | ||||||
| // 引入Naive UI的弹出框组件 |  | ||||||
| import { NPopover } from 'naive-ui' | import { NPopover } from 'naive-ui' | ||||||
| // 引入图标组件 |  | ||||||
| import { | import { | ||||||
|   Voice as IconVoice,    // 语音图标 |   Voice as IconVoice, | ||||||
|   SourceCode,            // 代码图标 |   SourceCode, | ||||||
|   Local,                 // 地理位置图标 |   Local, | ||||||
|   SmilingFace,           // 表情图标 |   SmilingFace, | ||||||
|   Pic,                   // 图片图标 |   Pic, | ||||||
|   FolderUpload,          // 文件上传图标 |   FolderUpload, | ||||||
|   Ranking,               // 排名图标(用于投票) |   Ranking, | ||||||
|   History                // 历史记录图标 |   History | ||||||
| } from '@icon-park/vue-next' | } from '@icon-park/vue-next' | ||||||
| // 引入Quill编辑器及其核心实例 |  | ||||||
| import { QuillEditor, Quill } from '@vueup/vue-quill' | import { QuillEditor, Quill } from '@vueup/vue-quill' | ||||||
| // 引入图片上传插件 |  | ||||||
| import ImageUploader from 'quill-image-uploader' | import ImageUploader from 'quill-image-uploader' | ||||||
| // 引入自定义表情符号格式 |  | ||||||
| import EmojiBlot from './formats/emoji' | import EmojiBlot from './formats/emoji' | ||||||
| // 引入自定义引用格式 |  | ||||||
| import QuoteBlot from './formats/quote' | import QuoteBlot from './formats/quote' | ||||||
| // 引入提及功能 |  | ||||||
| import 'quill-mention' | import 'quill-mention' | ||||||
| // 引入状态管理 |  | ||||||
| import { useDialogueStore, useEditorDraftStore } from '@/store' | import { useDialogueStore, useEditorDraftStore } from '@/store' | ||||||
| // 引入编辑器工具函数 |  | ||||||
| import { deltaToMessage, deltaToString, isEmptyDelta } from './util' | import { deltaToMessage, deltaToString, isEmptyDelta } from './util' | ||||||
| // 引入获取图片信息的工具函数 |  | ||||||
| import { getImageInfo } from '@/utils/functions' | import { getImageInfo } from '@/utils/functions' | ||||||
| // 引入编辑器常量定义 |  | ||||||
| import { EditorConst } from '@/constant/event-bus' | import { EditorConst } from '@/constant/event-bus' | ||||||
| // 引入事件调用工具 |  | ||||||
| import { emitCall } from '@/utils/common' | import { emitCall } from '@/utils/common' | ||||||
| // 引入默认头像常量 |  | ||||||
| import { defAvatar } from '@/constant/default' | import { defAvatar } from '@/constant/default' | ||||||
| // 引入编辑器各子组件 | import MeEditorVote from './MeEditorVote.vue' | ||||||
| import MeEditorVote from './MeEditorVote.vue'            // 投票组件 | import MeEditorEmoticon from './MeEditorEmoticon.vue' | ||||||
| import MeEditorEmoticon from './MeEditorEmoticon.vue'    // 表情组件 | import MeEditorCode from './MeEditorCode.vue' | ||||||
| import MeEditorCode from './MeEditorCode.vue'            // 代码编辑组件 | import MeEditorRecorder from './MeEditorRecorder.vue' | ||||||
| import MeEditorRecorder from './MeEditorRecorder.vue'    // 录音组件 |  | ||||||
| // 引入上传API |  | ||||||
| import { ServeUploadImage } from '@/api/upload' | import { ServeUploadImage } from '@/api/upload' | ||||||
| import { uploadImg } from '@/api/upload' | import { uploadImg } from '@/api/upload' | ||||||
| // 引入事件总线钩子 |  | ||||||
| import { useEventBus } from '@/hooks' | import { useEventBus } from '@/hooks' | ||||||
| 
 | 
 | ||||||
| // 注册Quill编辑器的自定义格式 | Quill.register('formats/emoji', EmojiBlot) | ||||||
| Quill.register('formats/emoji', EmojiBlot)       // 注册表情格式 | Quill.register('formats/quote', QuoteBlot) | ||||||
| Quill.register('formats/quote', QuoteBlot)       // 注册引用格式 | Quill.register('modules/imageUploader', ImageUploader) | ||||||
| Quill.register('modules/imageUploader', ImageUploader)  // 注册图片上传模块 |  | ||||||
| 
 | 
 | ||||||
| // 定义组件的事件 |  | ||||||
| const emit = defineEmits(['editor-event']) | const emit = defineEmits(['editor-event']) | ||||||
| // 获取对话状态管理 |  | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| // 获取编辑器草稿状态管理 |  | ||||||
| const editorDraftStore = useEditorDraftStore() | const editorDraftStore = useEditorDraftStore() | ||||||
| // 定义组件props |  | ||||||
| const props = defineProps({ | const props = defineProps({ | ||||||
|   vote: { |   vote: { | ||||||
|     type: Boolean, |     type: Boolean, | ||||||
|     default: false  // 是否显示投票功能 |     default: false | ||||||
|   }, |   }, | ||||||
|   members: { |   members: { | ||||||
|     default: () => []  // 聊天成员列表,用于@功能 |     default: () => [] | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| // 编辑器引用 |  | ||||||
| const editor = ref() | const editor = ref() | ||||||
| 
 | 
 | ||||||
| // 获取Quill编辑器实例 |  | ||||||
| const getQuill = () => { | const getQuill = () => { | ||||||
|   return editor.value?.getQuill() |   return editor.value?.getQuill() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 获取当前编辑器光标位置 |  | ||||||
| const getQuillSelectionIndex = () => { | const getQuillSelectionIndex = () => { | ||||||
|   let quill = getQuill() |   let quill = getQuill() | ||||||
| 
 | 
 | ||||||
|   return (quill.getSelection() || {}).index || quill.getLength() |   return (quill.getSelection() || {}).index || quill.getLength() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 计算当前对话索引名称(标识当前聊天) |  | ||||||
| const indexName = computed(() => dialogueStore.index_name) | const indexName = computed(() => dialogueStore.index_name) | ||||||
| // 控制是否显示编辑器的投票界面 |  | ||||||
| const isShowEditorVote = ref(false) | const isShowEditorVote = ref(false) | ||||||
| // 控制是否显示编辑器的代码界面 |  | ||||||
| const isShowEditorCode = ref(false) | const isShowEditorCode = ref(false) | ||||||
| // 控制是否显示录音界面 |  | ||||||
| const isShowEditorRecorder = ref(false) | const isShowEditorRecorder = ref(false) | ||||||
| // 图片文件上传DOM引用 |  | ||||||
| const fileImageRef = ref() | const fileImageRef = ref() | ||||||
| // 文件上传DOM引用 |  | ||||||
| const uploadFileRef = ref() | const uploadFileRef = ref() | ||||||
| // 表情面板引用 |  | ||||||
| const emoticonRef = ref() | const emoticonRef = ref() | ||||||
| 
 | 
 | ||||||
| // 编辑器配置选项 |  | ||||||
| const editorOption = { | const editorOption = { | ||||||
|   debug: false, |   debug: false, | ||||||
|   modules: { |   modules: { | ||||||
|     toolbar: false,  // 禁用默认工具栏 |     toolbar: false, | ||||||
|     clipboard: { |     clipboard: { | ||||||
|       // 粘贴处理,去除粘贴时的自带样式 |       // 粘贴版,处理粘贴时候的自带样式 | ||||||
|       matchers: [[Node.ELEMENT_NODE, onClipboardMatcher]] |       matchers: [[Node.ELEMENT_NODE, onClipboardMatcher]] | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
| @ -119,22 +83,19 @@ const editorOption = { | |||||||
|       bindings: { |       bindings: { | ||||||
|         enter: { |         enter: { | ||||||
|           key: 13, |           key: 13, | ||||||
|           handler: onSendMessage  // 按Enter键发送消息 |           handler: onSendMessage | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 图片上传配置 |  | ||||||
|     imageUploader: { |     imageUploader: { | ||||||
|       upload: onEditorUpload |       upload: onEditorUpload | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // @功能配置 |  | ||||||
|     mention: { |     mention: { | ||||||
|       allowedChars: /^[\u4e00-\u9fa5]*$/,  // 允许中文字符 |       allowedChars: /^[\u4e00-\u9fa5]*$/, | ||||||
|       mentionDenotationChars: ['@'],       // @符号触发 |       mentionDenotationChars: ['@'], | ||||||
|       positioningStrategy: 'fixed',        // 定位策略 |       positioningStrategy: 'fixed', | ||||||
|       // 渲染@项目的函数 |  | ||||||
|       renderItem: (data: any) => { |       renderItem: (data: any) => { | ||||||
|         const el = document.createElement('div') |         const el = document.createElement('div') | ||||||
|         el.className = 'ed-member-item' |         el.className = 'ed-member-item' | ||||||
| @ -142,7 +103,6 @@ const editorOption = { | |||||||
|         el.innerHTML += `<span class="nickname">${data.nickname}</span>` |         el.innerHTML += `<span class="nickname">${data.nickname}</span>` | ||||||
|         return el |         return el | ||||||
|       }, |       }, | ||||||
|       // 数据源函数,过滤匹配的用户 |  | ||||||
|       source: function (searchTerm: string, renderList: any) { |       source: function (searchTerm: string, renderList: any) { | ||||||
|         if (!props.members.length) { |         if (!props.members.length) { | ||||||
|           return renderList([]) |           return renderList([]) | ||||||
| @ -163,73 +123,66 @@ const editorOption = { | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   placeholder: '按Enter发送 / Shift+Enter 换行', |   placeholder: '按Enter发送 / Shift+Enter 换行', | ||||||
|   theme: 'snow'  // 使用snow主题 |   theme: 'snow' | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 底部工具栏配置 |  | ||||||
| const navs = reactive([ | const navs = reactive([ | ||||||
|   { |   { | ||||||
|     title: '图片', |     title: '图片', | ||||||
|     icon: markRaw(Pic), |     icon: markRaw(Pic), | ||||||
|     show: true, |     show: true, | ||||||
|     click: () => { |     click: () => { | ||||||
|       fileImageRef.value.click()  // 触发图片上传 |       fileImageRef.value.click() | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     title: '文件', |     title: '附件', | ||||||
|     icon: markRaw(FolderUpload), |     icon: markRaw(FolderUpload), | ||||||
|     show: true, |     show: true, | ||||||
|     click: () => { |     click: () => { | ||||||
|       uploadFileRef.value.click()  // 触发文件上传 |       uploadFileRef.value.click() | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   // 以下功能已被注释掉,但保留代码 |   { | ||||||
|   // { |     title: '代码', | ||||||
|   //   title: '代码', |     icon: markRaw(SourceCode), | ||||||
|   //   icon: markRaw(SourceCode), |     show: true, | ||||||
|   //   show: true, |     click: () => { | ||||||
|   //   click: () => { |       isShowEditorCode.value = true | ||||||
|   //     isShowEditorCode.value = true |     } | ||||||
|   //   } |   }, | ||||||
|   // }, |   { | ||||||
|   // { |     title: '语音消息', | ||||||
|   //   title: '语音消息', |     icon: markRaw(IconVoice), | ||||||
|   //   icon: markRaw(IconVoice), |     show: true, | ||||||
|   //   show: true, |     click: () => { | ||||||
|   //   click: () => { |       isShowEditorRecorder.value = true | ||||||
|   //     isShowEditorRecorder.value = true |     } | ||||||
|   //   } |   }, | ||||||
|   // }, |   { | ||||||
|   // { |     title: '地理位置', | ||||||
|   //   title: '地理位置', |     icon: markRaw(Local), | ||||||
|   //   icon: markRaw(Local), |     show: true, | ||||||
|   //   show: true, |     click: () => {} | ||||||
|   //   click: () => {} |   }, | ||||||
|   // }, |   { | ||||||
|   // { |     title: '群投票', | ||||||
|   //   title: '群投票', |     icon: markRaw(Ranking), | ||||||
|   //   icon: markRaw(Ranking), |     show: computed(() => props.vote), | ||||||
|   //   show: computed(() => props.vote), |     click: () => { | ||||||
|   //   click: () => { |       isShowEditorVote.value = true | ||||||
|   //     isShowEditorVote.value = true |     } | ||||||
|   //   } |   }, | ||||||
|   // }, |   { | ||||||
|   // { |     title: '历史记录', | ||||||
|   //   title: '历史记录', |     icon: markRaw(History), | ||||||
|   //   icon: markRaw(History), |     show: true, | ||||||
|   //   show: true, |     click: () => { | ||||||
|   //   click: () => { |       emit('editor-event', emitCall('history_event')) | ||||||
|   //     emit('editor-event', emitCall('history_event')) |     } | ||||||
|   //   } |   } | ||||||
|   // } |  | ||||||
| ]) | ]) | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 上传图片函数 |  | ||||||
|  * @param file 文件对象 |  | ||||||
|  * @returns Promise,成功时返回图片URL |  | ||||||
|  */ |  | ||||||
| function onUploadImage(file: File) { | function onUploadImage(file: File) { | ||||||
|   return new Promise((resolve) => { |   return new Promise((resolve) => { | ||||||
|     let image = new Image() |     let image = new Image() | ||||||
| @ -237,44 +190,35 @@ function onUploadImage(file: File) { | |||||||
|     image.onload = () => { |     image.onload = () => { | ||||||
|       const form = new FormData() |       const form = new FormData() | ||||||
|       form.append('file', file) |       form.append('file', file) | ||||||
|       form.append("source", "fonchain-chat");  // 图片来源标识 |       form.append("source", "fonchain-chat"); | ||||||
|       // 添加图片尺寸信息作为URL参数 |       // form.append('width', image.width.toString()) | ||||||
|  |       // form.append('height', image.height.toString()) | ||||||
|       form.append("urlParam", `width=${image.width}&height=${image.height}`); |       form.append("urlParam", `width=${image.width}&height=${image.height}`); | ||||||
| 
 | 
 | ||||||
|       // 调用上传API |  | ||||||
|       uploadImg(form).then(({ code, data, message }) => { |       uploadImg(form).then(({ code, data, message }) => { | ||||||
|         if (code == 0) { |         if (code == 0) { | ||||||
|           resolve(data.ori_url)  // 返回原始图片URL |           resolve(data.ori_url) | ||||||
|         } else { |         } else { | ||||||
|           resolve('') |           resolve('') | ||||||
|           window['$message'].error(message)  // 显示错误信息 |           window['$message'].error(message) | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 编辑器上传处理函数 |  | ||||||
|  * @param file 要上传的文件 |  | ||||||
|  * @returns Promise |  | ||||||
|  */ |  | ||||||
| function onEditorUpload(file: File) { | function onEditorUpload(file: File) { | ||||||
|   async function fn(file: File, resolve: Function, reject: Function) { |   async function fn(file: File, resolve: Function, reject: Function) { | ||||||
|     if (file.type.indexOf('image/') === 0) { |     if (file.type.indexOf('image/') === 0) { | ||||||
|       // 如果是图片,使用图片上传处理 |  | ||||||
|       return resolve(await onUploadImage(file)) |       return resolve(await onUploadImage(file)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     reject() |     reject() | ||||||
| 
 | 
 | ||||||
|     // 非图片文件的处理 |  | ||||||
|     if (file.type.indexOf('video/') === 0) { |     if (file.type.indexOf('video/') === 0) { | ||||||
|       // 视频文件 |  | ||||||
|       let fn = emitCall('video_event', file, () => {}) |       let fn = emitCall('video_event', file, () => {}) | ||||||
|       emit('editor-event', fn) |       emit('editor-event', fn) | ||||||
|     } else { |     } else { | ||||||
|       // 其他文件 |  | ||||||
|       let fn = emitCall('file_event', file, () => {}) |       let fn = emitCall('file_event', file, () => {}) | ||||||
|       emit('editor-event', fn) |       emit('editor-event', fn) | ||||||
|     } |     } | ||||||
| @ -285,40 +229,29 @@ function onEditorUpload(file: File) { | |||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 投票事件处理 |  | ||||||
|  * @param data 投票数据 |  | ||||||
|  */ |  | ||||||
| function onVoteEvent(data: any) { | function onVoteEvent(data: any) { | ||||||
|   const msg = emitCall('vote_event', data, (ok: boolean) => { |   const msg = emitCall('vote_event', data, (ok: boolean) => { | ||||||
|     if (ok) { |     if (ok) { | ||||||
|       isShowEditorVote.value = false  // 成功后关闭投票界面 |       isShowEditorVote.value = false | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   emit('editor-event', msg) |   emit('editor-event', msg) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 表情事件处理 |  | ||||||
|  * @param data 表情数据 |  | ||||||
|  */ |  | ||||||
| function onEmoticonEvent(data: any) { | function onEmoticonEvent(data: any) { | ||||||
|   emoticonRef.value.setShow(false)  // 关闭表情面板 |   emoticonRef.value.setShow(false) | ||||||
| 
 | 
 | ||||||
|   if (data.type == 1) { |   if (data.type == 1) { | ||||||
|     // 插入文本表情 |  | ||||||
|     const quill = getQuill() |     const quill = getQuill() | ||||||
|     let index = getQuillSelectionIndex() |     let index = getQuillSelectionIndex() | ||||||
| 
 | 
 | ||||||
|     // 删除编辑器中多余的换行符 |  | ||||||
|     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { |     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { | ||||||
|       quill.deleteText(0, 1) |       quill.deleteText(0, 1) | ||||||
|       index = 0 |       index = 0 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (data.img) { |     if (data.img) { | ||||||
|       // 插入图片表情 |  | ||||||
|       quill.insertEmbed(index, 'emoji', { |       quill.insertEmbed(index, 'emoji', { | ||||||
|         alt: data.value, |         alt: data.value, | ||||||
|         src: data.img, |         src: data.img, | ||||||
| @ -326,54 +259,40 @@ function onEmoticonEvent(data: any) { | |||||||
|         height: '24px' |         height: '24px' | ||||||
|       }) |       }) | ||||||
|     } else { |     } else { | ||||||
|       // 插入文本表情 |  | ||||||
|       quill.insertText(index, data.value) |       quill.insertText(index, data.value) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 设置光标位置 |  | ||||||
|     quill.setSelection(index + 1, 0, 'user') |     quill.setSelection(index + 1, 0, 'user') | ||||||
|   } else { |   } else { | ||||||
|     // 发送整个表情包 |  | ||||||
|     let fn = emitCall('emoticon_event', data.value, () => {}) |     let fn = emitCall('emoticon_event', data.value, () => {}) | ||||||
|     emit('editor-event', fn) |     emit('editor-event', fn) | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 代码事件处理 |  | ||||||
|  * @param data 代码数据 |  | ||||||
|  */ |  | ||||||
| function onCodeEvent(data: any) { | function onCodeEvent(data: any) { | ||||||
|   const msg = emitCall('code_event', data, (ok: boolean) => { |   const msg = emitCall('code_event', data, (ok: boolean) => { | ||||||
|     isShowEditorCode.value = false  // 成功后关闭代码界面 |     isShowEditorCode.value = false | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   emit('editor-event', msg) |   emit('editor-event', msg) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 文件上传处理 |  | ||||||
|  * @param e 上传事件对象 |  | ||||||
|  */ |  | ||||||
| async function onUploadFile(e: any) { | async function onUploadFile(e: any) { | ||||||
|   let file = e.target.files[0] |   let file = e.target.files[0] | ||||||
| 
 | 
 | ||||||
|   e.target.value = null  // 清空input,允许再次选择相同文件 |   e.target.value = null | ||||||
| 
 | 
 | ||||||
|   console.log("文件类型"+file.type) |     console.log("文件类型"+file.type) | ||||||
|   if (file.type.indexOf('image/') === 0) { |   if (file.type.indexOf('image/') === 0) { | ||||||
|     console.log("进入图片") |     console.log("进入图片") | ||||||
|     // 处理图片文件 |  | ||||||
|     const quill = getQuill() |     const quill = getQuill() | ||||||
|     let index = getQuillSelectionIndex() |     let index = getQuillSelectionIndex() | ||||||
| 
 | 
 | ||||||
|     // 删除编辑器中多余的换行符 |  | ||||||
|     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { |     if (index == 1 && quill.getLength() == 1 && quill.getText(0, 1) == '\n') { | ||||||
|       quill.deleteText(0, 1) |       quill.deleteText(0, 1) | ||||||
|       index = 0 |       index = 0 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 上传图片并插入到编辑器中 |  | ||||||
|     let src = await onUploadImage(file) |     let src = await onUploadImage(file) | ||||||
|     if (src) { |     if (src) { | ||||||
|       quill.insertEmbed(index, 'image', src) |       quill.insertEmbed(index, 'image', src) | ||||||
| @ -385,41 +304,29 @@ async function onUploadFile(e: any) { | |||||||
| 
 | 
 | ||||||
|   if (file.type.indexOf('video/') === 0) { |   if (file.type.indexOf('video/') === 0) { | ||||||
|     console.log("进入视频") |     console.log("进入视频") | ||||||
|     // 处理视频文件 |  | ||||||
|     let fn = emitCall('video_event', file, () => {}) |     let fn = emitCall('video_event', file, () => {}) | ||||||
|     emit('editor-event', fn) |     emit('editor-event', fn) | ||||||
|   } else { |   } else { | ||||||
|     console.log("进入其他") |     console.log("进入其他") | ||||||
|     // 处理其他类型文件 |  | ||||||
|     let fn = emitCall('file_event', file, () => {}) |     let fn = emitCall('file_event', file, () => {}) | ||||||
|     emit('editor-event', fn) |     emit('editor-event', fn) | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 录音事件处理 |  | ||||||
|  * @param file 录音文件 |  | ||||||
|  */ |  | ||||||
| function onRecorderEvent(file: any) { | function onRecorderEvent(file: any) { | ||||||
|   emit('editor-event', emitCall('file_event', file)) |   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) { | function onClipboardMatcher(node: any, Delta) { | ||||||
|   const ops: any[] = [] |   const ops: any[] = [] | ||||||
| 
 | 
 | ||||||
|   Delta.ops.forEach((op) => { |   Delta.ops.forEach((op) => { | ||||||
|     // 处理粘贴内容 |     // 如果粘贴了图片,这里会是一个对象,所以可以这样处理 | ||||||
|     if (op.insert && typeof op.insert === 'string') { |     if (op.insert && typeof op.insert === 'string') { | ||||||
|       ops.push({ |       ops.push({ | ||||||
|         insert: op.insert, // 文字内容 |         insert: op.insert, // 文字内容 | ||||||
|         attributes: {} // 移除所有样式 |         attributes: {} //文字样式(包括背景色和文字颜色等) | ||||||
|       }) |       }) | ||||||
|     } else { |     } else { | ||||||
|       ops.push(op) |       ops.push(op) | ||||||
| @ -430,16 +337,12 @@ function onClipboardMatcher(node: any, Delta) { | |||||||
|   return Delta |   return Delta | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 发送消息处理 |  | ||||||
|  * 根据编辑器内容类型发送不同类型的消息 |  | ||||||
|  */ |  | ||||||
| function onSendMessage() { | function onSendMessage() { | ||||||
|   var delta = getQuill().getContents() |   var delta = getQuill().getContents() | ||||||
|   let data = deltaToMessage(delta)  // 转换Delta为消息格式 |   let data = deltaToMessage(delta) | ||||||
| 
 | 
 | ||||||
|   if (data.items.length === 0) { |   if (data.items.length === 0) { | ||||||
|     return  // 没有内容不发送 |     return | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   switch (data.msgType) { |   switch (data.msgType) { | ||||||
| @ -448,72 +351,60 @@ function onSendMessage() { | |||||||
|         return window['$message'].info('发送内容超长,请分条发送') |         return window['$message'].info('发送内容超长,请分条发送') | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       // 发送文本消息 |  | ||||||
|       emit( |       emit( | ||||||
|         'editor-event', |         'editor-event', | ||||||
|         emitCall('text_event', data, (ok: any) => { |         emitCall('text_event', data, (ok: any) => { | ||||||
|           ok && getQuill().setContents([], Quill.sources.USER)  // 成功发送后清空编辑器 |           ok && getQuill().setContents([], Quill.sources.USER) | ||||||
|         }) |         }) | ||||||
|       ) |       ) | ||||||
|       break |       break | ||||||
|     case 3: // 图片消息 |     case 3: // 图片消息 | ||||||
|       // 发送图片消息 |  | ||||||
|       emit( |       emit( | ||||||
|         'editor-event', |         'editor-event', | ||||||
|         emitCall( |         emitCall( | ||||||
|           'image_event', |           'image_event', | ||||||
|           { ...getImageInfo(data.items[0].content), url: data.items[0].content, size: 10000 }, |           { ...getImageInfo(data.items[0].content), url: data.items[0].content, size: 10000 }, | ||||||
|           (ok: any) => { |           (ok: any) => { | ||||||
|             ok && getQuill().setContents([])  // 成功发送后清空编辑器 |             ok && getQuill().setContents([]) | ||||||
|           } |           } | ||||||
|         ) |         ) | ||||||
|       ) |       ) | ||||||
|       break |       break | ||||||
|     case 12: // 图文混合消息 |     case 12: // 图文消息 | ||||||
|       // 发送混合消息 |  | ||||||
|       emit( |       emit( | ||||||
|         'editor-event', |         'editor-event', | ||||||
|         emitCall('mixed_event', data, (ok: any) => { |         emitCall('mixed_event', data, (ok: any) => { | ||||||
|           ok && getQuill().setContents([])  // 成功发送后清空编辑器 |           ok && getQuill().setContents([]) | ||||||
|         }) |         }) | ||||||
|       ) |       ) | ||||||
|       break |       break | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 编辑器内容改变时的处理 |  | ||||||
|  * 保存草稿并触发输入事件 |  | ||||||
|  */ |  | ||||||
| function onEditorChange() { | function onEditorChange() { | ||||||
|   let delta = getQuill().getContents() |   let delta = getQuill().getContents() | ||||||
|   let text = deltaToString(delta)  // 将Delta转为纯文本 | 
 | ||||||
|  |   let text = deltaToString(delta) | ||||||
| 
 | 
 | ||||||
|   if (!isEmptyDelta(delta)) { |   if (!isEmptyDelta(delta)) { | ||||||
|     // 保存草稿到store |  | ||||||
|     editorDraftStore.items[indexName.value || ''] = JSON.stringify({ |     editorDraftStore.items[indexName.value || ''] = JSON.stringify({ | ||||||
|       text: text, |       text: text, | ||||||
|       ops: delta.ops |       ops: delta.ops | ||||||
|     }) |     }) | ||||||
|   } else { |   } else { | ||||||
|     // 编辑器为空时删除对应草稿 |     // 删除 editorDraftStore.items 下的元素 | ||||||
|     delete editorDraftStore.items[indexName.value || ''] |     delete editorDraftStore.items[indexName.value || ''] | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // 触发输入事件 |  | ||||||
|   emit('editor-event', emitCall('input_event', text)) |   emit('editor-event', emitCall('input_event', text)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 加载编辑器草稿内容 |  | ||||||
|  * 当切换聊天对象时,加载对应的草稿 |  | ||||||
|  */ |  | ||||||
| function loadEditorDraftText() { | function loadEditorDraftText() { | ||||||
|   if (!editor.value) return |   if (!editor.value) return | ||||||
| 
 | 
 | ||||||
|   // 延迟处理,确保DOM已渲染 |   // 这里延迟处理,不然会有问题 | ||||||
|   setTimeout(() => { |   setTimeout(() => { | ||||||
|     hideMentionDom()  // 隐藏@菜单 |     hideMentionDom() | ||||||
| 
 | 
 | ||||||
|     const quill = getQuill() |     const quill = getQuill() | ||||||
| 
 | 
 | ||||||
| @ -524,47 +415,33 @@ function loadEditorDraftText() { | |||||||
|     if (draft) { |     if (draft) { | ||||||
|       quill.setContents(JSON.parse(draft)?.ops || []) |       quill.setContents(JSON.parse(draft)?.ops || []) | ||||||
|     } else { |     } else { | ||||||
|       quill.setContents([])  // 没有草稿则清空编辑器 |       quill.setContents([]) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 设置光标位置到末尾 |  | ||||||
|     const index = getQuillSelectionIndex() |     const index = getQuillSelectionIndex() | ||||||
|     quill.setSelection(index, 0, 'user') |     quill.setSelection(index, 0, 'user') | ||||||
|   }, 0) |   }, 0) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 处理@成员事件 |  | ||||||
|  * @param data @成员数据 |  | ||||||
|  */ |  | ||||||
| function onSubscribeMention(data: any) { | function onSubscribeMention(data: any) { | ||||||
|   const mention = getQuill().getModule('mention') |   const mention = getQuill().getModule('mention') | ||||||
|   // 插入@项 | 
 | ||||||
|   mention.insertItem({ id: data?.id, denotationChar: '@', value: data.value }, true) |   mention.insertItem({ id: data?.id, denotationChar: '@', value: data.value }, true) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 处理引用事件 |  | ||||||
|  * @param data 引用数据 |  | ||||||
|  */ |  | ||||||
| function onSubscribeQuote(data: any) { | function onSubscribeQuote(data: any) { | ||||||
|   // 检查是否已有引用内容 |  | ||||||
|   const delta = getQuill().getContents() |   const delta = getQuill().getContents() | ||||||
|   if (delta.ops?.some((item: any) => item.insert.quote)) { |   if (delta.ops?.some((item: any) => item.insert.quote)) { | ||||||
|     return  // 已有引用则不再添加 |     return | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const quill = getQuill() |   const quill = getQuill() | ||||||
|   const index = getQuillSelectionIndex() |   const index = getQuillSelectionIndex() | ||||||
| 
 | 
 | ||||||
|   // 在编辑器开头插入引用 |  | ||||||
|   quill.insertEmbed(0, 'quote', data) |   quill.insertEmbed(0, 'quote', data) | ||||||
|   quill.setSelection(index + 1, 0, 'user')  // 设置光标到引用后 |   quill.setSelection(index + 1, 0, 'user') | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** |  | ||||||
|  * 隐藏@成员DOM元素 |  | ||||||
|  */ |  | ||||||
| function hideMentionDom() { | function hideMentionDom() { | ||||||
|   let el = document.querySelector('.ql-mention-list-container') |   let el = document.querySelector('.ql-mention-list-container') | ||||||
|   if (el) { |   if (el) { | ||||||
| @ -572,55 +449,27 @@ 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 }) | watch(indexName, loadEditorDraftText, { immediate: true }) | ||||||
| 
 | 
 | ||||||
| // 组件挂载时初始化 |  | ||||||
| onMounted(() => { | onMounted(() => { | ||||||
|   loadEditorDraftText() |   loadEditorDraftText() | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| // 组件卸载时清理 |  | ||||||
| onUnmounted(() => { | onUnmounted(() => { | ||||||
|   hideMentionDom() |   hideMentionDom() | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| // 订阅编辑器相关事件总线事件 |  | ||||||
| useEventBus([ | useEventBus([ | ||||||
|   { name: EditorConst.Mention, event: onSubscribeMention },  // @成员事件 |   { name: EditorConst.Mention, event: onSubscribeMention }, | ||||||
|   { name: EditorConst.Quote, event: onSubscribeQuote },       // 引用事件 |   { name: EditorConst.Quote, event: onSubscribeQuote } | ||||||
|   { name: EditorConst.Edit, event: onSubscribeEdit }          // 编辑消息事件 |  | ||||||
| ]) | ]) | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <!-- 编辑器容器 --> |  | ||||||
|   <section class="el-container editor"> |   <section class="el-container editor"> | ||||||
|     <section class="el-container is-vertical"> |     <section class="el-container is-vertical"> | ||||||
|       <!-- 工具栏区域 --> |  | ||||||
|       <header class="el-header toolbar bdr-t"> |       <header class="el-header toolbar bdr-t"> | ||||||
|         <div class="tools"> |         <div class="tools"> | ||||||
|           <!-- 表情选择器弹出框 --> |  | ||||||
|           <n-popover |           <n-popover | ||||||
|             placement="top-start" |             placement="top-start" | ||||||
|             trigger="click" |             trigger="click" | ||||||
| @ -640,7 +489,6 @@ useEventBus([ | |||||||
|             <MeEditorEmoticon @on-select="onEmoticonEvent" /> |             <MeEditorEmoticon @on-select="onEmoticonEvent" /> | ||||||
|           </n-popover> |           </n-popover> | ||||||
| 
 | 
 | ||||||
|           <!-- 工具栏其他功能按钮 --> |  | ||||||
|           <div |           <div | ||||||
|             class="item pointer" |             class="item pointer" | ||||||
|             v-for="nav in navs" |             v-for="nav in navs" | ||||||
| @ -654,7 +502,6 @@ useEventBus([ | |||||||
|         </div> |         </div> | ||||||
|       </header> |       </header> | ||||||
| 
 | 
 | ||||||
|       <!-- 编辑器主体区域 --> |  | ||||||
|       <main class="el-main height100"> |       <main class="el-main height100"> | ||||||
|         <QuillEditor |         <QuillEditor | ||||||
|           ref="editor" |           ref="editor" | ||||||
| @ -667,13 +514,11 @@ useEventBus([ | |||||||
|     </section> |     </section> | ||||||
|   </section> |   </section> | ||||||
| 
 | 
 | ||||||
|   <!-- 隐藏的文件上传表单 --> |  | ||||||
|   <form enctype="multipart/form-data" style="display: none"> |   <form enctype="multipart/form-data" style="display: none"> | ||||||
|     <input type="file" ref="fileImageRef" accept="image/*" @change="onUploadFile" /> |     <input type="file" ref="fileImageRef" accept="image/*" @change="onUploadFile" /> | ||||||
|     <input type="file" ref="uploadFileRef" @change="onUploadFile" /> |     <input type="file" ref="uploadFileRef" @change="onUploadFile" /> | ||||||
|   </form> |   </form> | ||||||
| 
 | 
 | ||||||
|   <!-- 条件渲染的功能组件 --> |  | ||||||
|   <MeEditorVote v-if="isShowEditorVote" @close="isShowEditorVote = false" @submit="onVoteEvent" /> |   <MeEditorVote v-if="isShowEditorVote" @close="isShowEditorVote = false" @submit="onVoteEvent" /> | ||||||
| 
 | 
 | ||||||
|   <MeEditorCode |   <MeEditorCode | ||||||
| @ -691,7 +536,7 @@ useEventBus([ | |||||||
| 
 | 
 | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .editor { | .editor { | ||||||
|   --tip-bg-color: rgb(241 241 241 / 90%);  /* 提示背景颜色 */ |   --tip-bg-color: rgb(241 241 241 / 90%); | ||||||
| 
 | 
 | ||||||
|   height: 100%; |   height: 100%; | ||||||
| 
 | 
 | ||||||
| @ -714,7 +559,7 @@ useEventBus([ | |||||||
|         user-select: none; |         user-select: none; | ||||||
| 
 | 
 | ||||||
|         .tip-title { |         .tip-title { | ||||||
|           display: none;  /* 默认隐藏提示文字 */ |           display: none; | ||||||
|           position: absolute; |           position: absolute; | ||||||
|           top: 40px; |           top: 40px; | ||||||
|           left: 0px; |           left: 0px; | ||||||
| @ -732,7 +577,7 @@ useEventBus([ | |||||||
| 
 | 
 | ||||||
|         &:hover { |         &:hover { | ||||||
|           .tip-title { |           .tip-title { | ||||||
|             display: block;  /* 悬停时显示提示文字 */ |             display: block; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
| @ -740,7 +585,6 @@ useEventBus([ | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 暗色模式样式调整 */ |  | ||||||
| html[theme-mode='dark'] { | html[theme-mode='dark'] { | ||||||
|   .editor { |   .editor { | ||||||
|     --tip-bg-color: #48484d; |     --tip-bg-color: #48484d; | ||||||
| @ -749,16 +593,13 @@ html[theme-mode='dark'] { | |||||||
| </style> | </style> | ||||||
| 
 | 
 | ||||||
| <style lang="less"> | <style lang="less"> | ||||||
| /* 全局编辑器样式 */ |  | ||||||
| #editor { | #editor { | ||||||
|   overflow: hidden; |   overflow: hidden; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 编辑器主体区域样式 */ |  | ||||||
| .ql-editor { | .ql-editor { | ||||||
|   padding: 8px; |   padding: 8px; | ||||||
| 
 | 
 | ||||||
|   /* 滚动条样式 */ |  | ||||||
|   &::-webkit-scrollbar { |   &::-webkit-scrollbar { | ||||||
|     width: 3px; |     width: 3px; | ||||||
|     height: 3px; |     height: 3px; | ||||||
| @ -770,7 +611,6 @@ html[theme-mode='dark'] { | |||||||
|     background-color: transparent; |     background-color: transparent; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /* 悬停时显示滚动条 */ |  | ||||||
|   &:hover { |   &:hover { | ||||||
|     &::-webkit-scrollbar-thumb { |     &::-webkit-scrollbar-thumb { | ||||||
|       background-color: var(--im-scrollbar-thumb); |       background-color: var(--im-scrollbar-thumb); | ||||||
| @ -778,7 +618,6 @@ html[theme-mode='dark'] { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 编辑器占位符样式 */ |  | ||||||
| .ql-editor.ql-blank::before { | .ql-editor.ql-blank::before { | ||||||
|   font-family: |   font-family: | ||||||
|     PingFang SC, |     PingFang SC, | ||||||
| @ -787,7 +626,6 @@ html[theme-mode='dark'] { | |||||||
|   left: 8px; |   left: 8px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 编辑器中图片样式 */ |  | ||||||
| .ql-snow .ql-editor img { | .ql-snow .ql-editor img { | ||||||
|   max-width: 100px; |   max-width: 100px; | ||||||
|   border-radius: 3px; |   border-radius: 3px; | ||||||
| @ -795,7 +633,6 @@ html[theme-mode='dark'] { | |||||||
|   margin: 0px 2px; |   margin: 0px 2px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 图片上传中样式 */ |  | ||||||
| .image-uploading { | .image-uploading { | ||||||
|   display: flex; |   display: flex; | ||||||
|   width: 100px; |   width: 100px; | ||||||
| @ -809,18 +646,15 @@ html[theme-mode='dark'] { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 表情符号样式 */ |  | ||||||
| .ed-emoji { | .ed-emoji { | ||||||
|   background-color: unset !important; |   background-color: unset !important; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 编辑器占位符样式 */ |  | ||||||
| .ql-editor.ql-blank::before { | .ql-editor.ql-blank::before { | ||||||
|   font-style: unset; |   font-style: unset; | ||||||
|   color: #b8b3b3; |   color: #b8b3b3; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 引用卡片样式 */ |  | ||||||
| .quote-card-content { | .quote-card-content { | ||||||
|   display: flex; |   display: flex; | ||||||
|   background-color: #f6f6f6; |   background-color: #f6f6f6; | ||||||
| @ -857,7 +691,6 @@ html[theme-mode='dark'] { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* 暗色模式下的样式调整 */ |  | ||||||
| html[theme-mode='dark'] { | html[theme-mode='dark'] { | ||||||
|   .ql-editor.ql-blank::before { |   .ql-editor.ql-blank::before { | ||||||
|     color: #57575a; |     color: #57575a; | ||||||
|  | |||||||
| @ -48,10 +48,10 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | |||||||
|     <input type="file" ref="fileImageRef" accept="image/*" @change="onUpload" /> |     <input type="file" ref="fileImageRef" accept="image/*" @change="onUpload" /> | ||||||
|   </form> |   </form> | ||||||
| 
 | 
 | ||||||
|   <section class="el-container is-vertical section height100 p-10px"> |   <section class="el-container is-vertical section height100"> | ||||||
|     <!-- <header class="el-header em-header bdr-b"> |     <header class="el-header em-header bdr-b"> | ||||||
|       <span>{{ items[tabIndex].name }}</span> |       <span>{{ items[tabIndex].name }}</span> | ||||||
|     </header> --> |     </header> | ||||||
| 
 | 
 | ||||||
|     <main class="el-main em-main me-scrollbar me-scrollbar-thumb"> |     <main class="el-main em-main me-scrollbar me-scrollbar-thumb"> | ||||||
|       <div class="symbol-box" v-if="tabIndex == 0"> |       <div class="symbol-box" v-if="tabIndex == 0"> | ||||||
| @ -82,7 +82,7 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | |||||||
|       </div> |       </div> | ||||||
|     </main> |     </main> | ||||||
| 
 | 
 | ||||||
|     <!-- <footer class="el-footer em-footer tabs"> |     <footer class="el-footer em-footer tabs"> | ||||||
|       <div |       <div | ||||||
|         class="tab pointer" |         class="tab pointer" | ||||||
|         v-for="(item, index) in items" |         v-for="(item, index) in items" | ||||||
| @ -93,7 +93,7 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | |||||||
|         <p class="tip">{{ item.name }}</p> |         <p class="tip">{{ item.name }}</p> | ||||||
|         <img width="20" height="20" :src="item.icon" /> |         <img width="20" height="20" :src="item.icon" /> | ||||||
|       </div> |       </div> | ||||||
|     </footer> --> |     </footer> | ||||||
|   </section> |   </section> | ||||||
| </template> | </template> | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| @ -185,18 +185,17 @@ const onSendEmoticon = (type: any, value: any, img = '') => { | |||||||
|       display: flex; |       display: flex; | ||||||
|       flex-wrap: wrap; |       flex-wrap: wrap; | ||||||
| 
 | 
 | ||||||
|       .option{ |       .option { | ||||||
|         margin: 7px; |         height: 32px; | ||||||
|         :deep(.emoji){ |         width: 32px; | ||||||
|           height: 22px; |         margin: 2px; | ||||||
|         width: 22px; |         font-size: 24px; | ||||||
|         user-select: none; |         user-select: none; | ||||||
|         transition: all 0.5s; |         transition: all 0.5s; | ||||||
| 
 | 
 | ||||||
|         &:hover { |         &:hover { | ||||||
|           transform: scale(1.5); |           transform: scale(1.5); | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import { ref, reactive } from 'vue' | |||||||
| import { PlayOne, PauseOne } from '@icon-park/vue-next' | import { PlayOne, PauseOne } from '@icon-park/vue-next' | ||||||
| import { ITalkRecordExtraAudio, ITalkRecord } from '@/types/chat' | import { ITalkRecordExtraAudio, ITalkRecord } from '@/types/chat' | ||||||
| 
 | 
 | ||||||
| const props = defineProps<{ | defineProps<{ | ||||||
|   extra: ITalkRecordExtraAudio |   extra: ITalkRecordExtraAudio | ||||||
|   data: ITalkRecord |   data: ITalkRecord | ||||||
|   maxWidth?: Boolean |   maxWidth?: Boolean | ||||||
| @ -18,8 +18,7 @@ const state = reactive({ | |||||||
|   progress: 0, |   progress: 0, | ||||||
|   duration: 0, |   duration: 0, | ||||||
|   currentTime: 0, |   currentTime: 0, | ||||||
|   loading: true, |   loading: true | ||||||
|   showText: false |  | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| const onPlay = () => { | const onPlay = () => { | ||||||
| @ -41,12 +40,6 @@ const onCanplay = () => { | |||||||
|   state.duration = audioRef.value.duration |   state.duration = audioRef.value.duration | ||||||
|   durationDesc.value = formatTime(parseInt(audioRef.value.duration)) |   durationDesc.value = formatTime(parseInt(audioRef.value.duration)) | ||||||
|   state.loading = false |   state.loading = false | ||||||
|    |  | ||||||
|   if (props.data.is_convert_text === 1 && props.data.extra.content) { |  | ||||||
|     setTimeout(() => { |  | ||||||
|       state.showText = true |  | ||||||
|     }, 300) |  | ||||||
|   } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onError = (e: any) => { | const onError = (e: any) => { | ||||||
| @ -68,12 +61,17 @@ const formatTime = (value: number = 0) => { | |||||||
|     return '-' |     return '-' | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return `${Math.floor(value)}"` |   const minutes = Math.floor(value / 60) | ||||||
|  |   let seconds = value | ||||||
|  |   if (minutes > 0) { | ||||||
|  |     seconds = Math.floor(value - minutes * 60) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return `${minutes}'${seconds}"` | ||||||
| } | } | ||||||
| </script> | </script> | ||||||
| <template> | <template> | ||||||
|   <div class="pointer w-200px bg-#f5f5f5 rounded-10px px-11px"> |   <div class="im-message-audio"> | ||||||
|     <div class="im-message-audio h-44px"> |  | ||||||
|     <audio |     <audio | ||||||
|       ref="audioRef" |       ref="audioRef" | ||||||
|       preload="auto" |       preload="auto" | ||||||
| @ -100,27 +98,20 @@ const formatTime = (value: number = 0) => { | |||||||
|     </div> |     </div> | ||||||
|     <div class="time">{{ durationDesc }}</div> |     <div class="time">{{ durationDesc }}</div> | ||||||
|   </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> | </template> | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .im-message-audio { | .im-message-audio { | ||||||
|   --audio-bg-color: #f5f5f5; |   --audio-bg-color: #f5f5f5; | ||||||
|   --audio-btn-bg-color: #ffffff; |   --audio-btn-bg-color: #ffffff; | ||||||
|  | 
 | ||||||
|  |   width: 200px; | ||||||
|  |   height: 45px; | ||||||
|  |   border-radius: 10px; | ||||||
|   display: flex; |   display: flex; | ||||||
|   align-items: center; |   align-items: center; | ||||||
|   overflow: hidden; |   overflow: hidden; | ||||||
|  |   background-color: var(--audio-bg-color); | ||||||
|  | 
 | ||||||
|   > div { |   > div { | ||||||
|     display: flex; |     display: flex; | ||||||
|     align-items: center; |     align-items: center; | ||||||
| @ -141,7 +132,6 @@ const formatTime = (value: number = 0) => { | |||||||
|       display: flex; |       display: flex; | ||||||
|       align-items: center; |       align-items: center; | ||||||
|       justify-content: center; |       justify-content: center; | ||||||
|       transition: all 0.3s ease; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -240,7 +230,6 @@ const formatTime = (value: number = 0) => { | |||||||
|       height: 70%; |       height: 70%; | ||||||
|       width: 1px; |       width: 1px; | ||||||
|       background-color: #9b9595; |       background-color: #9b9595; | ||||||
|       transition: left 0.1s linear; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -252,40 +241,6 @@ 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'] { | html[theme-mode='dark'] { | ||||||
|   .im-message-audio { |   .im-message-audio { | ||||||
|     --audio-bg-color: #2c2c32; |     --audio-bg-color: #2c2c32; | ||||||
|  | |||||||
| @ -1,222 +1,118 @@ | |||||||
| <script setup> | <script lang="ts" setup> | ||||||
| import { fileFormatSize } from '@/utils/strings' | import { fileFormatSize } from '@/utils/strings' | ||||||
| import { ref, computed } from 'vue' | import { download, getFileNameSuffix } from '@/utils/functions' | ||||||
| import { useUploadsStore } from '@/store' | import { ITalkRecordExtraFile, ITalkRecord } from '@/types/chat' | ||||||
| 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<{ | ||||||
| const props = defineProps({ |   extra: ITalkRecordExtraFile | ||||||
|   // 文件的额外信息 |   data: ITalkRecord | ||||||
|   extra: { |   maxWidth?: Boolean | ||||||
|     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> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <div class="file-message"> |   <section class="file-message"> | ||||||
|     <!-- 文件头部信息 --> |     <div class="main"> | ||||||
|     <div class="file-header"> |       <div class="ext">{{ getFileNameSuffix(extra.name) }}</div> | ||||||
|       <!-- 文件名 --> |       <div class="file-box"> | ||||||
|       <div class="file-name">{{ extra.name }}</div> |         <p class="info"> | ||||||
|       <!-- 文件图标区域 --> |           <span class="name">{{ extra.name }}</span> | ||||||
|       <div  class="file-icon-container"> |           <span class="size">({{ fileFormatSize(extra.size) }})</span> | ||||||
|         <img class="file-icon" :src="fileInfo.icon" alt="文件图标"> |         </p> | ||||||
|              |         <p class="notice">文件已成功发送, 文件助手永久保存</p> | ||||||
|         <!-- 上传进度圆环 - 上传状态 --> |  | ||||||
|         <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> |     </div> | ||||||
|     <!-- 文件大小信息 --> |     <div class="footer"> | ||||||
|     <div class="file-size">{{ fileFormatSize(extra.size) }}</div> |       <a @click="download(data.msg_id)">下载</a> | ||||||
|   </div> |       <a>在线预览</a> | ||||||
|  |     </div> | ||||||
|  |   </section> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .file-message { | .file-message { | ||||||
|   width: 243px; |   width: 250px; | ||||||
|   background-color: #fff; |   min-height: 85px; | ||||||
|   border-radius: 8px; |   padding: 10px; | ||||||
|   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); |   border-radius: 10px; | ||||||
|   padding: 0 14px; |   border: 1px solid var(--im-message-border-color); | ||||||
|   cursor: pointer; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| .file-header { |   .main { | ||||||
|   display: flex; |     height: 45px; | ||||||
|   padding: 14px 5px 14px 0; |     display: flex; | ||||||
|   justify-content: space-between; |     flex-direction: row; | ||||||
|   width: 100%; |     margin-top: 5px; | ||||||
|   border-bottom: 1px solid #EEEEEE; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| .file-name { |     .ext { | ||||||
|   color: #1A1A1A; |       display: flex; | ||||||
|   font-size: 14px; |       justify-content: center; | ||||||
|   word-break: break-word; |       align-items: center; | ||||||
|   overflow: hidden; |       width: 45px; | ||||||
|   text-overflow: ellipsis; |       height: 45px; | ||||||
|   display: -webkit-box; |       color: #ffffff; | ||||||
|   -webkit-line-clamp: 2; |       background: #49a4ff; | ||||||
|   -webkit-box-orient: vertical; |       border-radius: 5px; | ||||||
| } |       font-size: 12px; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
| .file-icon-container { |     .file-box { | ||||||
|   position: relative; |       flex: 1 1; | ||||||
| } |       height: 45px; | ||||||
|  |       margin-left: 10px; | ||||||
|  |       overflow: hidden; | ||||||
| 
 | 
 | ||||||
| .file-icon { |       .info { | ||||||
|   width: 48px; |         display: flex; | ||||||
|   height: 48px; |         justify-content: space-between; | ||||||
| } |         align-items: center; | ||||||
|  |         overflow: hidden; | ||||||
|  |         height: 24px; | ||||||
|  |         font-size: 14px; | ||||||
| 
 | 
 | ||||||
| .progress-overlay { |         .name { | ||||||
|   background-color: #fff; |           flex: 1 auto; | ||||||
|   position: absolute; |           white-space: nowrap; | ||||||
|   top: 6px; |           overflow: hidden; | ||||||
|   left: 11px; |           text-overflow: ellipsis; | ||||||
|   width: 30px; |         } | ||||||
|   height: 30px; |  | ||||||
|   display: flex; |  | ||||||
|   justify-content: center; |  | ||||||
|   align-items: center; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| .file-size { |         .size { | ||||||
|   color: #747474; |           font-size: 12px; | ||||||
|   font-size: 12px; |           color: #cac6c6; | ||||||
|   padding: 5px 0 11px; |           flex-shrink: 0; | ||||||
| } |         } | ||||||
|  |       } | ||||||
| 
 | 
 | ||||||
| .circle-progress-container { |       .notice { | ||||||
|   width: 20px; |         height: 25px; | ||||||
|   height: 20px; |         line-height: 25px; | ||||||
|   position: relative; |         font-size: 12px; | ||||||
|   cursor: pointer; |         color: #929191; | ||||||
| } |         white-space: nowrap; | ||||||
|  |         overflow: hidden; | ||||||
|  |         text-overflow: ellipsis; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
| .circle-progress { |   .footer { | ||||||
|   transform: rotate(-90deg); |     height: 30px; | ||||||
|   transform-origin: center; |     line-height: 37px; | ||||||
| } |     text-align: right; | ||||||
|  |     font-size: 12px; | ||||||
|  |     border-top: 1px solid var(--border-color); | ||||||
|  |     margin-top: 10px; | ||||||
| 
 | 
 | ||||||
| .progress-circle { |     a { | ||||||
|   transition: stroke-dashoffset 0.3s ease; |       margin: 0 3px; | ||||||
| } |       user-select: none; | ||||||
|  |       cursor: pointer; | ||||||
|  |       color: var(--im-text-color); | ||||||
| 
 | 
 | ||||||
| .pause-icon, .play-icon { |       &:hover { | ||||||
|   transform-origin: center; |         color: royalblue; | ||||||
| } |       } | ||||||
| 
 |     } | ||||||
| .pause-icon { |   } | ||||||
|   transform: rotate(90deg); |  | ||||||
| } | } | ||||||
| </style> | </style> | ||||||
|  | |||||||
| @ -46,8 +46,7 @@ const img = (src: string, width = 200) => { | |||||||
|   background: var(--im-message-left-bg-color); |   background: var(--im-message-left-bg-color); | ||||||
|   min-width: 30px; |   min-width: 30px; | ||||||
|   min-height: 30px; |   min-height: 30px; | ||||||
|   max-width:240px; | 
 | ||||||
|   height:149px |  | ||||||
|   &.left { |   &.left { | ||||||
|     background: var(--im-message-right-bg-color); |     background: var(--im-message-right-bg-color); | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -1,9 +1,7 @@ | |||||||
| <script setup> | <script setup> | ||||||
| import { formatTime } from '@/utils/datetime' | import { formatTime } from '@/utils/datetime' | ||||||
| import { bus } from '@/utils/event-bus' |  | ||||||
| import { EditorConst } from '@/constant/event-bus' |  | ||||||
| 
 | 
 | ||||||
| const props = defineProps({ | defineProps({ | ||||||
|   login_uid: { |   login_uid: { | ||||||
|     type: Number, |     type: Number, | ||||||
|     default: 0 |     default: 0 | ||||||
| @ -23,30 +21,13 @@ const props = defineProps({ | |||||||
|   datetime: { |   datetime: { | ||||||
|     type: String, |     type: String, | ||||||
|     default: '' |     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> | </script> | ||||||
| <template> | <template> | ||||||
|   <div class="im-message-revoke"> |   <div class="im-message-revoke"> | ||||||
|     <div class="content"> |     <div class="content"> | ||||||
|       <div v-if="login_uid === user_id"> |       <span v-if="login_uid == user_id"> 你撤回了一条消息 | {{ formatTime(datetime) }} </span> | ||||||
|         <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-if="talk_type == 1"> 对方撤回了一条消息 | {{ formatTime(datetime) }} </span> | ||||||
|       <span v-else> |       <span v-else> | ||||||
|         "{{ nickname }}" 撤回了一条消息 | |         "{{ nickname }}" 撤回了一条消息 | | ||||||
|  | |||||||
| @ -45,7 +45,7 @@ textContent = textReplaceEmoji(textContent) | |||||||
|   color: var(--im-message-left-text-color); |   color: var(--im-message-left-text-color); | ||||||
|   background: var(--im-message-left-bg-color); |   background: var(--im-message-left-bg-color); | ||||||
|   border-radius: 0px 10px 10px 10px; |   border-radius: 0px 10px 10px 10px; | ||||||
|   font-size: 14px; | 
 | ||||||
|   &.right { |   &.right { | ||||||
|     background-color: var(--im-message-right-bg-color); |     background-color: var(--im-message-right-bg-color); | ||||||
|     color: var(--im-message-right-text-color); |     color: var(--im-message-right-text-color); | ||||||
|  | |||||||
| @ -1,17 +1,11 @@ | |||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import 'xgplayer/dist/index.min.css' | import 'xgplayer/dist/index.min.css' | ||||||
| import { ref, nextTick, watch } from 'vue' | import { ref, nextTick } from 'vue' | ||||||
| import { NImage, NModal, NCard, NProgress, NPopconfirm } from 'naive-ui' | import { NImage, NModal, NCard } from 'naive-ui' | ||||||
| import { Play, Close, Pause, Right, Attention } from '@icon-park/vue-next' | import { Play, Close } from '@icon-park/vue-next' | ||||||
| import { getImageInfo } from '@/utils/functions' | import { getImageInfo } from '@/utils/functions' | ||||||
| import {PauseOutline} from '@vicons/ionicons5' |  | ||||||
| import Player from 'xgplayer' | import Player from 'xgplayer' | ||||||
| import { ITalkRecordExtraVideo, ITalkRecord } from '@/types/chat' | import { ITalkRecordExtraVideo, ITalkRecord } from '@/types/chat' | ||||||
| import { useUploadsStore } from '@/store' |  | ||||||
| // @ts-ignore |  | ||||||
| const message = window.$message |  | ||||||
| 
 |  | ||||||
| const uploadsStore = useUploadsStore() |  | ||||||
| 
 | 
 | ||||||
| const props = defineProps<{ | const props = defineProps<{ | ||||||
|   extra: ITalkRecordExtraVideo |   extra: ITalkRecordExtraVideo | ||||||
| @ -46,43 +40,8 @@ const img = (src: string, width = 200) => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const open = ref(false) | 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() { | async function onPlay() { | ||||||
|   // 如果视频正在上传,不执行播放操作 |  | ||||||
|   if (props.extra.is_uploading) { |  | ||||||
|     return |  | ||||||
|   } |  | ||||||
|    |  | ||||||
|   open.value = true |   open.value = true | ||||||
| 
 | 
 | ||||||
|   await nextTick() |   await nextTick() | ||||||
| @ -95,85 +54,18 @@ async function onPlay() { | |||||||
|     lang: 'zh-cn' |     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> | </script> | ||||||
| <template> | <template> | ||||||
|   <section |   <section | ||||||
|     class="im-message-video" |     class="im-message-video" | ||||||
|     :class="{ left: data.float === 'left' }" |     :class="{ left: data.float === 'left' }" | ||||||
|  |     :style="img(extra.cover, 350)" | ||||||
|     @click="onPlay" |     @click="onPlay" | ||||||
|   > |   > | ||||||
|  |     <n-image :src="extra.cover" preview-disabled /> | ||||||
| 
 | 
 | ||||||
|     <!-- <n-image :src="extra.cover" preview-disabled /> --> |     <div class="btn-video"> | ||||||
|     <video :src="props.extra.url" :controls="false"></video> |       <n-icon :component="Play" size="36" /> | ||||||
|      |  | ||||||
|     <!-- 上传进度显示 --> |  | ||||||
|     <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> |     </div> | ||||||
| 
 | 
 | ||||||
|     <n-modal v-model:show="open"> |     <n-modal v-model:show="open"> | ||||||
| @ -200,25 +92,23 @@ function retryUpload(e) { | |||||||
|   min-height: 30px; |   min-height: 30px; | ||||||
|   display: inline-flex; |   display: inline-flex; | ||||||
|   position: relative; |   position: relative; | ||||||
|   height:149px; | 
 | ||||||
|   width: 225px; |  | ||||||
|   &.left { |   &.left { | ||||||
|     background: var(--im-message-right-bg-color); |     background: var(--im-message-right-bg-color); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   video { |   :deep(.n-image img) { | ||||||
|     width: 100%; |     width: 100%; | ||||||
|     height: 100%; |     height: 100%; | ||||||
|     border-radius: 5px; |     border-radius: 5px; | ||||||
|     object-fit: cover; |  | ||||||
|     background-color: #333; /* 添加背景色,避免默认显示为灰色 */ |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .btn-video { |   .btn-video { | ||||||
|     left: 50%; |     width: 30px; | ||||||
|     top: 50%; |     height: 20px; | ||||||
|     transform: translate(-50%, -50%); |  | ||||||
|     position: absolute; |     position: absolute; | ||||||
|  |     left: calc(50% - 15px); | ||||||
|  |     top: calc(50% - 10px); | ||||||
|     cursor: pointer; |     cursor: pointer; | ||||||
|     color: #ffffff; |     color: #ffffff; | ||||||
|   } |   } | ||||||
| @ -244,54 +134,4 @@ function retryUpload(e) { | |||||||
|   align-items: center; |   align-items: center; | ||||||
|   justify-content: 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> | </style> | ||||||
|  | |||||||
| @ -1,10 +1,12 @@ | |||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { ref, computed, onMounted, watch } from 'vue' | import { ref, computed } from 'vue' | ||||||
|  | import { NModal, NInput, NScrollbar, NCheckbox, NTabs, NTab } from 'naive-ui' | ||||||
|  | import { Search, Delete } from '@icon-park/vue-next' | ||||||
| import { ServeGetContacts } from '@/api/contact' | import { ServeGetContacts } from '@/api/contact' | ||||||
| import { ServeGetGroups } from '@/api/group' | import { ServeGetGroups } from '@/api/group' | ||||||
| import XNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | 
 | ||||||
| const emit = defineEmits(['close', 'on-submit']) | const emit = defineEmits(['close', 'on-submit']) | ||||||
| import { CloseCircle } from '@vicons/ionicons5' | 
 | ||||||
| interface Item { | interface Item { | ||||||
|   id: number |   id: number | ||||||
|   type: number |   type: number | ||||||
| @ -15,18 +17,16 @@ interface Item { | |||||||
|   keyword: string |   keyword: string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const tabsIndex = ref<number>(1) | ||||||
| const isShowBox = ref(true) | const isShowBox = ref(true) | ||||||
| const loading = ref(true) | const loading = ref(true) | ||||||
| const items = ref<Item[]>([]) | const items = ref<Item[]>([]) | ||||||
| const keywords = ref('') | const keywords = ref('') | ||||||
| const loadGroupStatus = ref(false) | const loadGroupStatus = ref(false) | ||||||
|  defineProps<{ | 
 | ||||||
|   forwardMode: number |  | ||||||
| }>() |  | ||||||
| // 搜索过滤器:不再按类型过滤,将好友和群组融合在一起 |  | ||||||
| const searchFilter = computed(() => { | const searchFilter = computed(() => { | ||||||
|   return items.value.filter((item: Item) => { |   return items.value.filter((item: Item) => { | ||||||
|     return item.keyword.toLowerCase().includes(keywords.value.toLowerCase()) |     return tabsIndex.value == item.type && item.keyword.match(keywords.value) != null | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| @ -40,7 +40,6 @@ const isCanSubmit = computed(() => { | |||||||
| 
 | 
 | ||||||
| const onLoad = () => { | const onLoad = () => { | ||||||
|   onLoadContact() |   onLoadContact() | ||||||
|   onLoadGroup() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onLoadContact = () => { | const onLoadContact = () => { | ||||||
| @ -56,7 +55,7 @@ const onLoadContact = () => { | |||||||
|             avatar: item.avatar, |             avatar: item.avatar, | ||||||
|             type: 1, |             type: 1, | ||||||
|             name: item.remark || item.nickname, |             name: item.remark || item.nickname, | ||||||
|             keyword: (item.remark || '') + item.nickname, |             keyword: item.remark + item.nickname, | ||||||
|             remark: item.remark, |             remark: item.remark, | ||||||
|             checked: false |             checked: false | ||||||
|           } |           } | ||||||
| @ -76,7 +75,6 @@ const onLoadGroup = async () => { | |||||||
|   loading.value = true |   loading.value = true | ||||||
|   let { code, data } = await ServeGetGroups() |   let { code, data } = await ServeGetGroups() | ||||||
|   if (code != 200) { |   if (code != 200) { | ||||||
|     loading.value = false |  | ||||||
|     return |     return | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -103,13 +101,6 @@ const onMaskClick = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onTriggerContact = (item: any) => { | 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) |   let data = items.value.find((val: any) => val.id === item.id) | ||||||
| 
 | 
 | ||||||
|   if (data) { |   if (data) { | ||||||
| @ -117,18 +108,6 @@ 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 = () => { | const onSubmit = () => { | ||||||
|   let data = checkedFilter.value.map((item: any) => { |   let data = checkedFilter.value.map((item: any) => { | ||||||
|     return { |     return { | ||||||
| @ -140,110 +119,219 @@ const onSubmit = () => { | |||||||
|   emit('on-submit', data) |   emit('on-submit', data) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 1 单选 2 多选 | const onTabs = (value: number) => { | ||||||
| const selectType = ref(1) |   tabsIndex.value = value | ||||||
| const changeSelectType = () => { |   if (value == 2) { | ||||||
|   selectType.value = selectType.value == 1 ? 2 : 1 |     onLoadGroup() | ||||||
|    |   } | ||||||
|   // 切换选择模式时清空已选择的联系人 |  | ||||||
|   items.value.forEach(item => { |  | ||||||
|     item.checked = false |  | ||||||
|   }) |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| onLoad() | onLoad() | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <x-n-modal v-model:show="isShowBox" :title="forwardMode === 2 ? '合并转发' : '逐条转发'" style="width: 997px; height: 740px;background-color: #F9F9FD" |   <n-modal | ||||||
|     :on-after-leave="onMaskClick" content-style="display: flex; justify-content: center; align-items: center;"> |     v-model:show="isShowBox" | ||||||
|     <div class="w-927px h-627px bg-#fff rounded-3px px-35px py-20px"> |     preset="card" | ||||||
|       <div class="flex items-center justify-between mb-28px"> |     title="选择联系人" | ||||||
|         <div class="text-#333639">搜索</div> |     class="modal-radius" | ||||||
|         <div class="w-779px h-34px"> |     style="max-width: 650px; height: 550px" | ||||||
|           <n-input v-model:value="keywords" type="text" clearable placeholder="请输入"> |     :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> | ||||||
| 
 | 
 | ||||||
|           </n-input> |           <header class="el-header sub-header"> | ||||||
|         </div> |             <n-input placeholder="搜索" v-model:value="keywords" clearable size="small"> | ||||||
|       </div> |               <template #prefix> | ||||||
|       <div class="flex justify-between"> |                 <n-icon :component="Search" /> | ||||||
|         <div class="w-260px h-517px rounded-4px border-1px border-solid border-#E5E5E5 px-12px"> |               </template> | ||||||
|           <div class="border-b-2px border-b-solid border-b-#FBFBFB h-35px flex items-center justify-end"> |             </n-input> | ||||||
|             <n-button text color="#46299D" class="text-14px" @click="changeSelectType"> |           </header> | ||||||
|               {{ selectType === 1 ? '多选' : '单选' }} | 
 | ||||||
|             </n-button> |           <main class="el-main" v-loading="loading" loading-text="加载中..."> | ||||||
|           </div> |             <n-scrollbar> | ||||||
|           <div> |               <div class="friend-items"> | ||||||
|             <n-virtual-list v-if="!loading" style="max-height: 470px" :item-size="65" :items="searchFilter"> |                 <div | ||||||
|               <template #default="{ item }"> |                   class="friend-item pointer" | ||||||
|                 <div class="flex items-center border-b-2px border-b-solid h-65px border-b-#FBFBFB" |                   v-for="item in searchFilter" | ||||||
|                   @click="onTriggerContact(item)"> |                   :key="item.id" | ||||||
|                   <div class="mr-22px"> |                   @click="onTriggerContact(item)" | ||||||
|                     <n-radio v-if="selectType === 1" :checked="item.checked" /> |                 > | ||||||
|                     <n-checkbox v-else :checked="item.checked" /> |                   <div class="avatar"> | ||||||
|  |                     <im-avatar | ||||||
|  |                       class="pointer" | ||||||
|  |                       :src="item.avatar" | ||||||
|  |                       :size="25" | ||||||
|  |                       :username="item.remark || item.name" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="mr-10px"> | 
 | ||||||
|                     <n-image class="w-42px h-42px rounded-full" :src="item.avatar" /> |                   <div class="content"> | ||||||
|  |                     <span class="text-ellipsis">{{ item.remark || item.name }}</span> | ||||||
|                   </div> |                   </div> | ||||||
|                   <div class="flex items-center"> | 
 | ||||||
|                     <span class="text-ellipsis">{{ item.name }}</span> |                   <div class="checkbox"> | ||||||
|                     <span v-if="item.type == 2" class="badge group ml-2">群</span> |                     <n-checkbox size="small" :checked="item.checked" /> | ||||||
|                   </div> |                   </div> | ||||||
|                 </div> |                 </div> | ||||||
|               </template> |               </div> | ||||||
|             </n-virtual-list> |             </n-scrollbar> | ||||||
|             <div v-else class="flex-center h-470px"> |           </main> | ||||||
|               <span>加载中...</span> |         </section> | ||||||
|             </div> |       </aside> | ||||||
|           </div> | 
 | ||||||
|         </div> |       <main class="el-main"> | ||||||
|         <div class="w-578px h-517px rounded-4px border-1px border-solid border-#E5E5E5 px-12px"> |         <section class="el-container is-vertical height100"> | ||||||
|           <div class="border-b-2px border-b-solid border-b-#FBFBFB h-35px flex items-center text-14px text-#000"> |           <main class="el-main o-hidden"> | ||||||
|             发送给 |             <n-scrollbar class="friend-items"> | ||||||
|           </div> |               <div class="friend-items"> | ||||||
|           <div class="h-350px border-b-2px border-b-solid border-b-#FBFBFB"> |                 <div v-show="!checkedFilter.length" style="padding-top: 100px"> | ||||||
|             <div v-if="checkedFilter.length > 0"> |                   <n-empty size="200" description="暂无数据"> | ||||||
|               <n-virtual-list style="max-height: 350px" :item-size="65" :items="checkedFilter"> |                     <template #icon> | ||||||
|                 <template #default="{ item }"> |                       <img src="@/assets/image/no-data.svg" alt="" /> | ||||||
|                   <div class="flex items-center border-b-2px border-b-solid h-65px border-b-#FBFBFB pr-20px"> |                     </template> | ||||||
|                     <div class="mr-10px"> |                   </n-empty> | ||||||
|                       <n-image class="w-42px h-42px rounded-full" :src="item.avatar" /> |                 </div> | ||||||
|                     </div> | 
 | ||||||
|                     <div class="flex items-center"> |                 <div | ||||||
|                       <span class="text-ellipsis">{{ item.name }}</span> |                   class="friend-item pointer" | ||||||
|                       <span v-if="item.type == 2" class="badge group ml-2">群</span> |                   v-for="item in checkedFilter" | ||||||
|                     </div> |                   :key="item.id" | ||||||
|                     <n-button class="ml-auto" text color="#C7C7C9" @click="onRemoveContact(item)"> |                   @click="onTriggerContact(item)" | ||||||
|                       <n-icon :component="CloseCircle" size="18" /> |                 > | ||||||
|                     </n-button> |                   <div class="avatar"> | ||||||
|  |                     <im-avatar | ||||||
|  |                       class="pointer" | ||||||
|  |                       :src="item.avatar" | ||||||
|  |                       :size="25" | ||||||
|  |                       :username="item.remark || item.name" | ||||||
|  |                     /> | ||||||
|                   </div> |                   </div> | ||||||
|                 </template> | 
 | ||||||
|               </n-virtual-list> |                   <div class="content"> | ||||||
|             </div> |                     <span class="text-ellipsis"> | ||||||
|             <div v-else class="flex-center h-350px"> |                       {{ item.remark || item.name }} | ||||||
|               <n-empty size="medium" description="暂无选择联系人"> |                     </span> | ||||||
|                 <template #icon> |                     <span v-if="item.type == 2" class="badge group">群</span> | ||||||
|                   <img src="@/assets/image/no-data.svg" alt="" /> |                   </div> | ||||||
|                 </template> | 
 | ||||||
|               </n-empty> |                   <div class="checkbox"> | ||||||
|             </div> |                     <n-icon :size="16" :component="Delete" /> | ||||||
|           </div> |                   </div> | ||||||
|           <div class="flex flex-col items-center justify-center h-120px"> |                 </div> | ||||||
|             <div class="text-14px text-#999999 mb-23px"> |               </div> | ||||||
|               <span>[{{ forwardMode === 2 ? '合并转发' : '逐条转发' }}]</span>  |             </n-scrollbar> | ||||||
|               <span v-if="checkedFilter.length > 0"> |           </main> | ||||||
|                 {{ checkedFilter.map(item => item.name).join('、') }}的会话记录 |         </section> | ||||||
|               </span> |       </main> | ||||||
|               <span v-else>请选择联系人</span> |     </section> | ||||||
|             </div> | 
 | ||||||
|             <div class="flex justify-center items-center"> |     <template #footer> | ||||||
|               <n-button color="#C7C7C9" class="w-250px h-34px text-14px text-#fff mr-10px" @click="onCancel">取消</n-button> |       <div class="footer"> | ||||||
|               <n-button color="#46299D" class="w-250px h-34px text-14px text-#fff"  |         <div> | ||||||
|                 @click="onSubmit" :disabled="isCanSubmit">发送</n-button> |           <span>已选择({{ checkedFilter.length }})</span> | ||||||
|             </div> |         </div> | ||||||
|           </div> | 
 | ||||||
|  |         <div> | ||||||
|  |           <n-button type="tertiary" @click="isShowBox = false"> 取消 </n-button> | ||||||
|  |           <n-button type="primary" class="mt-l15" @click="onSubmit" :disabled="isCanSubmit"> | ||||||
|  |             确定 | ||||||
|  |           </n-button> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </template> | ||||||
|   </x-n-modal> |   </n-modal> | ||||||
| </template> | </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,16 +1,18 @@ | |||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { ref, computed, reactive } from 'vue' | 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 { CloseOne, Male, Female, SendOne } from '@icon-park/vue-next' | ||||||
| import { ServeSearchUser } from '@/api/contact' | import { ServeSearchUser } from '@/api/contact' | ||||||
| import { ServeCreateContact } from '@/api/contact' | import { ServeCreateContact } from '@/api/contact' | ||||||
| import { ServeContactGroupList, ServeContactMoveGroup, ServeEditContactRemark } from '@/api/contact' | import { ServeContactGroupList, ServeContactMoveGroup, ServeEditContactRemark } from '@/api/contact' | ||||||
| import { useTalkStore } from '@/store' | import { useTalkStore } from '@/store' | ||||||
| import { useRouter } from 'vue-router' | 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 router = useRouter() | ||||||
| const talkStore = useTalkStore() | const talkStore = useTalkStore() | ||||||
|  | 
 | ||||||
| const emit = defineEmits(['update:show', 'update:uid', 'updateRemark']) | const emit = defineEmits(['update:show', 'update:uid', 'updateRemark']) | ||||||
|  | 
 | ||||||
| const props = defineProps({ | const props = defineProps({ | ||||||
|   show: { |   show: { | ||||||
|     type: Boolean, |     type: Boolean, | ||||||
| @ -24,7 +26,7 @@ const props = defineProps({ | |||||||
| 
 | 
 | ||||||
| const loading = ref(true) | const loading = ref(true) | ||||||
| const isOpenFrom = ref(false) | const isOpenFrom = ref(false) | ||||||
| const userInfo: any = ref({ | const state: any = reactive({ | ||||||
|   id: 0, |   id: 0, | ||||||
|   avatar: '', |   avatar: '', | ||||||
|   gender: 0, |   gender: 0, | ||||||
| @ -41,26 +43,26 @@ const editCardPopover: any = ref(false) | |||||||
| const modelRemark = ref('') | const modelRemark = ref('') | ||||||
| 
 | 
 | ||||||
| const options = ref<any>([]) | const options = ref<any>([]) | ||||||
| // const groupName = computed(() => { | const groupName = computed(() => { | ||||||
| //   const item = options.value.find((item: any) => { |   const item = options.value.find((item: any) => { | ||||||
| //     return item.key == state.group_id |     return item.key == state.group_id | ||||||
| //   }) |   }) | ||||||
| 
 | 
 | ||||||
| //   if (item) { |   if (item) { | ||||||
| //     return item.label |     return item.label | ||||||
| //   } |   } | ||||||
| 
 | 
 | ||||||
| //   return '未设置分组' |   return '未设置分组' | ||||||
| // }) | }) | ||||||
| 
 | 
 | ||||||
| const onLoadData = () => { | const onLoadData = () => { | ||||||
|   ServeSearchUser({ |   ServeSearchUser({ | ||||||
|     erp_user_id: props.uid |     user_id: props.uid | ||||||
|   }).then(({ code, data }) => { |   }).then(({ code, data }) => { | ||||||
|     if (code == 200) { |     if (code == 200) { | ||||||
|       userInfo.value = data |       Object.assign(state, data) | ||||||
| 
 | 
 | ||||||
|       // modelRemark.value = state.remark |       modelRemark.value = state.remark | ||||||
| 
 | 
 | ||||||
|       loading.value = false |       loading.value = false | ||||||
|     } else { |     } else { | ||||||
| @ -68,15 +70,15 @@ const onLoadData = () => { | |||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   // ServeContactGroupList().then((res) => { |   ServeContactGroupList().then((res) => { | ||||||
|   //   if (res.code == 200) { |     if (res.code == 200) { | ||||||
|   //     let items = res.data.items || [] |       let items = res.data.items || [] | ||||||
|   //     options.value = [] |       options.value = [] | ||||||
|   //     for (const iter of items) { |       for (const iter of items) { | ||||||
|   //       options.value.push({ label: iter.name, key: iter.id }) |         options.value.push({ label: iter.name, key: iter.id }) | ||||||
|   //     } |       } | ||||||
|   //   } |     } | ||||||
|   // }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onToTalk = () => { | const onToTalk = () => { | ||||||
| @ -84,84 +86,84 @@ const onToTalk = () => { | |||||||
|   emit('update:show', false) |   emit('update:show', false) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // const onJoinContact = () => { | const onJoinContact = () => { | ||||||
| //   if (!state.text.length) { |   if (!state.text.length) { | ||||||
| //     return window['$message'].info('备注信息不能为空') |     return window['$message'].info('备注信息不能为空') | ||||||
| //   } |   } | ||||||
| 
 | 
 | ||||||
| //   ServeCreateContact({ |   ServeCreateContact({ | ||||||
| //     friend_id: props.uid, |     friend_id: props.uid, | ||||||
| //     remark: state.text |     remark: state.text | ||||||
| //   }).then((res) => { |   }).then((res) => { | ||||||
| //     if (res.code == 200) { |     if (res.code == 200) { | ||||||
| //       isOpenFrom.value = false |       isOpenFrom.value = false | ||||||
| //       window['$message'].success('申请发送成功') |       window['$message'].success('申请发送成功') | ||||||
| //     } else { |     } else { | ||||||
| //       window['$message'].error(res.message) |       window['$message'].error(res.message) | ||||||
| //     } |     } | ||||||
| //   }) |   }) | ||||||
| // } | } | ||||||
| 
 | 
 | ||||||
| // const onChangeRemark = () => { | const onChangeRemark = () => { | ||||||
| //   ServeEditContactRemark({ |   ServeEditContactRemark({ | ||||||
| //     friend_id: props.uid, |     friend_id: props.uid, | ||||||
| //     remark: modelRemark.value |     remark: modelRemark.value | ||||||
| //   }).then(({ code, message }) => { |   }).then(({ code, message }) => { | ||||||
| //     if (code == 200) { |     if (code == 200) { | ||||||
| //       editCardPopover.value.setShow(false) |       editCardPopover.value.setShow(false) | ||||||
| //       window['$message'].success('备注成功') |       window['$message'].success('备注成功') | ||||||
| //       state.remark = modelRemark.value |       state.remark = modelRemark.value | ||||||
| 
 | 
 | ||||||
| //       emit('updateRemark', { |       emit('updateRemark', { | ||||||
| //         user_id: props.uid, |         user_id: props.uid, | ||||||
| //         remark: modelRemark.value |         remark: modelRemark.value | ||||||
| //       }) |       }) | ||||||
| //     } else { |     } else { | ||||||
| //       window['$message'].error(message) |       window['$message'].error(message) | ||||||
| //     } |     } | ||||||
| //   }) |   }) | ||||||
| // } | } | ||||||
| 
 | 
 | ||||||
| // const handleSelectGroup = (value) => { | const handleSelectGroup = (value) => { | ||||||
| //   ServeContactMoveGroup({ |   ServeContactMoveGroup({ | ||||||
| //     user_id: props.uid, |     user_id: props.uid, | ||||||
| //     group_id: value |     group_id: value | ||||||
| //   }).then(({ code, message }) => { |   }).then(({ code, message }) => { | ||||||
| //     if (code == 200) { |     if (code == 200) { | ||||||
| //       state.group_id = value |       state.group_id = value | ||||||
| //       window['$message'].success('分组修改成功') |       window['$message'].success('分组修改成功') | ||||||
| //     } else { |     } else { | ||||||
| //       window['$message'].error(message) |       window['$message'].error(message) | ||||||
| //     } |     } | ||||||
| //   }) |   }) | ||||||
| // } | } | ||||||
| 
 | 
 | ||||||
| // const reset = () => { | const reset = () => { | ||||||
| //   loading.value = true |   loading.value = true | ||||||
| 
 | 
 | ||||||
| //   Object.assign(state, { |   Object.assign(state, { | ||||||
| //     id: 0, |     id: 0, | ||||||
| //     avatar: '', |     avatar: '', | ||||||
| //     gender: 0, |     gender: 0, | ||||||
| //     mobile: '', |     mobile: '', | ||||||
| //     motto: '', |     motto: '', | ||||||
| //     nickname: '', |     nickname: '', | ||||||
| //     remark: '', |     remark: '', | ||||||
| //     email: '', |     email: '', | ||||||
| //     status: 1, |     status: 1, | ||||||
| //     text: '' |     text: '' | ||||||
| //   }) |   }) | ||||||
| 
 | 
 | ||||||
| //   isOpenFrom.value = false |   isOpenFrom.value = false | ||||||
| // } | } | ||||||
| 
 | 
 | ||||||
| // const onUpdate = (value) => { | const onUpdate = (value) => { | ||||||
| //   if (!value) { |   if (!value) { | ||||||
| //     setTimeout(reset, 100) |     setTimeout(reset, 100) | ||||||
| //   } |   } | ||||||
| 
 | 
 | ||||||
| //   emit('update:show', value) |   emit('update:show', value) | ||||||
| // } | } | ||||||
| 
 | 
 | ||||||
| const onAfterEnter = () => { | const onAfterEnter = () => { | ||||||
|   onLoadData() |   onLoadData() | ||||||
| @ -169,90 +171,162 @@ const onAfterEnter = () => { | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <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"> |   <n-modal :show="show" :on-update:show="onUpdate" :on-after-enter="onAfterEnter"> | ||||||
|     <div class="section relative px-7px pt-82px pb-20px"> |     <div class="section" v-loading="loading"> | ||||||
|       <div class="absolute top-9px right-7px pointer z-10" @click="emit('update:show', false)"> |       <section class="el-container container is-vertical"> | ||||||
|         <img class="w-20px h-20px" src="@/assets/image/close.png" alt=""> |         <header class="el-header header"> | ||||||
|       </div> |           <im-avatar | ||||||
|  |             class="avatar" | ||||||
|  |             :size="100" | ||||||
|  |             :src="state.avatar" | ||||||
|  |             :username="state.remark || state.nickname" | ||||||
|  |             :font-size="30" | ||||||
|  |           /> | ||||||
| 
 | 
 | ||||||
|       <template v-if="loading"> |           <div class="gender" v-show="state.gender > 0"> | ||||||
|         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> |             <n-icon v-if="state.gender == 1" :component="Male" color="#508afe" /> | ||||||
|           <div class="w-59px h-59px rounded-8px mr-12px overflow-hidden"> |             <n-icon v-if="state.gender == 2" :component="Female" color="#ff5722" /> | ||||||
|             <n-skeleton circle height="59px" width="59px" /> |  | ||||||
|           </div> |           </div> | ||||||
|           <div class="w-full"> |  | ||||||
|             <n-skeleton text style="width: 80%; margin-bottom: 5px;" /> |  | ||||||
|             <n-skeleton text style="width: 60%;" /> |  | ||||||
|           </div> |  | ||||||
|         </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> |  | ||||||
|         </div> |  | ||||||
|         <div> |  | ||||||
|           <n-skeleton text style="width: 100%; height: 42px; border-radius: 4px;" /> |  | ||||||
|         </div> |  | ||||||
|       </template> |  | ||||||
| 
 | 
 | ||||||
|       <template v-else> |           <div class="close" @click="onUpdate(false)"> | ||||||
|         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> |             <close-one theme="outline" size="22" fill="#fff" :strokeWidth="2" /> | ||||||
|           <div class="w-59px h-59px  rounded-8px mr-12px overflow-hidden"> |           </div> | ||||||
|            <n-image width="59" :src="userInfo.avatar" > |  | ||||||
| 
 | 
 | ||||||
|            </n-image> |           <div class="nickname text-ellipsis"> | ||||||
|  |             {{ state.remark || state.nickname || '未设置昵称' }} | ||||||
|           </div> |           </div> | ||||||
|           <div> |         </header> | ||||||
|             <div class="text-#000 text-16px mb-5px">{{ userInfo.nickname }}</div> | 
 | ||||||
|             <div class="text-#ACACAC text-12px">工号:{{ userInfo.job_num }}</div> |         <main class="el-main main me-scrollbar me-scrollbar-thumb"> | ||||||
|  |           <div class="motto"> | ||||||
|  |             {{ state.motto || '编辑个签,展示我的独特态度。' }} | ||||||
|           </div> |           </div> | ||||||
|         </div> | 
 | ||||||
|         <div class="bg-#fff rounded-4px mb-20px"> |           <div class="infos"> | ||||||
|           <div class="flex px-15px py-9px"> |             <div class="info-item"> | ||||||
|             <div class="text-#000 text-12px w-84px">公司别</div> |               <span class="name">工号 :</span> | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.company_name }}</div> |               <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> | ||||||
|           <div class="flex px-15px py-9px"> |         </main> | ||||||
|             <div class="text-#000 text-12px w-84px">主管</div> | 
 | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.leaders?.map(x=>x.user_name)?.join(',') }}</div> |         <footer v-if="state.friend_status == 2" class="el-footer footer bdr-t flex-center"> | ||||||
|           </div> |           <n-button | ||||||
|           <div class="flex px-15px py-9px"> |             round | ||||||
|             <div class="text-#000 text-12px w-84px">部门</div> |             block | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.erp_dept_position?.map(x=>x.department_name)?.join(',') }}</div> |             type="primary" | ||||||
|           </div> |             text-color="#ffffff" | ||||||
|           <div class="flex px-15px py-9px"> |             @click="onToTalk" | ||||||
|             <div class="text-#000 text-12px w-84px">手机号</div> |             style="width: 91%" | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.tel_num }}</div> |           > | ||||||
|           </div> |             <template #icon> | ||||||
|           <div class="flex px-15px py-9px"> |               <n-icon :component="SendOne" /> | ||||||
|             <div class="text-#000 text-12px w-84px">岗位</div> |             </template> | ||||||
|             <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> |           </n-button> | ||||||
|         </div> |         </footer> | ||||||
|       </template> | 
 | ||||||
|  |         <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> |     </div> | ||||||
|   </x-n-modal> |   </n-modal> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .section { | .section { | ||||||
|   width: 100%; |  | ||||||
|   height: 100%; |  | ||||||
|   position: relative; |   position: relative; | ||||||
|   background-image: url('@/assets/image/zu6254@2x.png'); |   width: 360px; | ||||||
|  |   height: 600px; | ||||||
|   border-radius: 10px; |   border-radius: 10px; | ||||||
|   overflow: hidden; |   overflow: hidden; | ||||||
|   background-color: var(--im-bg-color); |   background-color: var(--im-bg-color); | ||||||
| @ -262,6 +336,7 @@ const onAfterEnter = () => { | |||||||
|     height: 230px; |     height: 230px; | ||||||
|     align-items: center; |     align-items: center; | ||||||
|     justify-content: center; |     justify-content: center; | ||||||
|  |     background: linear-gradient(to right, rgb(137, 104, 255), rgb(175, 152, 255)); | ||||||
|     display: flex; |     display: flex; | ||||||
|     padding: 20px; |     padding: 20px; | ||||||
|     position: relative; |     position: relative; | ||||||
| @ -271,6 +346,7 @@ const onAfterEnter = () => { | |||||||
|       width: 150px; |       width: 150px; | ||||||
|       height: 150px; |       height: 150px; | ||||||
|       content: ''; |       content: ''; | ||||||
|  |       background: linear-gradient(to right, rgb(142, 110, 255), rgb(208, 195, 255)); | ||||||
|       position: absolute; |       position: absolute; | ||||||
|       z-index: 1; |       z-index: 1; | ||||||
|       border-radius: 50%; |       border-radius: 50%; | ||||||
|  | |||||||
							
								
								
									
										144
									
								
								src/components/x-naive-ui/x-preview-img/index-1.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,144 @@ | |||||||
|  | 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,6 +4,5 @@ export const enum ContactConst { | |||||||
| 
 | 
 | ||||||
| export const enum EditorConst { | export const enum EditorConst { | ||||||
|   Mention = 'editor:mention', |   Mention = 'editor:mention', | ||||||
|   Quote = 'editor:quote', |   Quote = 'editor:quote' | ||||||
|   Edit = 'editor:edit' |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,28 +1,14 @@ | |||||||
| // 主题配置
 | // 主题配置
 | ||||||
| const primaryColor='#46299D' |  | ||||||
| export const overrides = { | export const overrides = { | ||||||
|   DataTable: { |  | ||||||
|     sorterIconColor:'#fff', |  | ||||||
|     thColorHover: primaryColor, |  | ||||||
|     thTextColor: "#fff", |  | ||||||
|     thColor: primaryColor, |  | ||||||
|     thBackgroundColor: primaryColor, |  | ||||||
|     itemColorActive:primaryColor, |  | ||||||
|   }, |  | ||||||
|   Button: { |  | ||||||
|     textColor: primaryColor, |  | ||||||
|   }, |  | ||||||
|   Dropdown:{ |  | ||||||
|     optionTextColorHover:'#46299D', |  | ||||||
|     optionColorHover:'#EEE9F8' |  | ||||||
|   }, |  | ||||||
|   common: { |   common: { | ||||||
|     primaryColorPressed: primaryColor, |     primaryColor: '#46299D', | ||||||
|     primaryHover:primaryColor, |     primaryColorHover: '#46299D', | ||||||
|     primaryDefault: primaryColor, |     primaryColorPressed: '#46299D', | ||||||
|     primaryActive: primaryColor, |     primaryColorSuppl: '#46299D', | ||||||
|     primarySuppl: primaryColor, |     bodyColor: '#ffffff' | ||||||
|     primaryColor: primaryColor, |   }, | ||||||
|     primaryColorHover: primaryColor | 
 | ||||||
|  |   Dialog: { | ||||||
|  |     borderRadius: '10px' | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -7,21 +7,21 @@ const settingsStore = useSettingsStore() | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <section class="top-container flex-center"> |   <section class="container flex-center"> | ||||||
|     <section |     <section | ||||||
|       class="el-container im-container" |       class="el-container im-container" | ||||||
|       :class="{ |       :class="{ | ||||||
|         'small-screen': !settingsStore.isFullScreen |         'small-screen': !settingsStore.isFullScreen | ||||||
|       }" |       }" | ||||||
|     > |     > | ||||||
|       <!-- <aside |       <aside | ||||||
|         class="el-aside" |         class="el-aside" | ||||||
|         :class="{ |         :class="{ | ||||||
|           'pd-t15': isElectronMode() |           'pd-t15': isElectronMode() | ||||||
|         }" |         }" | ||||||
|       > |       > | ||||||
|         <Menu /> |         <Menu /> | ||||||
|       </aside> --> |       </aside> | ||||||
|       <main class="el-main"> |       <main class="el-main"> | ||||||
|         <router-view /> |         <router-view /> | ||||||
|       </main> |       </main> | ||||||
| @ -30,8 +30,8 @@ const settingsStore = useSettingsStore() | |||||||
| 
 | 
 | ||||||
| </template> | </template> | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .top-container { | .container { | ||||||
|  width: 100%; |   | ||||||
|   background: url(@/assets/image/background.jpeg); |   background: url(@/assets/image/background.jpeg); | ||||||
|   background-position: center center; |   background-position: center center; | ||||||
|   background-repeat: no-repeat; |   background-repeat: no-repeat; | ||||||
| @ -39,8 +39,8 @@ const settingsStore = useSettingsStore() | |||||||
|   background-size: cover; |   background-size: cover; | ||||||
| 
 | 
 | ||||||
|   .im-container { |   .im-container { | ||||||
|     height: 85vh; |     height: 80vh; | ||||||
|  width: 100%; |     width: 100vw; | ||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
|     background-color: #fff; |     background-color: #fff; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -226,30 +226,6 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|         useEditorStore().loadUserEmoticon() |         useEditorStore().loadUserEmoticon() | ||||||
|         window['$message'] && window['$message'].success('收藏成功') |         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,34 +1,10 @@ | |||||||
| import { defineStore } from 'pinia' | import { defineStore } from 'pinia' | ||||||
| import { ServeFindFileSplitInfo, ServeFileSubareaUpload } from '@/api/upload' | import { ServeFindFileSplitInfo, ServeFileSubareaUpload } from '@/api/upload' | ||||||
| import { ServeSendTalkFile } from '@/api/chat' | import { ServeSendTalkFile } from '@/api/chat' | ||||||
| import { uploadImg } from '@/api/upload' | 
 | ||||||
| import { |  | ||||||
|   useDialogueStore |  | ||||||
| } from '@/store' |  | ||||||
| // @ts-ignore
 | // @ts-ignore
 | ||||||
| const message = window.$message | 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) { | function fileSlice(file: File, uploadId: string, eachSize: number) { | ||||||
|   const splitNum = Math.ceil(file.size / eachSize) // 分片总数
 |   const splitNum = Math.ceil(file.size / eachSize) // 分片总数
 | ||||||
| @ -55,8 +31,7 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|   state: () => { |   state: () => { | ||||||
|     return { |     return { | ||||||
|       isShow: false, |       isShow: false, | ||||||
|       items: [] as UploadItem[], |       items: [] | ||||||
|       dialogueStore: useDialogueStore() |  | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   getters: { |   getters: { | ||||||
| @ -70,282 +45,81 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|     close() { |     close() { | ||||||
|       this.isShow = false |       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) { |         if (res.code == 200) { | ||||||
|           const { upload_id, split_size } = res.data |           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
 |           // @ts-ignore
 | ||||||
|           this.items.unshift({ |           this.items.unshift({ | ||||||
|             file: file, |             file: file, | ||||||
|             talk_type: talkType, |             talk_type: talkType, | ||||||
|             receiver_id: receiverId, |             receiver_id: receiverId, | ||||||
|             upload_id: upload_id, |             upload_id: upload_id, | ||||||
|             client_upload_id: uploadId, // 客户端生成的上传ID,用于前端标识
 |  | ||||||
|             uploadIndex: 0, |             uploadIndex: 0, | ||||||
|             percentage: 0, |             percentage: 0, | ||||||
|             status: 0, // 文件上传状态 0:等待上传 1:上传中 2:上传完成 3:网络异常
 |             status: 0, // 文件上传状态 0:等待上传 1:上传中 2:上传完成 3:网络异常
 | ||||||
|             files: fileChunks, |             files: fileSlice(file, upload_id, split_size), | ||||||
|             avatar: '', |             avatar: '', | ||||||
|             username: username, |             username: username | ||||||
|             is_paused: false, |  | ||||||
|             onProgress: onProgress, |  | ||||||
|             onComplete: onComplete, |  | ||||||
|           }) |           }) | ||||||
| 
 | 
 | ||||||
|           this.isShow = false // 不显示上传管理抽屉
 |           this.triggerUpload(upload_id) | ||||||
|            |           this.isShow = true | ||||||
|           // 开始上传分片
 |  | ||||||
|           this.triggerUpload(upload_id, uploadId) |  | ||||||
|         } else { |         } else { | ||||||
|           message.error(res.message) |           message.error(res.message) | ||||||
|           onProgress(-1) // 通知上传失败
 |  | ||||||
|         } |         } | ||||||
|       } catch (error) { |       }) | ||||||
|         console.error("初始化分片上传失败:", error); |  | ||||||
|         message.error("初始化上传失败,请重试") |  | ||||||
|         onProgress(-1) |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 触发分片上传
 |     // 获取分片文件数组索引
 | ||||||
|     async triggerUpload(uploadId: string, clientUploadId?: string) { |     findItem(uploadId: string): any { | ||||||
|       const currentItem = this.findItem(uploadId) |       return this.items.find((item: any) => item.upload_id === uploadId) | ||||||
|       if (!currentItem) return |     }, | ||||||
| 
 | 
 | ||||||
|       // 如果已暂停,不继续上传
 |     // 触发上传
 | ||||||
|       if (currentItem.is_paused) return |     triggerUpload(uploadId: string) { | ||||||
|  |       const item = this.findItem(uploadId) | ||||||
| 
 | 
 | ||||||
|       // 如果已上传完成,不继续上传
 |       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) | ||||||
|       currentItem.status = 1 |         .then((res) => { | ||||||
|  |           if (res.code == 200) { | ||||||
|  |             item.uploadIndex++ | ||||||
| 
 | 
 | ||||||
|       // 上传当前分片
 |             if (item.uploadIndex === item.files.length) { | ||||||
|       try { |               item.status = 2 | ||||||
|         const res = await ServeFileSubareaUpload(form) |               item.percentage = 100 | ||||||
|          |               this.sendUploadMessage(item) | ||||||
|         // 获取最新的项目状态,确保仍然存在且没有被暂停
 |             } else { | ||||||
|         const updatedItem = this.findItem(uploadId) |               const percentage = (item.uploadIndex / item.files.length) * 100 | ||||||
|         if (!updatedItem || updatedItem.is_paused) return |               item.percentage = percentage.toFixed(1) | ||||||
|          |               this.triggerUpload(uploadId) | ||||||
|         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 { |           } else { | ||||||
|             // 继续上传下一个分片
 |             item.status = 3 | ||||||
|             this.triggerUpload(uploadId, clientUploadId) |  | ||||||
|           } |           } | ||||||
|         } 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, '网络错误,正在重试') |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|      |  | ||||||
|     // 重试上传
 |  | ||||||
|     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 |  | ||||||
|         }) |         }) | ||||||
|          |         .catch(() => { | ||||||
|         if (item.onComplete) { |           item.status = 3 | ||||||
|           item.onComplete(item) |         }) | ||||||
|         } |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error("发送文件消息失败:", error); |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|      |     // 发送上传消息
 | ||||||
|     pauseUpload(clientUploadId: string) { |     sendUploadMessage(item: any) { | ||||||
|       const item = this.findItemByClientId(clientUploadId) |       ServeSendTalkFile({ | ||||||
|       if (!item) return |         upload_id: item.upload_id, | ||||||
|        |         receiver_id: item.receiver_id, | ||||||
|       item.is_paused = true |         talk_type: item.talk_type | ||||||
|     }, |       }) | ||||||
|      |     } | ||||||
|     // 恢复上传
 |  | ||||||
|     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 { ServeGroupApplyUnread } from '@/api/group' | ||||||
| import { delAccessToken } from '@/utils/auth' | import { delAccessToken } from '@/utils/auth' | ||||||
| import { storage } from '@/utils/storage' | import { storage } from '@/utils/storage' | ||||||
| import { GetUserInfo } from '@/api/auth' | 
 | ||||||
| interface UserStoreState { | interface UserStoreState { | ||||||
|   uid: number |   uid: number | ||||||
|   nickname: string |   nickname: string | ||||||
| @ -35,7 +35,7 @@ export const useUserStore = defineStore('user', { | |||||||
|       online: false, // 在线状态
 |       online: false, // 在线状态
 | ||||||
|       isQiye: false, |       isQiye: false, | ||||||
|       isContactApply: false, |       isContactApply: false, | ||||||
|       isGroupApply: false, |       isGroupApply: false | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   getters: {}, |   getters: {}, | ||||||
|  | |||||||
| @ -47,10 +47,7 @@ export interface ITalkRecord { | |||||||
|   extra: any |   extra: any | ||||||
|   isCheck: boolean |   isCheck: boolean | ||||||
|   send_status: number |   send_status: number | ||||||
|   float: string, |   float: string | ||||||
|   is_convert_text?:number//语音记录的 是否是在展示转文本状态 1:是 0:否,
 |  | ||||||
|   erp_user_id:number |  | ||||||
|    |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export interface ITalkRecordExtraText { | export interface ITalkRecordExtraText { | ||||||
| @ -68,7 +65,6 @@ export interface ITalkRecordExtraFile { | |||||||
|   name: string |   name: string | ||||||
|   path: string |   path: string | ||||||
|   size: number |   size: number | ||||||
|   percentage: number |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export interface ITalkRecordExtraForward { | export interface ITalkRecordExtraForward { | ||||||
| @ -94,9 +90,6 @@ export interface ITalkRecordExtraVideo { | |||||||
|   url: string |   url: string | ||||||
|   duration: number |   duration: number | ||||||
|   size: number |   size: number | ||||||
|   is_uploading?: boolean |  | ||||||
|   upload_id?: string |  | ||||||
|   percentage?: number |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export interface ITalkRecordExtraMixed { | export interface ITalkRecordExtraMixed { | ||||||
|  | |||||||
| @ -10,36 +10,29 @@ defineProps({ | |||||||
|   username: String, |   username: String, | ||||||
|   active: Boolean |   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> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <div class="talk pointer" :class="{ actived: active }" @click="emit('tab-talk', data)"> |   <div class="talk pointer" :class="{ actived: active }" @click="emit('tab-talk', data)"> | ||||||
|     <div class="avatar-box relative"> |     <div class="avatar-box"> | ||||||
|        |        | ||||||
|       <avatarModule    :mode="data?.group_type === 0 ? 1 : 2" |       <avatarModule    :mode="data?.group_type === 0 ? 1 : 2" | ||||||
|               :avatar="data?.avatar" |               :avatar="data?.avatar" | ||||||
|               :groupType="data?.group_type" |               :groupType="data?.group_type" | ||||||
|               :userName="data?.name" :customStyle="{width:'42px',height:'42px'}"></avatarModule> |               :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" /> --> |       <!-- <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" /> |         <n-icon :component="data.is_top == 1 ? ArrowDown : ArrowUp" /> | ||||||
|       </div> --> |       </div> | ||||||
|     </div> |     </div> | ||||||
| 
 | 
 | ||||||
|     <div class="content-box"> |     <div class="content-box"> | ||||||
|       <div class="header"> |       <div class="header"> | ||||||
|         <div class="title"> |         <div class="title"> | ||||||
|           <span class="nickname">{{ username }}</span> |           <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 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> | ||||||
|         <div class="datetime"><Xtime :time="data.updated_at" /></div> |         <div class="datetime"><Xtime :time="data.updated_at" /></div> | ||||||
|       </div> |       </div> | ||||||
| @ -87,8 +80,8 @@ const labelColor=[ | |||||||
|   border-radius: 5px; |   border-radius: 5px; | ||||||
| 
 | 
 | ||||||
|   .avatar-box { |   .avatar-box { | ||||||
|     height: 42px; |     height: 34px; | ||||||
|     width: 42px; |     width: 34px; | ||||||
|     border-radius: 50%; |     border-radius: 50%; | ||||||
|     display: flex; |     display: flex; | ||||||
|     justify-content: center; |     justify-content: center; | ||||||
| @ -97,7 +90,7 @@ const labelColor=[ | |||||||
|     user-select: none; |     user-select: none; | ||||||
|     transition: ease 1s; |     transition: ease 1s; | ||||||
|     position: relative; |     position: relative; | ||||||
| 
 |     overflow: hidden; | ||||||
| 
 | 
 | ||||||
|     .top-mask { |     .top-mask { | ||||||
|       width: 100%; |       width: 100%; | ||||||
| @ -200,9 +193,9 @@ const labelColor=[ | |||||||
|           user-select: none; |           user-select: none; | ||||||
| 
 | 
 | ||||||
|           .badge { |           .badge { | ||||||
|             background-color: #D03050; |             background-color: #f44336; | ||||||
|             color: #ffffff; |             color: #ffffff; | ||||||
|             border-radius: 50%; |             border-radius: 3px; | ||||||
|             transform-origin: right; |             transform-origin: right; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -69,49 +69,38 @@ const onContactModal = (data: { id: number; type: number }[]) => { | |||||||
|     <div class="multi-title"> |     <div class="multi-title"> | ||||||
|       <span>已选中:{{ dialogueStore.selectItems.length }} 条消息</span> |       <span>已选中:{{ dialogueStore.selectItems.length }} 条消息</span> | ||||||
|     </div> |     </div> | ||||||
|     <div class="flex items-center relative"> |     <div class="multi-groups"> | ||||||
|       <div class="multi-groups"> |       <div class="btn-group"> | ||||||
|       <div class="btn-group mr-156px"> |  | ||||||
|         <div class="multi-icon pointer flex-center" @click="onMergeForward"> |         <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> |         </div> | ||||||
|         <p>合并转发</p> |         <p>合并转发</p> | ||||||
|       </div> |       </div> | ||||||
|       <div class="btn-group mr-156px"> |       <div class="btn-group"> | ||||||
|         <div class="multi-icon pointer flex-center" @click="onSingleForward"> |         <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> |         </div> | ||||||
|         <p>逐条转发</p> |         <p>逐条转发</p> | ||||||
|       </div> |       </div> | ||||||
|       <div class="btn-group "> |       <div class="btn-group"> | ||||||
|         <div class="multi-icon pointer flex-center" @click="onMultiDelete"> |         <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> |         </div> | ||||||
|         <p>批量删除</p> |         <p>批量删除</p> | ||||||
|       </div> |       </div> | ||||||
|       <!-- <div class="btn-group"> |       <div class="btn-group"> | ||||||
|         <div class="multi-icon pointer flex-center" @click="onClose"> |         <div class="multi-icon pointer flex-center" @click="onClose"> | ||||||
|           |           <n-icon :size="22" :component="Close" /> | ||||||
|         </div> |         </div> | ||||||
|         <p>关闭</p> |         <p>关闭</p> | ||||||
|       </div> --> |       </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> |   </section> | ||||||
| 
 | 
 | ||||||
|   <ContactModal |   <ContactModal | ||||||
|     v-if="isShowContactModal" |     v-if="isShowContactModal" | ||||||
|     v-on:close="isShowContactModal = false" |     v-on:close="isShowContactModal = false" | ||||||
|     v-on:on-submit="onContactModal" |     v-on:on-submit="onContactModal" | ||||||
|     :forward-mode="forwardMode" |  | ||||||
|   /> |   /> | ||||||
| </template> | </template> | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| @ -137,9 +126,13 @@ const onContactModal = (data: { id: number; type: number }[]) => { | |||||||
|     justify-content: center; |     justify-content: center; | ||||||
| 
 | 
 | ||||||
|     .btn-group { |     .btn-group { | ||||||
|  |       width: 50px; | ||||||
|  |       height: 80px; | ||||||
|  |       margin: 0 15px; | ||||||
|  | 
 | ||||||
|       .multi-icon { |       .multi-icon { | ||||||
|         width: 72px; |         width: 50px; | ||||||
|         height: 72px; |         height: 50px; | ||||||
|         background-color: var(--im-active-bg-color); |         background-color: var(--im-active-bg-color); | ||||||
|         border-radius: 50%; |         border-radius: 50%; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,10 +13,7 @@ import SkipBottom from './SkipBottom.vue' | |||||||
| import { ITalkRecord } from '@/types/chat' | import { ITalkRecord } from '@/types/chat' | ||||||
| import { EditorConst } from '@/constant/event-bus' | import { EditorConst } from '@/constant/event-bus' | ||||||
| import { useInject, useTalkRecord, useUtil } from '@/hooks' | 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({ | const props = defineProps({ | ||||||
|   uid: { |   uid: { | ||||||
|     type: Number, |     type: Number, | ||||||
| @ -42,12 +39,15 @@ const { useMessage } = useUtil() | |||||||
| const { dropdown, showDropdownMenu, closeDropdownMenu } = useMenu() | const { dropdown, showDropdownMenu, closeDropdownMenu } = useMenu() | ||||||
| const { showUserInfoModal } = useInject() | const { showUserInfoModal } = useInject() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| const userStore = useUserStore() | 
 | ||||||
| // const showUserInfoModal = (uid: number) => { | watch(() => records, (newValue, oldValue) => { | ||||||
| //   userStore.getUserInfo(uid) |   console.log(newValue); | ||||||
| // } | 
 | ||||||
|  | },{deep:true,immediate:true}) | ||||||
|  | 
 | ||||||
| // 置底按钮 | // 置底按钮 | ||||||
| const skipBottom = ref(false) | const skipBottom = ref(false) | ||||||
|  | 
 | ||||||
| // 是否显示消息时间 | // 是否显示消息时间 | ||||||
| const isShowTalkTime = (index: number, datetime: string) => { | const isShowTalkTime = (index: number, datetime: string) => { | ||||||
|   if (datetime == undefined) { |   if (datetime == undefined) { | ||||||
| @ -236,17 +236,6 @@ const onContextMenu = (e: any, item: ITalkRecord) => { | |||||||
|   e.preventDefault() |   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 = { | const evnets = { | ||||||
|   copy: onCopyText, |   copy: onCopyText, | ||||||
|   revoke: onRevokeTalk, |   revoke: onRevokeTalk, | ||||||
| @ -254,9 +243,7 @@ const evnets = { | |||||||
|   multiSelect: onMultiSelect, |   multiSelect: onMultiSelect, | ||||||
|   download: onDownloadFile, |   download: onDownloadFile, | ||||||
|   quote: onQuoteMessage, |   quote: onQuoteMessage, | ||||||
|   collect: onCollectImage, |   collect: onCollectImage | ||||||
|   convertText: onConvertText, |  | ||||||
|   closeConvertText:onloseConvertText |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 会话列表右键菜单回调事件 | // 会话列表右键菜单回调事件 | ||||||
| @ -287,7 +274,11 @@ onMounted(() => { | |||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <section class="section"> |   <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"> |       <div class="load-toolbar pointer"> | ||||||
|         <span v-if="loadConfig.status == 0"> 正在加载数据中 ... </span> |         <span v-if="loadConfig.status == 0"> 正在加载数据中 ... </span> | ||||||
| @ -295,54 +286,89 @@ onMounted(() => { | |||||||
|         <span v-else class="no-more"> 没有更多消息了 </span> |         <span v-else class="no-more"> 没有更多消息了 </span> | ||||||
|       </div> |       </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"> |         <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> | ||||||
| 
 | 
 | ||||||
|         <!-- 撤回消息 --> |         <!-- 撤回消息 --> | ||||||
|         <div v-else-if="item.is_revoke == 1" class="message-box"> |         <div v-else-if="item.is_revoke == 1" class="message-box"> | ||||||
|           <revoke-message :login_uid="uid" :data="item" :user_id="item.user_id" :nickname="item.nickname" :talk_type="item.talk_type" |           <revoke-message | ||||||
|             :datetime="item.created_at" /> |             :login_uid="uid" | ||||||
|  |             :user_id="item.user_id" | ||||||
|  |             :nickname="item.nickname" | ||||||
|  |             :talk_type="item.talk_type" | ||||||
|  |             :datetime="item.created_at" | ||||||
|  |           /> | ||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|         <div v-else class="message-box record-box" :class="{ |         <div | ||||||
|           'direction-rt': item.float == 'right', |           v-else | ||||||
|           'multi-select': dialogueStore.isOpenMultiSelect, |           class="message-box record-box" | ||||||
|           'multi-select-check': item.isCheck |           :class="{ | ||||||
|         }"> |             'direction-rt': item.float == 'right', | ||||||
|  |             'multi-select': dialogueStore.isOpenMultiSelect, | ||||||
|  |             'multi-select-check': item.isCheck | ||||||
|  |           }" | ||||||
|  |         > | ||||||
|           <!-- 多选按钮 --> |           <!-- 多选按钮 --> | ||||||
|           <aside v-if="dialogueStore.isOpenMultiSelect" class="checkbox-column"> |           <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> | ||||||
|           <!-- 头像信息 --> |  | ||||||
| 
 | 
 | ||||||
|  |           <!-- 头像信息 --> | ||||||
|           <aside class="avatar-column"> |           <aside class="avatar-column"> | ||||||
|             <im-avatar class="pointer" :src="item.avatar" :size="42" :username="item.nickname" |             <im-avatar | ||||||
|               @click="showUserInfoModal(item.erp_user_id)" /> |               class="pointer" | ||||||
|  |               :src="item.avatar" | ||||||
|  |               :size="30" | ||||||
|  |               :username="item.nickname" | ||||||
|  |               @click="showUserInfoModal(item.user_id)" | ||||||
|  |             /> | ||||||
|           </aside> |           </aside> | ||||||
| 
 | 
 | ||||||
|           <!-- 主体信息 --> |           <!-- 主体信息 --> | ||||||
|           <main class="main-column"> |           <main class="main-column"> | ||||||
|             <div class="talk-title"> |             <div class="talk-title"> | ||||||
|               <span class="nickname pointer" v-show="talk_type == 2 && item.float == 'left'" |               <span | ||||||
|                 @click="onClickNickname(item)"> |                 class="nickname pointer" | ||||||
|  |                 v-show="talk_type == 2 && item.float == 'left'" | ||||||
|  |                 @click="onClickNickname(item)" | ||||||
|  |               > | ||||||
|                 <span class="at">@</span>{{ item.nickname }} |                 <span class="at">@</span>{{ item.nickname }} | ||||||
|               </span> |               </span> | ||||||
|               <span>{{ parseTime(item.created_at, '{y}/{m}/{d} {h}:{i}') }}</span> |               <span>{{ parseTime(item.created_at, '{m}/{d} {h}:{i}') }}</span> | ||||||
|             </div> |             </div> | ||||||
| 
 | 
 | ||||||
|             <div class="talk-content" :class="{ pointer: dialogueStore.isOpenMultiSelect }" @click="onRowClick(item)"> |             <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)" | ||||||
|  |               /> | ||||||
| 
 | 
 | ||||||
|               <component :is="MessageComponents[item.msg_type] || 'unknown-message'" :extra="item.extra" :data="item" |               <div class="talk-tools"> | ||||||
|                 :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'"> |                 <template v-if="talk_type == 1 && item.float == 'right'"> | ||||||
|                   <loading |                   <loading | ||||||
|                     theme="outline" |                     theme="outline" | ||||||
| @ -354,14 +380,22 @@ 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> |                   <!-- <span v-show="item.send_status != 1"> 已送达 </span> --> | ||||||
|                 </template> |                 </template> | ||||||
| 
 | 
 | ||||||
| <n-icon class="more-tools pointer" :component="MoreThree" @click="onContextMenu($event, item)" /> |                 <n-icon | ||||||
| </div> --> |                   class="more-tools pointer" | ||||||
|  |                   :component="MoreThree" | ||||||
|  |                   @click="onContextMenu($event, item)" | ||||||
|  |                 /> | ||||||
|  |               </div> | ||||||
|             </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" /> |               <n-icon :component="ToTop" size="14" class="icon-top" /> | ||||||
|               <span class="ellipsis"> |               <span class="ellipsis"> | ||||||
|                 回复 {{ item.extra?.reply?.nickname }}: |                 回复 {{ item.extra?.reply?.nickname }}: | ||||||
| @ -382,8 +416,14 @@ onMounted(() => { | |||||||
|   </section> |   </section> | ||||||
| 
 | 
 | ||||||
|   <!-- 右键菜单 --> |   <!-- 右键菜单 --> | ||||||
|   <n-dropdown :show="dropdown.show" :x="dropdown.x" :y="dropdown.y" style="width: 142px;" :options="dropdown.options" |   <n-dropdown | ||||||
|     @select="onContextMenuHandle" @clickoutside="closeDropdownMenu" /> |     :show="dropdown.show" | ||||||
|  |     :x="dropdown.x" | ||||||
|  |     :y="dropdown.y" | ||||||
|  |     :options="dropdown.options" | ||||||
|  |     @select="onContextMenuHandle" | ||||||
|  |     @clickoutside="closeDropdownMenu" | ||||||
|  |   /> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| @ -408,7 +448,6 @@ onMounted(() => { | |||||||
|     text-align: center; |     text-align: center; | ||||||
|     line-height: 38px; |     line-height: 38px; | ||||||
|     font-size: 13px; |     font-size: 13px; | ||||||
| 
 |  | ||||||
|     .no-more { |     .no-more { | ||||||
|       color: #b9b3b3; |       color: #b9b3b3; | ||||||
|     } |     } | ||||||
| @ -450,7 +489,7 @@ onMounted(() => { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     .avatar-column { |     .avatar-column { | ||||||
|       width: 47px; |       width: 35px; | ||||||
|       display: flex; |       display: flex; | ||||||
|       align-items: center; |       align-items: center; | ||||||
|       order: 2; |       order: 2; | ||||||
| @ -485,7 +524,6 @@ onMounted(() => { | |||||||
|         .nickname { |         .nickname { | ||||||
|           color: var(--im-text-color); |           color: var(--im-text-color); | ||||||
|           margin-right: 5px; |           margin-right: 5px; | ||||||
|           font-size: 12px; |  | ||||||
| 
 | 
 | ||||||
|           .at { |           .at { | ||||||
|             display: none; |             display: none; | ||||||
|  | |||||||
| @ -5,18 +5,16 @@ import { | |||||||
|   useDialogueStore, |   useDialogueStore, | ||||||
|   useSettingsStore, |   useSettingsStore, | ||||||
|   useUploadsStore, |   useUploadsStore, | ||||||
|   useEditorStore, |   useEditorStore | ||||||
|   useUserStore |  | ||||||
| } from '@/store' | } from '@/store' | ||||||
| import ws from '@/connect' | import ws from '@/connect' | ||||||
| import { ServePublishMessage, ServeSendVote } from '@/api/chat' | import { ServePublishMessage, ServeSendVote } from '@/api/chat' | ||||||
| import { throttle, getVideoImage } from '@/utils/common' | import { throttle, getVideoImage } from '@/utils/common' | ||||||
| import { parseTime } from '@/utils/datetime' |  | ||||||
| import Editor from '@/components/editor/Editor.vue' | import Editor from '@/components/editor/Editor.vue' | ||||||
| import MultiSelectFooter from './MultiSelectFooter.vue' | import MultiSelectFooter from './MultiSelectFooter.vue' | ||||||
| import HistoryRecord from '@/components/talk/HistoryRecord.vue' | import HistoryRecord from '@/components/talk/HistoryRecord.vue' | ||||||
| import { uploadImg } from '@/api/upload' | import { uploadImg } from '@/api/upload' | ||||||
| const userStore = useUserStore() | 
 | ||||||
| const talkStore = useTalkStore() | const talkStore = useTalkStore() | ||||||
| const editorStore = useEditorStore() | const editorStore = useEditorStore() | ||||||
| const settingsStore = useSettingsStore() | const settingsStore = useSettingsStore() | ||||||
| @ -100,78 +98,26 @@ const onSendImageEvent = ({ data, callBack }) => { | |||||||
| 
 | 
 | ||||||
| // 发送视频消息 | // 发送视频消息 | ||||||
| const onSendVideoEvent = async ({ data }) => { | const onSendVideoEvent = async ({ data }) => { | ||||||
|   console.log('onSendVideoEvent') |   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(form.get('file')); | ||||||
|   // let resp = await getVideoImage(data) |   let video = await uploadImg(form) | ||||||
|  |   if (video.code != 0) return | ||||||
| 
 | 
 | ||||||
|   // 先创建一个带有上传ID的临时消息对象,用于显示进度 |   let message = { | ||||||
|   const uploadId = `video-${Date.now()}-${Math.floor(Math.random() * 1000)}` |     type: 'video', | ||||||
|    |     url: video.data.ori_url, | ||||||
|   // 创建临时消息记录 |     cover: video.data.cover_url, | ||||||
|   const tempMessage = { |     duration: parseInt(resp.duration), | ||||||
|     msg_id: uploadId, |     size: data.size | ||||||
|     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]) |  | ||||||
|       // }) |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 发送代码消息 | // 发送代码消息 | ||||||
| @ -185,41 +131,8 @@ const onSendFileEvent = ({ data }) => { | |||||||
|   if (data.size > maxsize) { |   if (data.size > maxsize) { | ||||||
|     return window['$message'].warning('上传文件不能超过100M!') |     return window['$message'].warning('上传文件不能超过100M!') | ||||||
|   } |   } | ||||||
|   const uploadId = `file-${Date.now()}-${Math.floor(Math.random() * 1000)}` |  | ||||||
| 
 | 
 | ||||||
|   const tempMessage = { |   uploadsStore.initUploadFile(data, props.talk_type, props.receiver_id, dialogueStore.talk.username) | ||||||
|     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,29 +30,21 @@ export function useMenu() { | |||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   const showDropdownMenu = (e: any, uid: number, item: any) => { |   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' }) |  | ||||||
|       } |  | ||||||
| 
 | 
 | ||||||
|     } |     dropdown.options = [] | ||||||
|     if ([1, 3].includes(item.msg_type)) { |     if ([1, 3].includes(item.msg_type)) { | ||||||
|       dropdown.options.push({ label: '复制', key: 'copy' }) |       dropdown.options.push({ label: '复制', key: 'copy' }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     dropdown.options.push({ label: '多选', key: 'multiSelect' }) |  | ||||||
|     dropdown.options.push({ label: '引用', key: 'quote' }) |  | ||||||
|     if (isRevoke(uid, item)) { |     if (isRevoke(uid, item)) { | ||||||
|       dropdown.options.push({ label: `撤回`, key: 'revoke' }) |       dropdown.options.push({ label: `撤回`, key: 'revoke' }) | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     dropdown.options.push({ label: '回复', key: 'quote' }) | ||||||
|     dropdown.options.push({ label: '删除', key: 'delete' }) |     dropdown.options.push({ label: '删除', key: 'delete' }) | ||||||
| 
 | 
 | ||||||
| 
 |     dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||||
| 
 | 
 | ||||||
|     if ([3, 4, 5].includes(item.msg_type)) { |     if ([3, 4, 5].includes(item.msg_type)) { | ||||||
|       dropdown.options.push({ label: '下载', key: 'download' }) |       dropdown.options.push({ label: '下载', key: 'download' }) | ||||||
| @ -62,7 +54,6 @@ export function useMenu() { | |||||||
|       dropdown.options.push({ label: '收藏', key: 'collect' }) |       dropdown.options.push({ label: '收藏', key: 'collect' }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     dropdown.x = e.clientX |     dropdown.x = e.clientX | ||||||
|     dropdown.y = e.clientY |     dropdown.y = e.clientY | ||||||
|     dropdown.show = true |     dropdown.show = true | ||||||
|  | |||||||
| @ -1,22 +1,17 @@ | |||||||
| import { defineConfig } from 'unocss' | import { defineConfig, presetUno, presetAttributify, presetIcons } from 'unocss' | ||||||
| import { presetUno, presetAttributify, presetIcons } from 'unocss' |  | ||||||
| 
 | 
 | ||||||
| export default defineConfig({ | export default defineConfig({ | ||||||
|   // 预设
 |   // 预设
 | ||||||
|   presets: [ |   presets: [ | ||||||
|     presetUno(),  |     presetUno(), // 添加核心预设
 | ||||||
|     presetAttributify(), // 启用属性模式
 |     presetAttributify(), // 启用属性模式
 | ||||||
|     presetIcons(), // 启用图标
 |     presetIcons(), // 启用图标
 | ||||||
|   ], |   ], | ||||||
|  |   // 自定义规则
 | ||||||
|   rules: [ |   rules: [ | ||||||
|     // 通过自定义规则覆盖默认的 container 样式
 |  | ||||||
|     ['container', { 'max-width': 'none' }], // 或者根据需要设置其他样式
 |  | ||||||
|   ], |  | ||||||
|   safelist: [ |  | ||||||
|     'container' // 确保 container 在 safelist 中,以便 UnoCSS 忽略它
 |  | ||||||
|   ], |   ], | ||||||
|   // 快捷方式
 |   // 快捷方式
 | ||||||
|   shortcuts: { |   shortcuts: { | ||||||
|     'uno-container': 'container' // 创建 container 的别名
 |     'btn': 'px-4 py-2 rounded-lg bg-blue-500 text-white hover:bg-blue-600', | ||||||
|   }, |   }, | ||||||
| })  | })  | ||||||
| @ -4,9 +4,6 @@ import vue from '@vitejs/plugin-vue' | |||||||
| import vueJsx from '@vitejs/plugin-vue-jsx' | import vueJsx from '@vitejs/plugin-vue-jsx' | ||||||
| import compressPlugin from 'vite-plugin-compression' | import compressPlugin from 'vite-plugin-compression' | ||||||
| import UnoCSS from 'unocss/vite' | 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'; | import vueDevTools from 'vite-plugin-vue-devtools'; | ||||||
| // https://vitejs.dev/config/
 | // https://vitejs.dev/config/
 | ||||||
| export default defineConfig(({ mode }) => { | export default defineConfig(({ mode }) => { | ||||||
| @ -27,28 +24,12 @@ export default defineConfig(({ mode }) => { | |||||||
|     assetsInclude: ['./src/assets'], |     assetsInclude: ['./src/assets'], | ||||||
|     plugins: [ |     plugins: [ | ||||||
|       vue(),  |       vue(),  | ||||||
|       AutoImport({ |  | ||||||
|         imports: [ |  | ||||||
|           'vue', |  | ||||||
|           { |  | ||||||
|             'naive-ui': [ |  | ||||||
|               'useDialog', |  | ||||||
|               'useMessage', |  | ||||||
|               'useNotification', |  | ||||||
|               'useLoadingBar' |  | ||||||
|             ] |  | ||||||
|           } |  | ||||||
|         ] |  | ||||||
|       }), |  | ||||||
|       Components({ |  | ||||||
|         resolvers: [NaiveUiResolver()] |  | ||||||
|       }), |  | ||||||
|       vueJsx({}),  |       vueJsx({}),  | ||||||
|       compressPlugin(),  |       compressPlugin(),  | ||||||
|       UnoCSS(), |       UnoCSS(), | ||||||
|       vueDevTools({ |       // vueDevTools({
 | ||||||
|         launchEditor: 'cursor', |       //   launchEditor: 'cursor',
 | ||||||
|       }) |       // })
 | ||||||
|     ], |     ], | ||||||
|     define: { |     define: { | ||||||
|       __APP_ENV__: env.APP_ENV |       __APP_ENV__: env.APP_ENV | ||||||
|  | |||||||