Compare commits
	
		
			6 Commits
		
	
	
		
			main
			...
			zhangyuans
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 42c1b64a99 | ||
|  | ed8da6cde9 | ||
|  | e832181432 | ||
|  | 7baca98e4d | ||
|  | 5962e0b12d | ||
|  | 4f66ff8b1d | 
							
								
								
									
										2
									
								
								auto-imports.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -70,6 +70,6 @@ declare global { | ||||
| // 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' | ||||
|   export type { Component, ComponentPublicInstance, ComputedRef, DirectiveBinding, ExtractDefaultPropTypes, ExtractPropTypes, ExtractPublicPropTypes, InjectionKey, PropType, Ref, MaybeRef, MaybeRefOrGetter, VNode, WritableComputedRef } from 'vue' | ||||
|   import('vue') | ||||
| } | ||||
|  | ||||
							
								
								
									
										8
									
								
								components.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -8,8 +8,6 @@ export {} | ||||
| /* prettier-ignore */ | ||||
| declare module 'vue' { | ||||
|   export interface GlobalComponents { | ||||
|     AsyncError: typeof import('./src/components/async-error/index.vue')['default'] | ||||
|     AsyncLoading: typeof import('./src/components/async-loading/index.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'] | ||||
| @ -32,8 +30,6 @@ declare module 'vue' { | ||||
|     LoginMessage: typeof import('./src/components/talk/message/LoginMessage.vue')['default'] | ||||
|     Message: typeof import('./src/components/x-message/message/index.vue')['default'] | ||||
|     MixedMessage: typeof import('./src/components/talk/message/MixedMessage.vue')['default'] | ||||
|     NButton: typeof import('naive-ui')['NButton'] | ||||
|     NIcon: typeof import('naive-ui')['NIcon'] | ||||
|     PageAnimation: typeof import('./src/components/page-animation/index.vue')['default'] | ||||
|     RevokeMessage: typeof import('./src/components/talk/message/RevokeMessage.vue')['default'] | ||||
|     RouterLink: typeof import('vue-router')['RouterLink'] | ||||
| @ -41,17 +37,13 @@ declare module 'vue' { | ||||
|     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'] | ||||
|     SysPushMessage: typeof import('./src/components/talk/message/system/sysPushMessage.vue')['default'] | ||||
|     SysTextMessage: typeof import('./src/components/talk/message/system/SysTextMessage.vue')['default'] | ||||
|     TabbarItem: typeof import('./src/components/x-tabbar/components/tabbar-item/index.vue')['default'] | ||||
|     TextMessage: typeof import('./src/components/talk/message/TextMessage.vue')['default'] | ||||
|  | ||||
							
								
								
									
										6
									
								
								env/.env.dev
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -5,10 +5,8 @@ VITE_SHOW_CONSOLE = true | ||||
| # 是否开启sourcemap | ||||
| VITE_SHOW_SOURCEMAP = true | ||||
| # baseUrl | ||||
| # VITE_BASEURL = 'http://172.16.100.93:8503' | ||||
| VITE_BASEURL = 'http://192.168.88.47:9503' | ||||
| VITE_BASEURL = 'http://172.16.100.93:8503' | ||||
| #VITE_SOCKET_API | ||||
| # VITE_SOCKET_API = 'ws://172.16.100.93:8504' | ||||
| VITE_SOCKET_API = 'ws://192.168.88.47:9504' | ||||
| VITE_SOCKET_API = 'ws://172.16.100.93:8504' | ||||
| # EPRAPI baseUrl | ||||
| VITE_EPR_BASEURL = 'http://114.218.158.24:9020' | ||||
|  | ||||
							
								
								
									
										15
									
								
								env/.env.prod
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -4,18 +4,5 @@ NODE_ENV = 'prod' | ||||
| VITE_SHOW_CONSOLE = false | ||||
| # 是否开启sourcemap | ||||
| VITE_SHOW_SOURCEMAP = false | ||||
| 
 | ||||
| # baseUrl | ||||
| VITE_BASEURL = 'https://chat-out.szjixun.cn' #体制外 | ||||
| #VITE_SOCKET_API | ||||
| VITE_SOCKET_API = 'wss://chat-out.szjixun.cn' #体制外 | ||||
| # EPRAPI baseUrl | ||||
| VITE_EPR_BASEURL = 'https://erpapi-out.szjixun.cn' #体制外 | ||||
| 
 | ||||
| # # baseUrl | ||||
| # VITE_BASEURL = 'https://chat.szjixun.cn' #体制内 | ||||
| # #VITE_SOCKET_API | ||||
| # VITE_SOCKET_API = 'wss://chat.szjixun.cn' #体制内 | ||||
| # # EPRAPI baseUrl | ||||
| # VITE_EPR_BASEURL = 'https://erpapi.fontree.cn' #体制内 | ||||
| 
 | ||||
| VITE_BASEURL = 'https://oa-a.szjixun.cn/api' | ||||
|  | ||||
							
								
								
									
										11
									
								
								env/.env.test
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -4,16 +4,9 @@ NODE_ENV = 'test' | ||||
| VITE_SHOW_CONSOLE = true | ||||
| # 是否开启sourcemap | ||||
| VITE_SHOW_SOURCEMAP = true | ||||
| 
 | ||||
| # # baseUrl | ||||
| VITE_BASEURL = 'http://172.16.100.93:8503' | ||||
| # #VITE_SOCKET_API | ||||
| VITE_SOCKET_API = 'ws://172.16.100.93:8504' | ||||
| 
 | ||||
| # baseUrl | ||||
| # VITE_BASEURL = 'http://192.168.88.21:9503' | ||||
| VITE_BASEURL = 'http://172.16.100.93:8503' | ||||
| #VITE_SOCKET_API | ||||
| # VITE_SOCKET_API = 'ws://192.168.88.21:9504' | ||||
| 
 | ||||
| VITE_SOCKET_API = 'ws://172.16.100.93:8504' | ||||
| # EPRAPI baseUrl | ||||
| VITE_EPR_BASEURL = 'http://114.218.158.24:9020' | ||||
|  | ||||
							
								
								
									
										10
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -5,7 +5,6 @@ | ||||
|   "packageManager": "pnpm@8.14.1", | ||||
|   "license": "MIT", | ||||
|   "scripts": { | ||||
|     "dev:h5": "uni --mode dev --port 2468", | ||||
|     "test:h5": "uni --mode test --port 2468", | ||||
|     "prod:h5": "uni --mode prod", | ||||
|     "build:h5:test": "uni build --mode test", | ||||
| @ -32,17 +31,16 @@ | ||||
|     "@uni-helper/axios-adapter": "^1.5.2", | ||||
|     "@uni-helper/localforage-adapter": "^1.0.2", | ||||
|     "@uni-helper/uni-use": "^0.19.12", | ||||
|     "@vueup/vue-quill": "^1.2.0", | ||||
|     "@vueuse/core": "^9.13.0", | ||||
|     "@vueup/vue-quill": "^1.2.0", | ||||
|     "quill": "^1.3.7", | ||||
|     "quill-mention": "^4.1.0", | ||||
|     "axios": "^1.7.2", | ||||
|     "dayjs": "^1.11.12", | ||||
|     "less": "^4.2.0", | ||||
|     "lodash": "^4.17.21", | ||||
|     "nzh": "^1.0.13", | ||||
|     "pinia-plugin-persistedstate": "^4.1.3", | ||||
|     "quill": "^1.3.7", | ||||
|     "quill-mention": "^4.1.0", | ||||
|     "recorder-core": "^1.3.25011100", | ||||
|     "vconsole": "^3.15.1", | ||||
|     "vue": "^3.3.8", | ||||
|     "vue-i18n": "11.0.0-rc.1" | ||||
| @ -66,7 +64,7 @@ | ||||
|     "lint-staged": "^15.2.0", | ||||
|     "naive-ui": "^2.41.0", | ||||
|     "pinia": "2.0.36", | ||||
|     "sass": "1.62.1", | ||||
|     "sass": "^1.77.8", | ||||
|     "simple-git-hooks": "^2.9.0", | ||||
|     "typescript": "^5.3.3", | ||||
|     "unocss": "^0.58.9", | ||||
|  | ||||
							
								
								
									
										5416
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										11
									
								
								src/App.vue
									
									
									
									
									
								
							
							
						
						| @ -1,11 +1,10 @@ | ||||
| <script setup> | ||||
| import { useStatus } from '@/store/status' | ||||
| import { useUserStore, useDialogueListStore } from '@/store' | ||||
| import { useUserStore } from '@/store' | ||||
| import { useProvideUserModal } from '@/hooks' | ||||
| import {useAuth} from "@/store/auth"; | ||||
| const {token} = useAuth() | ||||
| import ws from '@/connect' | ||||
| import {uniStorage} from "@/utils/uniStorage.js" | ||||
| const { statusBarHeight } = useStatus() | ||||
| const { uid, isShow } = useProvideUserModal() | ||||
| const userStore = useUserStore() | ||||
| @ -20,14 +19,10 @@ const handleWebview = () => { | ||||
|   // }) | ||||
|   console.log("webview", webview) | ||||
|   token.value = webview.token | ||||
|   if(webview?.doClearDialogueList){ | ||||
|     useDialogueListStore().dialogueList.value = [] | ||||
|     uniStorage.removeItem('dialogueList') | ||||
|   } | ||||
|   userStore.loadSetting() | ||||
|   ws.connect() | ||||
| } | ||||
| const init = () => { | ||||
|   userStore.loadSetting() | ||||
|   ws.connect() | ||||
|   if (typeof plus !== 'undefined') { | ||||
|     handleWebview() | ||||
|   } else { | ||||
|  | ||||
| @ -1,47 +0,0 @@ | ||||
| import request from '@/service/index.js' | ||||
| 
 | ||||
| // 查询用户是否需要添加好友
 | ||||
| export const ServeCheckFriend = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/friend/check', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 主动添加好友(单向好友)
 | ||||
| export const ServeAddFriend = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/friend/add', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 查询我的好友列表
 | ||||
| export const ServeQueryFriendsList = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/friend/list', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 删除好友(单向好友)
 | ||||
| export const ServeDeleteFriend = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/friend/delete', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| //添加我的好友时候的搜索接口
 | ||||
| export const ServeFriendSearch = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/friend/search', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| @ -1,7 +1,5 @@ | ||||
| import request from '@/service/index.js' | ||||
| import qs from 'qs' | ||||
| import { useTalkStore, useDialogueStore } from '@/store' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| 
 | ||||
| // 获取聊天列表服务接口
 | ||||
| export const ServeGetTalkList = (data) => { | ||||
| @ -40,15 +38,7 @@ export const ServeTopTalkList = (data) => { | ||||
| } | ||||
| 
 | ||||
| // 清除聊天消息未读数服务接口
 | ||||
| export const ServeClearTalkUnreadNum = (data, unReadNum) => { | ||||
|   console.log('=======chatApp==UnreadNum', unReadNum) | ||||
|   if ( | ||||
|     !useTalkStore().items[ | ||||
|       useTalkStore().findTalkIndex(useDialogueStore().index_name) | ||||
|     ]?.is_disturb | ||||
|   ) { | ||||
|     handleFindWebview(`updateUnreadMsgNumReduce('${unReadNum}')`) | ||||
|   } | ||||
| export const ServeClearTalkUnreadNum = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/talk/unread/clear', | ||||
|     method: 'POST', | ||||
| @ -68,7 +58,6 @@ export const ServeTalkRecords = (data) => { | ||||
| // 获取转发会话记录详情列表服务接口
 | ||||
| export const ServeGetForwardRecords = (data) => { | ||||
|   return request({ | ||||
|     // url: '/api/v1/talk/records/forward/v2',
 | ||||
|     url: '/api/v1/talk/records/forward', | ||||
|     method: 'GET', | ||||
|     data, | ||||
| @ -180,15 +169,14 @@ export const ServeConfirmVoteHandle = (data) => { | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const uploadImg = (data, onProgressFn) => { | ||||
| export const uploadImg = (data,onProgressFn) => { | ||||
|   return request({ | ||||
|     url: '/upload/img', | ||||
|     method: 'POST', | ||||
|     data: data, | ||||
|     baseURL: import.meta.env.VITE_EPR_BASEURL, | ||||
|     isFormData: true, | ||||
|     onUploadProgress: (progressEvent) => | ||||
|       onProgressFn(progressEvent, data.get('file')), | ||||
|     data:data, | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|     isFormData:true, | ||||
|     onUploadProgress:(progressEvent)=>onProgressFn(progressEvent,data.get('file')) | ||||
|   }) | ||||
| } | ||||
| // 根据msg_id获取消息
 | ||||
| @ -199,58 +187,3 @@ export const detailGetRecordsContext = (data) => { | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 获取自己消息已读回执列表
 | ||||
| export const ServeReadConditionList = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/talk/my-records/read/condition', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 获取消息已读未读详情
 | ||||
| export const ServeMessageReadDetail = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/talk/my-records/read/condition', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 语音转文字
 | ||||
| export const ServeConvertText = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/talk/message/voice-to-text', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 获取用户所在群聊列表(我的群聊)
 | ||||
| export const ServeUserGroupChatList = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/group/user/list', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //查询聊天助手是否开启
 | ||||
| export const ServeContactRobotQuery = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/robot/query', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //开关聊天助手
 | ||||
| export const ServeContactRobotUpdate = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/contact/robot/update', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -61,14 +61,6 @@ export const userV2List = (data) => { | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| // 过滤测试数据的erp用户接口
 | ||||
| export const userV2List2 = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/group/erp/users', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const groupCreateDept = (data) => { | ||||
|   return request({ | ||||
|  | ||||
| @ -4,7 +4,6 @@ import qs from 'qs' | ||||
| // ES搜索聊天记录-主页搜索什么都有
 | ||||
| export const ServeSeachQueryAll = (data) => { | ||||
|   return request({ | ||||
|     // url: '/api/v1/elasticsearch/query-all/v2',
 | ||||
|     url: '/api/v1/elasticsearch/query-all', | ||||
|     method: 'POST', | ||||
|     data, | ||||
| @ -14,7 +13,6 @@ export const ServeSeachQueryAll = (data) => { | ||||
| // ES搜索用户数据
 | ||||
| export const ServeQueryUser = (data) => { | ||||
|   return request({ | ||||
|     // url: '/api/v1/elasticsearch/query-user/v2',
 | ||||
|     url: '/api/v1/elasticsearch/query-user', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|  | ||||
| @ -10,9 +10,10 @@ export const ServeGetUserSetting = (data) => { | ||||
| 
 | ||||
| export const userInfoApi = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/users/erp/info', | ||||
|     url: '/user/info', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -1,89 +0,0 @@ | ||||
| <template> | ||||
|   <div class="chat-app-error-page"> | ||||
|     <div class="error-container"> | ||||
|       <div class="error-icon"> | ||||
|         <i class="iconfont icon-wifi"></i> | ||||
|       </div> | ||||
|       <div class="error-message"> | ||||
|         <span>您的网络好像波动了一下~</span> | ||||
|       </div> | ||||
|       <div class="reload-button" @click="handleReload"> | ||||
|         <span>重新加载</span> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import { onMounted, ref, computed, nextTick } from 'vue' | ||||
| 
 | ||||
| const handleReload = () => { | ||||
|   location.reload(true); | ||||
| } | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   if (typeof plus !== 'undefined') { | ||||
| 
 | ||||
|   } else { | ||||
|     document.addEventListener('plusready', () => { | ||||
| 
 | ||||
|     }) | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <style scoped lang="scss"> | ||||
| .chat-app-error-page { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
|   height: 100vh; | ||||
|   background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|   background-size: cover; | ||||
|   background-position: bottom center; | ||||
| 
 | ||||
|   .error-container { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     padding: 32px; | ||||
|     background: #ffffff; | ||||
|     border-radius: 16px; | ||||
|     box-shadow: 0 4px 12px rgba(0, 0, 0, 0.2); | ||||
|     max-width: 80%; | ||||
|     width: 320px; | ||||
| 
 | ||||
|     .error-icon { | ||||
|       font-size: 48px; | ||||
|       color: #ff6b6b; | ||||
|       margin-bottom: 16px; | ||||
|     } | ||||
| 
 | ||||
|     .error-message { | ||||
|       font-size: 16px; | ||||
|       color: #333333; | ||||
|       margin-bottom: 24px; | ||||
|       text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .reload-button { | ||||
|       padding: 12px 32px; | ||||
|       background: $theme-primary; | ||||
|       color: #ffffff; | ||||
|       border-radius: 24px; | ||||
|       font-size: 16px; | ||||
|       cursor: pointer; | ||||
|       transition: all 0.3s ease; | ||||
| 
 | ||||
|       &:hover { | ||||
|         opacity: 0.9; | ||||
|         transform: translateY(-1px); | ||||
|       } | ||||
| 
 | ||||
|       &:active { | ||||
|         transform: translateY(0); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -1,81 +0,0 @@ | ||||
| <script setup> | ||||
| import { computed } from 'vue'; | ||||
| 
 | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="loader"> | ||||
|   <p class="heading">加载中</p> | ||||
|   <div class="loading"> | ||||
|     <div class="load"></div> | ||||
|     <div class="load"></div> | ||||
|     <div class="load"></div> | ||||
|     <div class="load"></div> | ||||
|   </div> | ||||
| </div> | ||||
| </template> | ||||
| 
 | ||||
| <style scoped lang="scss"> | ||||
| .loader { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
|   height: 100vh; | ||||
|   background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|   background-size: cover; | ||||
|   background-position: bottom center; | ||||
| } | ||||
| 
 | ||||
| .heading { | ||||
|   color: black; | ||||
|   letter-spacing: 0.2em; | ||||
|   margin-bottom: 1em; | ||||
| } | ||||
| 
 | ||||
| .loading { | ||||
|   display: flex; | ||||
|   width: 5em; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
| } | ||||
| 
 | ||||
| .load { | ||||
|   width: 23px; | ||||
|   height: 3px; | ||||
|   background-color: limegreen; | ||||
|   animation: 1s move_5011 infinite; | ||||
|   border-radius: 5px; | ||||
|   margin: 0.1em; | ||||
| } | ||||
| 
 | ||||
| .load:nth-child(1) { | ||||
|   animation-delay: 0.2s; | ||||
| } | ||||
| 
 | ||||
| .load:nth-child(2) { | ||||
|   animation-delay: 0.4s; | ||||
| } | ||||
| 
 | ||||
| .load:nth-child(3) { | ||||
|   animation-delay: 0.6s; | ||||
| } | ||||
| 
 | ||||
| @keyframes move_5011 { | ||||
|   0% { | ||||
|     width: 0.2em; | ||||
|   } | ||||
| 
 | ||||
|   25% { | ||||
|     width: 0.7em; | ||||
|   } | ||||
| 
 | ||||
|   50% { | ||||
|     width: 1.5em; | ||||
|   } | ||||
| 
 | ||||
|   100% { | ||||
|     width: 0.2em; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -89,7 +89,6 @@ const text_avatar = computed(() => { | ||||
|   img { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     object-fit: cover; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -90,6 +90,8 @@ const onSubmit = () => { | ||||
|     ServeUploadAvatar(form).then((res) => { | ||||
|       if (res.code == 200) { | ||||
|         emit('success', res.data.avatar) | ||||
|       } else { | ||||
|         message.warning(res.message) | ||||
|       } | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
| @ -19,7 +19,6 @@ | ||||
|       :disabled="props?.disabled" | ||||
|       :class="[props?.disabled ? 'custom-btn-class-disabled' : '']" | ||||
|       :plain="props?.plain" | ||||
|       :loading="props?.isLoading" | ||||
|     > | ||||
|       {{ props.btnText }} | ||||
|     </wd-button> | ||||
| @ -36,7 +35,6 @@ const props = defineProps({ | ||||
|   subBtnText: '', //次要按钮文字 | ||||
|   disabled: false, //是否禁用 | ||||
|   plain: false, //是否镂空 | ||||
|   isLoading: false, //是否正在加载中 | ||||
| }) | ||||
| 
 | ||||
| //点击副按钮 | ||||
|  | ||||
| @ -1,7 +1,7 @@ | ||||
| <template> | ||||
|   <tm-navbar | ||||
|     :hideBack="props.hideBack" | ||||
|     :hideHome="props.hideHome" | ||||
|     hideHome | ||||
|     :title="props.title" | ||||
|     :shadow="props.shadowNum" | ||||
|     :fontSize="34" | ||||
| @ -38,10 +38,6 @@ const props = defineProps({ | ||||
|     type: Number, | ||||
|     default: 1, | ||||
|   }, | ||||
|   hideHome: { | ||||
|     type: Boolean, | ||||
|     default: false, | ||||
|   }, | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
|  | ||||
| @ -25,18 +25,9 @@ | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
|             <tm-image :width="40" :height="40" :src="copy07"></tm-image> | ||||
|             <div class="mt-1">复制</div> | ||||
|             <div>复制</div> | ||||
|           </div> | ||||
|           <div | ||||
|             v-if="props.isShowConvertText" | ||||
|             @click="() => itemClick('convertText')" | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
|             <tm-image :width="40" :height="40" :src="copy07"></tm-image> | ||||
|             <div class="mt-1">转文字</div> | ||||
|           </div> | ||||
|           <div | ||||
|             v-if="props.isShowMultipleChoose && !props.isChatRobot" | ||||
|             @click="() => itemClick('multipleChoose')" | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
| @ -45,15 +36,15 @@ | ||||
|               :height="40" | ||||
|               :src="multipleChoices" | ||||
|             ></tm-image> | ||||
|             <div class="mt-1">多选</div> | ||||
|             <div>多选</div> | ||||
|           </div> | ||||
|           <div | ||||
|             v-if="props.isShowCite && !props.isChatRobot" | ||||
|             v-if="props.isShowCite" | ||||
|             @click="() => itemClick('actionCite')" | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
|             <tm-image :width="40" :height="40" :src="cite"></tm-image> | ||||
|             <div class="mt-1">引用</div> | ||||
|             <div>引用</div> | ||||
|           </div> | ||||
|           <div | ||||
|             v-if="props.isShowWithdraw" | ||||
| @ -61,15 +52,14 @@ | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
|             <tm-image :width="40" :height="40" :src="withdraw"></tm-image> | ||||
|             <div class="mt-1">撤回</div> | ||||
|             <div>撤回</div> | ||||
|           </div> | ||||
|           <div | ||||
|             :class="{ 'w-full': props.isChatRobot }" | ||||
|             @click="() => itemClick('actionDelete')" | ||||
|             class="flex flex-col items-center justify-center" | ||||
|           > | ||||
|             <tm-image :width="40" :height="40" :src="delete07"></tm-image> | ||||
|             <div class="mt-1">删除</div> | ||||
|             <div>删除</div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div :style="data.iconStyle" class="icon"></div> | ||||
| @ -88,8 +78,8 @@ | ||||
| // 组件 | ||||
| 
 | ||||
| // uniapp & vue | ||||
| import { onLoad, onReady } from '@dcloudio/uni-app' | ||||
| import { defineEmits, defineProps } from 'vue' | ||||
| import { onLoad, onReady } from "@dcloudio/uni-app"; | ||||
| import { defineEmits, defineProps } from "vue"; | ||||
| import { | ||||
|   reactive, | ||||
|   ref, | ||||
| @ -97,153 +87,135 @@ import { | ||||
|   computed, | ||||
|   nextTick, | ||||
|   getCurrentInstance, | ||||
|   onMounted, | ||||
|   onBeforeUnmount, | ||||
| } from 'vue' | ||||
| import copy07 from '@/static/image/chatList/copy07@2x.png' | ||||
| import multipleChoices from '@/static/image/chatList/multipleChoices@2x.png' | ||||
| import cite from '@/static/image/chatList/cite@2x.png' | ||||
| import withdraw from '@/static/image/chatList/withdraw@2x.png' | ||||
| import delete07 from '@/static/image/chatList/delete@2x.png' | ||||
|   onMounted,  | ||||
|   onBeforeUnmount | ||||
| } from "vue"; | ||||
| import copy07 from "@/static/image/chatList/copy07@2x.png"; | ||||
| import multipleChoices from "@/static/image/chatList/multipleChoices@2x.png"; | ||||
| import cite from "@/static/image/chatList/cite@2x.png"; | ||||
| import withdraw from "@/static/image/chatList/withdraw@2x.png"; | ||||
| import delete07 from "@/static/image/chatList/delete@2x.png"; | ||||
| 
 | ||||
| // pinia | ||||
| const systemInfo = uni.getSystemInfoSync() | ||||
| const bubbleRef = ref(null) | ||||
| const systemInfo = uni.getSystemInfoSync(); | ||||
| const bubbleRef = ref(null); | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   isShowCopy: { | ||||
|     //是否显示复制 | ||||
|     type: Boolean, | ||||
|     default: true, | ||||
|   }, | ||||
|   isShowCite: { | ||||
|     //是否显示引用 | ||||
|     type: Boolean, | ||||
|     default: true, | ||||
|   }, | ||||
|   isShowWithdraw: { | ||||
|     //是否显示撤回 | ||||
|     type: Boolean, | ||||
|     default: true, | ||||
|   }, | ||||
|   isShowConvertText: { | ||||
|     //是否显示转文字 | ||||
|     type: Boolean, | ||||
|     default: false, | ||||
|   }, | ||||
|   isShowMultipleChoose: { | ||||
|     //是否显示多选 | ||||
|     type: Boolean, | ||||
|     default: true, | ||||
|   }, | ||||
|   isChatRobot: { | ||||
|     //是否是智能助手 | ||||
|     type: Boolean, | ||||
|     default: false, | ||||
|   } | ||||
| }) | ||||
| }); | ||||
| 
 | ||||
| const emits = defineEmits(['clickMenu']) | ||||
| const emits = defineEmits(["clickMenu"]); | ||||
| 
 | ||||
| /** | ||||
|  * @name  生成UUID | ||||
|  */ | ||||
| const uuid = () => { | ||||
|   const reg = /[xy]/g | ||||
|   return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx' | ||||
|   const reg = /[xy]/g; | ||||
|   return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx" | ||||
|     .replace(reg, function (c) { | ||||
|       var r = (Math.random() * 16) | 0, | ||||
|         v = c == 'x' ? r : (r & 0x3) | 0x8 | ||||
|       return v.toString(16) | ||||
|         v = c == "x" ? r : (r & 0x3) | 0x8; | ||||
|       return v.toString(16); | ||||
|     }) | ||||
|     .replace(/-/g, '') | ||||
| } | ||||
|     .replace(/-/g, ""); | ||||
| }; | ||||
| 
 | ||||
| const popoverBoxId = `ID${uuid()}` | ||||
| const popoverContentId = `ID${uuid()}` | ||||
| const instance = getCurrentInstance() | ||||
| const popoverBoxId = `ID${uuid()}`; | ||||
| const popoverContentId = `ID${uuid()}`; | ||||
| const instance = getCurrentInstance(); | ||||
| const data = reactive({ | ||||
|   popoverShow: false, | ||||
|   defaultStyle: {}, | ||||
|   showStyle: { | ||||
|     left: 0, | ||||
|     right: '', | ||||
|     transform: '', | ||||
|     right: "", | ||||
|     transform: "", | ||||
|   }, | ||||
|   iconStyle: { | ||||
|     left: '', | ||||
|     right: '', | ||||
|     transform: '', | ||||
|     left: "", | ||||
|     right: "", | ||||
|     transform: "", | ||||
|   }, | ||||
| }) | ||||
| }); | ||||
| 
 | ||||
| /** | ||||
|  * @name  获取DOM | ||||
|  */ | ||||
| const getDom = (dom) => { | ||||
|   return new Promise((resolve, reject) => { | ||||
|     const query = uni.createSelectorQuery().in(instance) | ||||
|     let select = query.select(dom) | ||||
|     const query = uni.createSelectorQuery().in(instance); | ||||
|     let select = query.select(dom); | ||||
|     const boundingClientRect = select.boundingClientRect((data) => { | ||||
|       resolve(data) | ||||
|     }) | ||||
|     boundingClientRect.exec() | ||||
|   }) | ||||
| } | ||||
|       resolve(data); | ||||
|     }); | ||||
|     boundingClientRect.exec(); | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| const itemClick = (item) => { | ||||
|   emits('clickMenu', item) | ||||
| } | ||||
|   emits("clickMenu", item); | ||||
| }; | ||||
| 
 | ||||
| // 弹起 长按5 | ||||
| let pressDownTime = 0 | ||||
| let time = null | ||||
| let pressDownTime = 0; | ||||
| let time = null; | ||||
| const onTouchstart = () => { | ||||
|   time && clearTimeout(time) | ||||
|   time = setTimeout(open, 500) | ||||
| } | ||||
|   time && clearTimeout(time); | ||||
|   time = setTimeout(open, 500); | ||||
| }; | ||||
| 
 | ||||
| const onTouchend = () => { | ||||
|   time && clearTimeout(time) | ||||
| } | ||||
|   time && clearTimeout(time); | ||||
| }; | ||||
| 
 | ||||
| const open = async () => { | ||||
|   let popoverContent = await getDom(`#${popoverContentId}`) | ||||
|   let popoverBox = await getDom(`#${popoverBoxId}`) | ||||
|   let popoverContent = await getDom(`#${popoverContentId}`); | ||||
|   let popoverBox = await getDom(`#${popoverBoxId}`); | ||||
| 
 | ||||
|   // 缩放中心点 | ||||
|   let originX = popoverBox.width / 2 | ||||
|   let originX = popoverBox.width / 2; | ||||
|   // 根据距离 初始化位置 判断是顶部还是底部 | ||||
|   let isTop = popoverBox.top - 50 > popoverContent.height | ||||
|   let isTop = popoverBox.top - 50 > popoverContent.height; | ||||
|   // 上面还是下面 | ||||
|   data.defaultStyle = { | ||||
|     top: isTop ? '60rpx' : 'auto', | ||||
|     bottom: !isTop ? '-20rpx' : 'auto', | ||||
|     transform: `translateY(${isTop ? '-100%' : '100%'}) scale(.8)`, | ||||
|   } | ||||
|     top: isTop ? "60rpx" : "auto", | ||||
|     bottom: !isTop ? "-20rpx" : "auto", | ||||
|     transform: `translateY(${isTop ? "-100%" : "100%"}) scale(.8)`, | ||||
|   }; | ||||
|   // 左边还是右边 | ||||
|   if (popoverBox.left > systemInfo.windowWidth - popoverBox.right) { | ||||
|     data.defaultStyle.right = 0 | ||||
|     data.defaultStyle.right = 0; | ||||
|     // 动画缩放中心点 | ||||
|     data.defaultStyle['transform-origin'] = `${ | ||||
|     data.defaultStyle["transform-origin"] = `${ | ||||
|       popoverContent.width - originX | ||||
|     }px ${isTop ? '100%' : '0%'}` | ||||
|     }px ${isTop ? "100%" : "0%"}`; | ||||
|   } else { | ||||
|     data.defaultStyle.left = 0 | ||||
|     data.defaultStyle.left = 0; | ||||
|     // 动画缩放中心点 | ||||
|     data.defaultStyle['transform-origin'] = `${originX}px ${ | ||||
|       isTop ? '100%' : '0%' | ||||
|     }` | ||||
|     data.defaultStyle["transform-origin"] = `${originX}px ${ | ||||
|       isTop ? "100%" : "0%" | ||||
|     }`; | ||||
|   } | ||||
|   data.showStyle = { ...data.defaultStyle } | ||||
|   data.showStyle = { ...data.defaultStyle }; | ||||
|   // icon位置样式 | ||||
|   let iconDefsultStyle = { | ||||
|     transform: `translate(0%, ${isTop ? '20%' : '-20%'})`, | ||||
|     'border-top-color': isTop ? '#333333' : '', | ||||
|     'border-bottom-color': !isTop ? '#333333' : '', | ||||
|     top: !isTop ? '-20rpx' : 'auto', | ||||
|     bottom: isTop ? '-20rpx' : 'auto', | ||||
|   } | ||||
|     transform: `translate(0%, ${isTop ? "20%" : "-20%"})`, | ||||
|     "border-top-color": isTop ? "#333333" : "", | ||||
|     "border-bottom-color": !isTop ? "#333333" : "", | ||||
|     top: !isTop ? "-20rpx" : "auto", | ||||
|     bottom: isTop ? "-20rpx" : "auto", | ||||
|   }; | ||||
| 
 | ||||
|   setTimeout(() => { | ||||
|     if (popoverBox.left > systemInfo.windowWidth - popoverBox.right) { | ||||
| @ -252,55 +224,55 @@ const open = async () => { | ||||
|         ...data.defaultStyle, | ||||
|         // 显示 | ||||
|         opacity: 1, | ||||
|         transform: `translateY(${isTop ? '-100%' : '100%'}) scale(1)`, | ||||
|         'pointer-events': 'auto', | ||||
|       } | ||||
|         transform: `translateY(${isTop ? "-100%" : "100%"}) scale(1)`, | ||||
|         "pointer-events": "auto", | ||||
|       }; | ||||
|       data.iconStyle = { | ||||
|         right: `${originX}px`, | ||||
|         left: 'auto', | ||||
|         left: "auto", | ||||
|         ...iconDefsultStyle, | ||||
|       } | ||||
|       }; | ||||
|     } else { | ||||
|       data.showStyle = { | ||||
|         // 位置 | ||||
|         ...data.defaultStyle, | ||||
|         // 显示 | ||||
|         opacity: 1, | ||||
|         transform: `translateY(${isTop ? '-100%' : '100%'}) scale(1)`, | ||||
|         'pointer-events': 'auto', | ||||
|       } | ||||
|         transform: `translateY(${isTop ? "-100%" : "100%"}) scale(1)`, | ||||
|         "pointer-events": "auto", | ||||
|       }; | ||||
|       data.iconStyle = { | ||||
|         left: `${originX}px`, | ||||
|         right: 'auto', | ||||
|         right: "auto", | ||||
|         ...iconDefsultStyle, | ||||
|       } | ||||
|       }; | ||||
|     } | ||||
| 
 | ||||
|     if (!data.popoverShow) data.popoverShow = true | ||||
|   }, 200) | ||||
| } | ||||
|     if (!data.popoverShow) data.popoverShow = true; | ||||
|   }, 200); | ||||
| }; | ||||
| 
 | ||||
| const close = (time) => { | ||||
|   setTimeout(() => { | ||||
|     data.popoverShow = false | ||||
|     data.showStyle = data.defaultStyle | ||||
|   }, time || 0) | ||||
| } | ||||
|     data.popoverShow = false; | ||||
|     data.showStyle = data.defaultStyle; | ||||
|   }, time || 0); | ||||
| }; | ||||
| const handleClickOutside = (event) => { | ||||
|   if ((data.popoverShow = false)) { | ||||
|   if (data.popoverShow = false) { | ||||
|     return false | ||||
|   } | ||||
|   if (bubbleRef.value && !bubbleRef.value.contains(event.target)) { | ||||
|     close() | ||||
|     close(); | ||||
|   } | ||||
| } | ||||
| }; | ||||
| onMounted(() => { | ||||
|   document.addEventListener('click', handleClickOutside) | ||||
| }) | ||||
|       document.addEventListener('click', handleClickOutside); | ||||
|     }); | ||||
| 
 | ||||
| onBeforeUnmount(() => { | ||||
|   document.removeEventListener('click', handleClickOutside) | ||||
| }) | ||||
|   document.removeEventListener('click', handleClickOutside); | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
| <style lang="scss" scoped> | ||||
|  | ||||
| @ -1,155 +1,109 @@ | ||||
| <script lang="ts" setup> | ||||
| import { ref, reactive, onMounted, onUnmounted, computed } from 'vue' | ||||
| import { PlayOne, PauseOne } from '@icon-park/vue-next' | ||||
| import { ITalkRecordExtraAudio, ITalkRecord } from '@/types/chat' | ||||
| import { onHide } from '@dcloudio/uni-app' | ||||
| import aTrumpet from '@/uni_modules/a-trumpet/components/a-trumpet/a-trumpet.vue' | ||||
| import { useUserStore } from '@/store' | ||||
| import { ref, reactive, onMounted } from "vue"; | ||||
| import { PlayOne, PauseOne } from "@icon-park/vue-next"; | ||||
| import { ITalkRecordExtraAudio, ITalkRecord } from "@/types/chat"; | ||||
| 
 | ||||
| const props = defineProps<{ | ||||
|   extra: ITalkRecordExtraAudio | ||||
|   data: ITalkRecord | ||||
|   maxWidth?: Boolean | ||||
| }>() | ||||
|   extra: ITalkRecordExtraAudio; | ||||
|   data: ITalkRecord; | ||||
|   maxWidth?: Boolean; | ||||
| }>(); | ||||
| 
 | ||||
| const userStore = useUserStore() | ||||
| const talkParams = reactive({ | ||||
|   uid: computed(() => userStore.uid), | ||||
| }) | ||||
| const audioRef = ref(); | ||||
| const audioContext = ref<any>(null); | ||||
| 
 | ||||
| const durationDesc = ref("-"); | ||||
| 
 | ||||
| const audioContext = ref<any>(null) | ||||
| const durationDesc = ref('-') | ||||
| const state = reactive({ | ||||
|   isAudioPlay: false, | ||||
|   progress: 0, | ||||
|   duration: 0, | ||||
|   currentTime: 0, | ||||
|   loading: true, | ||||
| }) | ||||
| 
 | ||||
| // 监听其它音频播放事件 | ||||
| function onOtherAudioPlay(e) { | ||||
|   if (e.detail !== props.data.msg_id && state.isAudioPlay) { | ||||
|     audioContext.value.pause() | ||||
|     state.isAudioPlay = false | ||||
|   } | ||||
| } | ||||
| }); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   audioContext.value = uni.createInnerAudioContext() | ||||
|   audioContext.value.src = props.extra.url | ||||
|   // 使用uni-app的方式创建内部音频上下文 | ||||
|   audioContext.value = uni.createInnerAudioContext(); | ||||
|   audioContext.value.src = props.extra.url; | ||||
| 
 | ||||
|   audioContext.value.onCanplay(() => { | ||||
|     state.duration = audioContext.value.duration | ||||
|     durationDesc.value = formatTime(parseInt(audioContext.value.duration)) | ||||
|     state.loading = false | ||||
|   }) | ||||
|     state.duration = audioContext.value.duration; | ||||
|     durationDesc.value = formatTime(parseInt(audioContext.value.duration)); | ||||
|     state.loading = false; | ||||
|   }); | ||||
| 
 | ||||
|   audioContext.value.onTimeUpdate(() => { | ||||
|     if (audioContext.value.duration == 0) { | ||||
|       state.progress = 0 | ||||
|       state.progress = 0; | ||||
|     } else { | ||||
|       state.currentTime = audioContext.value.currentTime | ||||
|       state.currentTime = audioContext.value.currentTime; | ||||
|       state.progress = | ||||
|         (audioContext.value.currentTime / audioContext.value.duration) * 100 | ||||
|         (audioContext.value.currentTime / audioContext.value.duration) * 100; | ||||
|     } | ||||
|   }) | ||||
|   }); | ||||
| 
 | ||||
|   audioContext.value.onEnded(() => { | ||||
|     state.isAudioPlay = false | ||||
|     state.progress = 0 | ||||
|   }) | ||||
|     state.isAudioPlay = false; | ||||
|     state.progress = 0; | ||||
|   }); | ||||
| 
 | ||||
|   audioContext.value.onError((e) => { | ||||
|     console.log('音频播放异常===>', e) | ||||
|   }) | ||||
| 
 | ||||
|   window.addEventListener('audio-play', onOtherAudioPlay) | ||||
| }) | ||||
| 
 | ||||
| onUnmounted(() => { | ||||
|   window.removeEventListener('audio-play', onOtherAudioPlay) | ||||
|   audioContext.value && | ||||
|     audioContext.value.destroy && | ||||
|     audioContext.value.destroy() | ||||
| }) | ||||
| 
 | ||||
| onHide(() => { | ||||
|   if (audioContext.value && audioContext.value.pause) { | ||||
|     audioContext.value.pause() | ||||
|     state.isAudioPlay = false | ||||
|   } | ||||
| }) | ||||
|     console.log("音频播放异常===>", e); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| const onPlay = () => { | ||||
|   if (state.isAudioPlay) { | ||||
|     audioContext.value.pause() | ||||
|     state.isAudioPlay = false | ||||
|     audioContext.value.pause(); | ||||
|   } else { | ||||
|     // 通知其它音频暂停 | ||||
|     window.dispatchEvent( | ||||
|       new CustomEvent('audio-play', { detail: props.data.msg_id }), | ||||
|     ) | ||||
|     audioContext.value.play() | ||||
|     state.isAudioPlay = true | ||||
|     audioContext.value.play(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|   state.isAudioPlay = !state.isAudioPlay; | ||||
| }; | ||||
| 
 | ||||
| const onPlayEnd = () => { | ||||
|   state.isAudioPlay = false; | ||||
|   state.progress = 0; | ||||
| }; | ||||
| 
 | ||||
| const formatTime = (value: number = 0) => { | ||||
|   if (value == 0) { | ||||
|     return '-' | ||||
|     return "-"; | ||||
|   } | ||||
|   const minutes = Math.floor(value / 60) | ||||
|   let seconds = value | ||||
| 
 | ||||
|   const minutes = Math.floor(value / 60); | ||||
|   let seconds = value; | ||||
|   if (minutes > 0) { | ||||
|     seconds = Math.floor(value - minutes * 60) | ||||
|     seconds = Math.floor(value - minutes * 60); | ||||
|   } | ||||
|   return `${minutes}'${seconds}"` | ||||
| } | ||||
| 
 | ||||
|   return `${minutes}'${seconds}"`; | ||||
| }; | ||||
| </script> | ||||
| <template> | ||||
|   <div | ||||
|     class="audio-message" | ||||
|     @click.stop="onPlay" | ||||
|     :class=" | ||||
|       props?.data?.user_id == talkParams.uid | ||||
|         ? 'justify-end py-[22rpx] pl-[30rpx] pr-[16rpx]' | ||||
|         : 'justify-start py-[22rpx] pr-[30rpx] pl-[16rpx]' | ||||
|     " | ||||
|   > | ||||
|     <a-trumpet | ||||
|       v-if="props?.data?.user_id != talkParams.uid" | ||||
|       :isPlay="state.isAudioPlay" | ||||
|       color="#C1C1C1" | ||||
|       :size="30" | ||||
|     ></a-trumpet> | ||||
|     <div | ||||
|       :class=" | ||||
|         props?.data?.user_id == talkParams.uid ? 'mr-[8rpx]' : 'ml-[8rpx]' | ||||
|       " | ||||
|     > | ||||
|       {{ Math.ceil(props?.extra?.duration / 1000) }}s | ||||
|   <div class="im-message-audio"> | ||||
|     <div class="play"> | ||||
|       <div class="btn pointer" @click.stop="onPlay"> | ||||
|         <n-icon | ||||
|           :size="18" | ||||
|           :component="state.isAudioPlay ? PauseOne : PlayOne" | ||||
|         /> | ||||
|       </div> | ||||
|     </div> | ||||
|     <a-trumpet | ||||
|       v-if="props?.data?.user_id == talkParams.uid" | ||||
|       :isPlay="state.isAudioPlay" | ||||
|       color="#C1C1C1" | ||||
|       :size="30" | ||||
|       direction="left" | ||||
|     ></a-trumpet> | ||||
|     <div class="desc"> | ||||
|       <span class="line" v-for="i in 23" :key="i"></span> | ||||
|       <span | ||||
|         class="indicator" | ||||
|         :style="{ left: state.progress + '%' }" | ||||
|         v-show="state.progress > 0" | ||||
|       ></span> | ||||
|     </div> | ||||
|     <div class="time">{{ durationDesc }}</div> | ||||
|   </div> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| .audio-message { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   align-items: center; | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   background-color: #fff; | ||||
|   border-radius: 10px; | ||||
| } | ||||
| 
 | ||||
| .im-message-audio { | ||||
|   --audio-bg-color: #f5f5f5; | ||||
|   --audio-btn-bg-color: #ffffff; | ||||
| @ -291,7 +245,7 @@ const formatTime = (value: number = 0) => { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| html[theme-mode='dark'] { | ||||
| html[theme-mode="dark"] { | ||||
|   .im-message-audio { | ||||
|     --audio-bg-color: #2c2c32; | ||||
|     --audio-btn-bg-color: rgb(78, 75, 75); | ||||
|  | ||||
| @ -55,7 +55,7 @@ const getFileTypeIMG = computed(() => { | ||||
|     default: | ||||
|       objT.finishedImg = filePaperOther | ||||
|       objT.blankImg = filePaperOtherBlank | ||||
|       objT.progressColor = '#46299d' | ||||
|       objT.progressColor = '#747474' | ||||
|   } | ||||
|   return objT | ||||
| }) | ||||
| @ -132,7 +132,7 @@ const downloadAndOpenFile = () => { | ||||
|           :height="95" | ||||
|           :src="getFileTypeIMG.blankImg" | ||||
|         ></tm-image> | ||||
|         <wd-circle v-if="data.uploadStatus  === 1" | ||||
|         <wd-circle | ||||
|           customClass="circleProgress" | ||||
|           :modelValue="data.uploadCurrent" | ||||
|           layerColor="#E3E3E3" | ||||
| @ -140,9 +140,6 @@ const downloadAndOpenFile = () => { | ||||
|           :strokeWidth="3" | ||||
|           :size="20" | ||||
|         ></wd-circle> | ||||
|         <div class="upload-failed" v-if="data.uploadStatus === 3"> | ||||
|           <tm-icon :font-size="20" name="tmicon-times" color="#fff"></tm-icon> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="divider mt-[28rpx]"></div> | ||||
| @ -291,20 +288,4 @@ const downloadAndOpenFile = () => { | ||||
|   width: 40rpx !important; | ||||
|   height: 40rpx !important; | ||||
| } | ||||
| 
 | ||||
| .upload-failed { | ||||
|   position: absolute; | ||||
|   top: 120rpx; | ||||
|   right: 52rpx; | ||||
|   transform: translate(-50%, -50%); | ||||
|   z-index: 1; | ||||
|   width: 40rpx; | ||||
|   height: 40rpx; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
|   color: #ff4d4f; | ||||
|   background: #ff4d4f; | ||||
|   border-radius: 50%; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -12,25 +12,19 @@ const props = defineProps<{ | ||||
| const isShowRecord = ref(false) | ||||
| 
 | ||||
| const title = computed(() => { | ||||
|   const uniqueNames = [...new Set(props.extra.records.map(v => v.nickname))]; | ||||
|   if (uniqueNames.length <= 2) { | ||||
|     return uniqueNames.join('和'); | ||||
|   } else { | ||||
|     return uniqueNames.slice(0, 2).join('和') + '等'; | ||||
|   } | ||||
|   // return [...new Set(props.extra.records.map((v) => v.nickname))].join('和') | ||||
|   return [...new Set(props.extra.records.map((v) => v.nickname))].join('和') | ||||
| }) | ||||
| console.log(props.extra.records) | ||||
| const onClick = () => { | ||||
|   // isShowRecord.value = true | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/forwardRecord/index?msgId=' + props.data.msg_id + '&created_at=' + props.data?.created_at | ||||
|     url: '/pages/forwardRecord/index?msgId=' + props.data.msg_id | ||||
|   }) | ||||
| } | ||||
| </script> | ||||
| <template> | ||||
|   <section class="im-message-forward pointer" :class="{ left: data.float === 'left' }" @click="onClick"> | ||||
|     <div class="title">{{ extra.forward_name || title}}的会话记录</div> | ||||
|     <div class="title">{{ title }}<span v-if="props.extra.records.length > 2">等</span>的会话记录</div> | ||||
|     <div class="list" v-for="(record, index) in extra.records" :key="index"> | ||||
|       <p> | ||||
|         <span>{{ record.nickname }}: </span> | ||||
| @ -65,14 +59,13 @@ const onClick = () => { | ||||
|   } | ||||
| 
 | ||||
|   .title { | ||||
|     max-height: 88rpx; | ||||
|     height: 44rpx; | ||||
|     line-height: 44rpx; | ||||
|     font-size: 32rpx; | ||||
|     color: #1A1A1A; | ||||
|     display: -webkit-box; | ||||
|     -webkit-box-orient: vertical; | ||||
|     -webkit-line-clamp: 2; | ||||
|     overflow: hidden; | ||||
|     text-overflow: ellipsis; | ||||
|     white-space: nowrap; | ||||
|     font-weight: 400; | ||||
|     margin-bottom: 8rpx; | ||||
|   } | ||||
|  | ||||
| @ -12,40 +12,40 @@ const img = computed(() => { | ||||
|   // console.log(props.extra); | ||||
|   let info = { | ||||
|     width: 0, | ||||
|     height: 0, | ||||
|     height: 0 | ||||
|   } | ||||
|   if (props.extra.url.includes('blob:http://')) { | ||||
|     info = { | ||||
|       width: props.extra.width, | ||||
|       height: props.extra.height, | ||||
|       height: props.extra.height | ||||
|     } | ||||
|   } else { | ||||
|   }else { | ||||
|     info = getImageInfo(props.extra.url) | ||||
|   } | ||||
| 
 | ||||
|   if (info.width == 0 || info.height == 0) { | ||||
|     return { | ||||
|       width: 450, | ||||
|       height: 298, | ||||
|       height: 298 | ||||
|     } | ||||
|   } | ||||
|   if (info.width < 300) { | ||||
|   if(info.width<300){ | ||||
|     return { | ||||
|       width: 300, | ||||
|       height: info.height / (info.width / 300), | ||||
|       height: info.height / (info.width / 300) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (info.width < 350) { | ||||
|     return { | ||||
|       width: info.width, | ||||
|       height: info.height, | ||||
|       height: info.height | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     width: 350, | ||||
|     height: info.height / (info.width / 350), | ||||
|     height: info.height / (info.width / 350) | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
| @ -54,30 +54,13 @@ const img = computed(() => { | ||||
|     class="im-message-image" | ||||
|     :class="{ | ||||
|       left: data.float === 'left', | ||||
|       right: data.float === 'right', | ||||
|       right: data.float === 'right' | ||||
|     }" | ||||
|   > | ||||
|     <div class="image-container"> | ||||
|       <div class="relative"> | ||||
|         <tm-image | ||||
|           preview | ||||
|           :width="img.width" | ||||
|           :height="img.height" | ||||
|           :src="extra.url" | ||||
|           model="aspectFill" | ||||
|         /> | ||||
|         <wd-circle | ||||
|           custom-class="circleProgress" | ||||
|           v-if="data.uploadStatus  === 1" | ||||
|           v-model="props.data.uploadCurrent" | ||||
|           color="#46299d" | ||||
|           layer-color="#E3E3E3" | ||||
|         ></wd-circle> | ||||
|         <div class="upload-failed" v-if="data.uploadStatus === 3"> | ||||
|           <tm-icon :font-size="20" name="tmicon-times" color="#fff"></tm-icon> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   <div class="image-container"> | ||||
|     <tm-image preview :width="img.width" :height="img.height" :src="extra.url" /> | ||||
|     <wd-circle custom-class="circleProgress" v-if="props.data.uploadCurrent && props.data.uploadCurrent<100"  v-model="props.data.uploadCurrent" color="#ffffff" layer-color="#E3E3E3"></wd-circle> | ||||
|   </div> | ||||
|   </section> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| @ -96,12 +79,12 @@ const img = computed(() => { | ||||
|   } | ||||
| 
 | ||||
|   &.right { | ||||
|     background-color: #46299d; | ||||
|     background-color: #46299D; | ||||
|     border-radius: 16rpx 0 16rpx 16rpx; | ||||
|   } | ||||
| } | ||||
| .image-container { | ||||
|   position: relative; | ||||
|     position: relative; | ||||
| 
 | ||||
|   .circleProgress { | ||||
|     position: absolute; | ||||
| @ -111,18 +94,4 @@ const img = computed(() => { | ||||
|     z-index: 1; | ||||
|   } | ||||
| } | ||||
| .upload-failed { | ||||
|   position: absolute; | ||||
|   top: 50%; | ||||
|   left: 50%; | ||||
|   transform: translate(-50%, -50%); | ||||
|   z-index: 1; | ||||
|   width: 40rpx; | ||||
|   height: 40rpx; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
|   background: #ff4d4f; | ||||
|   border-radius: 50%; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -66,6 +66,7 @@ defineProps({ | ||||
|       <span v-if="talk_type === 2 && login_uid !== revokeInfo.withdraw_id && login_uid === revokeInfo.retracted_id && revokeInfo.withdraw_id !== revokeInfo.retracted_id"> | ||||
|         {{revokeInfo.withdraw_name}}撤回了你一条消息 | | ||||
|         {{ formatTime(datetime) }} | ||||
|         {{extra}} | ||||
|       </span> | ||||
|       <span v-if="talk_type === 2 && login_uid !== revokeInfo.withdraw_id && login_uid !== revokeInfo.retracted_id && revokeInfo.withdraw_id !== revokeInfo.retracted_id"> | ||||
|         {{revokeInfo.withdraw_name}}撤回了{{revokeInfo.retracted_name}}一条消息 | | ||||
|  | ||||
| @ -2,7 +2,6 @@ | ||||
| import { textReplaceEmoji } from '@/utils/emojis' | ||||
| import { textReplaceLink, textReplaceMention } from '@/utils/strings' | ||||
| import { ITalkRecordExtraText, ITalkRecord } from '@/types/chat' | ||||
| import { computed } from 'vue' | ||||
| 
 | ||||
| const props = defineProps<{ | ||||
|   extra: ITalkRecordExtraText | ||||
| @ -13,14 +12,15 @@ const props = defineProps<{ | ||||
| 
 | ||||
| const float = props.data.float | ||||
| 
 | ||||
| const textContent = computed(() => { | ||||
|   let text = props.extra?.content || '' | ||||
|   // text = textReplaceLink(text) | ||||
|   if (props.data.talk_type == 2) { | ||||
|     text = textReplaceMention(text, '#1890ff') | ||||
|   } | ||||
|   return textReplaceEmoji(text) | ||||
| }) | ||||
| let textContent = props.extra?.content || '' | ||||
| 
 | ||||
| textContent = textReplaceLink(textContent) | ||||
| 
 | ||||
| if (props.data.talk_type == 2) { | ||||
|   textContent = textReplaceMention(textContent, '#1890ff') | ||||
| } | ||||
| 
 | ||||
| textContent = textReplaceEmoji(textContent) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|  | ||||
| @ -1,8 +1,8 @@ | ||||
| <script lang="ts" setup> | ||||
| import { ref, nextTick, getCurrentInstance, computed, onMounted } from 'vue' | ||||
| import { getImageInfo } from '@/utils/functions' | ||||
| import playCircle from '@/static/image/chatList/playCircle@2x.png' | ||||
| import { useStatus } from '@/store/status' | ||||
| import playCircle from "@/static/image/chatList/playCircle@2x.png"; | ||||
| import { useStatus } from "@/store/status"; | ||||
| 
 | ||||
| const { statusBarHeight } = useStatus() | ||||
| const instance = getCurrentInstance() | ||||
| @ -20,12 +20,12 @@ const open = ref(false) | ||||
| const img = computed(() => { | ||||
|   let info = { | ||||
|     width: 0, | ||||
|     height: 0, | ||||
|     height: 0 | ||||
|   } | ||||
|   if (props.extra.url.includes('blob:http://')) { | ||||
|     info = { | ||||
|       width: props.extra.width, | ||||
|       height: props.extra.height, | ||||
|       height: props.extra.height | ||||
|     } | ||||
|   } else { | ||||
|     info = getImageInfo(props.extra.url) | ||||
| @ -34,26 +34,26 @@ const img = computed(() => { | ||||
|   if (info.width == 0 || info.height == 0) { | ||||
|     return { | ||||
|       width: 450, | ||||
|       height: 298, | ||||
|       height: 298 | ||||
|     } | ||||
|   } | ||||
|   if (info.width < 300) { | ||||
|     return { | ||||
|       width: 300, | ||||
|       height: info.height / (info.width / 300), | ||||
|       height: info.height / (info.width / 300) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (info.width < 350) { | ||||
|     return { | ||||
|       width: info.width, | ||||
|       height: info.height, | ||||
|       height: info.height | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     width: 350, | ||||
|     height: info.height / (info.width / 350), | ||||
|     height: info.height / (info.width / 350) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| @ -67,98 +67,57 @@ const fullscreenchange = (e) => { | ||||
| 
 | ||||
| /* 视频播放 获取第一帧 */ | ||||
| const canplay = (e) => { | ||||
|   console.log('Video can play:', e) | ||||
|   console.log('Video can play:', e); | ||||
| 
 | ||||
|   if (e.target) { | ||||
|     setTimeout(() => { | ||||
|       e.target.pause() | ||||
|     }, 200) | ||||
|       e.target.pause(); | ||||
|     }, 200); | ||||
|   } | ||||
| } | ||||
| }; | ||||
| 
 | ||||
| async function onPlay() { | ||||
|   videoContext.value = uni.createVideoContext(props.extra.url, instance); | ||||
|   videoContext.value.requestFullScreen({ direction: 2 }); | ||||
|   videoContext.value.play() | ||||
|   open.value = true | ||||
|   await nextTick() | ||||
|   videoContext.value = uni.createVideoContext(props.extra.url, instance) | ||||
| 
 | ||||
|   setTimeout(() => { | ||||
|     // 先请求全屏 | ||||
|     videoContext.value.requestFullScreen({ direction: 2 }) | ||||
| 
 | ||||
|     // 延迟一下再播放,确保全屏已经完成 | ||||
|     setTimeout(() => { | ||||
|       videoContext.value.play() | ||||
|     }, 100) | ||||
|   }, 200) | ||||
| } | ||||
| onMounted(() => { | ||||
|   videoRef.value = uni.createVideoContext(props.data.msg_id) | ||||
|   videoRef.value = uni.createVideoContext(props.data.msg_id); | ||||
|   videoRef.value.play() | ||||
|   setTimeout(() => { | ||||
|     videoRef.value.pause() | ||||
|   }, 200) | ||||
|   }, 200); | ||||
| 
 | ||||
| }) | ||||
| </script> | ||||
| <template> | ||||
|   <section | ||||
|     class="im-message-video" | ||||
|     :class="{ left: data.float === 'left' }" | ||||
|     @click="onPlay" | ||||
|   > | ||||
|     <div | ||||
|       class="coverVideo" | ||||
|       :style="{ | ||||
|         width: img.width + 'rpx', | ||||
|         height: img.height + 'rpx', | ||||
|       }" | ||||
|       v-if="props.extra.url.includes('blob:http://')" | ||||
|     > | ||||
|       <video | ||||
|         :id="data.msg_id" | ||||
|         :autoplay="false" | ||||
|         disablepictureinpicture | ||||
|         muted | ||||
|         :src="props.extra.url" | ||||
|         width="100%" | ||||
|         height="100%" | ||||
|         playsinline | ||||
|         preload="auto" | ||||
|         controls="false" | ||||
|         x5-playsinline | ||||
|         webkit-playsinline | ||||
|         style="object-fit: cover; pointer-events: none;" | ||||
|       ></video> | ||||
|   <section class="im-message-video" :class="{ left: data.float === 'left' }" @click="onPlay"> | ||||
|     <div class="coverVideo" :style="{ | ||||
|       width: img.width + 'rpx', | ||||
|       height: img.height + 'rpx' | ||||
|     }" v-if="props.extra.url.includes('blob:http://')"> | ||||
|       <video :id="data.msg_id" :autoplay="false" disablepictureinpicture muted :src="props.extra.url" width="100%" | ||||
|         height="100%" playsinline preload="auto" controls="false" x5-playsinline | ||||
|         webkit-playsinline style="object-fit: cover; pointer-events: none;"> | ||||
|       </video> | ||||
|     </div> | ||||
|     <wd-img | ||||
|       v-else | ||||
|       :width="`${img.width}rpx`" | ||||
|       :height="`${img.height}rpx`" | ||||
|       :src="data.extra.cover" | ||||
|     /> | ||||
|     <div | ||||
|       v-if="data.uploadStatus === 2 || !data.uploadStatus" | ||||
|       class="btn-video" | ||||
|       :style="{ | ||||
|         width: img.width + 'rpx', | ||||
|         height: img.height + 'rpx', | ||||
|       }" | ||||
|     > | ||||
|     <wd-img v-else :width="`${img.width}rpx`" :height="`${img.height}rpx`" :src="data.extra.cover" /> | ||||
|     <div v-if="data.uploadStatus === 2 || !data.uploadStatus" class="btn-video" :style="{ | ||||
|       width: img.width + 'rpx', | ||||
|       height: img.height + 'rpx' | ||||
|     }"> | ||||
|       <tm-image :src="playCircle" :width="80" :height="80" /> | ||||
|     </div> | ||||
|     <div | ||||
|       v-else | ||||
|       class="btn-video" | ||||
|       :style="{ | ||||
|         width: img.width + 'rpx', | ||||
|         height: img.height + 'rpx', | ||||
|       }" | ||||
|     > | ||||
|     <div v-else class="btn-video" :style="{ | ||||
|       width: img.width + 'rpx', | ||||
|       height: img.height + 'rpx' | ||||
|     }" > | ||||
|       <wd-circle | ||||
|         v-if="data.uploadStatus === 1" | ||||
|         v-model="props.data.uploadCurrent" | ||||
|         customClass="circleProgress" | ||||
|         color="#46299d" | ||||
|         layer-color="#E3E3E3" | ||||
|         layerColor="#E3E3E3" | ||||
|         color="#FFFFFF" | ||||
|         :strokeWidth="6" | ||||
|         :size="40" | ||||
|       ></wd-circle> | ||||
| @ -172,32 +131,18 @@ onMounted(() => { | ||||
|         :width="70" | ||||
|         :percent="props.data.uploadCurrent"> | ||||
|       </tm-progress> --> | ||||
|       <div class="upload-failed" v-if="data.uploadStatus === 3"> | ||||
|         <tm-icon :font-size="20" name="tmicon-times" color="#fff"></tm-icon> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div v-show="open"> | ||||
|       <video :src="props.extra.url" controls @fullscreenchange="fullscreenchange" :id="props.extra.url"> | ||||
|       </video> | ||||
|     </div> | ||||
|   </section> | ||||
| 
 | ||||
|   <teleport to="body"> | ||||
|     <div v-show="open" class="video-container"> | ||||
|       <video | ||||
|         :src="props.extra.url" | ||||
|         controls | ||||
|         @fullscreenchange="fullscreenchange" | ||||
|         :id="props.extra.url" | ||||
|         playsinline | ||||
|         webkit-playsinline | ||||
|         x5-playsinline | ||||
|         class="fullscreen-video" | ||||
|       ></video> | ||||
|     </div> | ||||
|   </teleport> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| .im-message-video { | ||||
|   overflow: hidden; | ||||
|   padding: 20rpx 18rpx; | ||||
|   background: #46299d; | ||||
|   background: #46299D; | ||||
|   min-width: 30rpx; | ||||
|   min-height: 30rpx; | ||||
|   display: inline-flex; | ||||
| @ -255,43 +200,10 @@ onMounted(() => { | ||||
|   :deep(.uni-video-bar) { | ||||
|     display: none; | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| .circleProgress { | ||||
|   width: 80rpx !important; | ||||
|   height: 80rpx !important; | ||||
| } | ||||
| 
 | ||||
| .video-container { | ||||
|   position: fixed; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|   width: 100vw; | ||||
|   height: 100vh; | ||||
|   background: #000; | ||||
|   z-index: 9999; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
| } | ||||
| 
 | ||||
| .fullscreen-video { | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   object-fit: contain; | ||||
| } | ||||
| 
 | ||||
| .upload-failed { | ||||
|   position: absolute; | ||||
|   top: 50%; | ||||
|   left: 50%; | ||||
|   transform: translate(-50%, -50%); | ||||
|   z-index: 1; | ||||
|   width: 40rpx; | ||||
|   height: 40rpx; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
|   background: #ff4d4f; | ||||
|   border-radius: 50%; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -1,19 +0,0 @@ | ||||
| <script setup> | ||||
| import './sys-message.less' | ||||
| import { useInject } from '@/hooks' | ||||
| 
 | ||||
| const { showUserInfoModal } = useInject() | ||||
| 
 | ||||
| defineProps({ | ||||
|   extra: Object, | ||||
|   data: Object | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="im-message-sys-text"> | ||||
|     <div class="sys-text"> | ||||
|       <span>{{ extra.content }}</span> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| @ -1,25 +0,0 @@ | ||||
| <script setup> | ||||
| import './sys-message.less' | ||||
| import { useInject } from '@/hooks' | ||||
| 
 | ||||
| const { showUserInfoModal } = useInject() | ||||
| 
 | ||||
| defineProps({ | ||||
|   extra: Object, | ||||
|   data: Object | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="im-message-sys-text"> | ||||
|     <div class="sys-text"> | ||||
|       <a @click="showUserInfoModal(data.user_id)"> | ||||
|         <!-- {{ data.nickname }} --> | ||||
|           管理员 | ||||
|       </a> | ||||
|       <!-- <span>修改群名为</span> | ||||
|       <span>"{{ extra.group_name }}"</span> --> | ||||
|       <span>修改了群信息</span> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| @ -1,23 +0,0 @@ | ||||
| <script setup> | ||||
| import './sys-message.less' | ||||
| import { useInject } from '@/hooks' | ||||
| 
 | ||||
| const { showUserInfoModal } = useInject() | ||||
| 
 | ||||
| defineProps({ | ||||
|   extra: Object, | ||||
|   data: Object, | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="im-message-sys-text"> | ||||
|     <div class="sys-text"> | ||||
|       <template v-for="(user, index) in extra?.members" :key="index"> | ||||
|         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||
|         <em v-show="index < extra.members.length - 1">、</em> | ||||
|       </template> | ||||
|       <span>已离开此群</span> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| @ -1,153 +0,0 @@ | ||||
| <script setup> | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| import noClockAfterWork from '@/static/image/chatBotMessageCard/noClockAfterWork.png' | ||||
| import noClockBeforeWork from '@/static/image/chatBotMessageCard/noClockBeforeWork.png' | ||||
| import copy from '@/static/image/chatBotMessageCard/copy.png' | ||||
| import errorClock from '@/static/image/chatBotMessageCard/errorClock.png' | ||||
| import prompt from '@/static/image/chatBotMessageCard/prompt.png' | ||||
| 
 | ||||
| const emits = defineEmits(['openPromptDrawer']) | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   extra: Object, | ||||
|   data: Object, | ||||
| }) | ||||
| 
 | ||||
| //聊天助手发送的系统消息卡片图片 | ||||
| const getMessageCard = computed(() => { | ||||
|   switch (props.data.msg_type) { | ||||
|     case 1117: //催办提醒 | ||||
|       return prompt | ||||
|     case 1118: //上班未打卡 | ||||
|       return noClockBeforeWork | ||||
|     case 1119: //下班未打卡 | ||||
|       return noClockAfterWork | ||||
|     case 1120: //待审批 | ||||
|       return prompt | ||||
|     case 1121: //抄送 | ||||
|       return copy | ||||
|     case 1122: //缺卡 | ||||
|       return errorClock | ||||
|     case 1123: //迟到 | ||||
|       return errorClock | ||||
|     case 1124: //早退 | ||||
|       return errorClock | ||||
|     default: | ||||
|       return '' | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| //聊天助手发送的系统消息文字颜色 | ||||
| const getTextColor = computed(() => { | ||||
|   const msgType = props.data.msg_type | ||||
|   //审批催办、审批 | ||||
|   if ([1117, 1120].includes(msgType)) return '#2668BF' | ||||
|   //上班未打卡、下班未打卡 | ||||
|   if ([1118, 1119].includes(msgType)) return '#46299D' | ||||
|   //抄送 | ||||
|   if (msgType === 1121) return '#2099BE' | ||||
|   //缺卡、迟到、早退 | ||||
|   if ([1122, 1123, 1124].includes(msgType)) return '#933BA3' | ||||
|   return '' | ||||
| }) | ||||
| 
 | ||||
| //聊天助手发送的系统消息文字内容 | ||||
| const getTextContent = computed(() => { | ||||
|   const msgType = props.data.msg_type | ||||
|   //审批催办、审批 | ||||
|   if ([1117, 1120].includes(msgType)) return '立即审批' | ||||
|   //上班未打卡、下班未打卡 | ||||
|   if ([1118, 1119].includes(msgType)) return '立即打卡' | ||||
|   //抄送、缺卡、迟到、早退 | ||||
|   if ([1121, 1122, 1123, 1124].includes(msgType)) return '立即查看' | ||||
|   return '' | ||||
| }) | ||||
| 
 | ||||
| //点击向webview通信,处理智能助手系统消息 | ||||
| const handleSysMessagePush = () => { | ||||
|   if (props?.extra?.url) { | ||||
|     //直接返回地址的,直接在当前页面弹出抽屉打开 | ||||
|     emits('openPromptDrawer', encodeURIComponent(props?.extra?.url)) | ||||
|   } else { | ||||
|     handleFindWebview( | ||||
|       `handleChatRobotSysMessagePush('${encodeURIComponent( | ||||
|         JSON.stringify({ | ||||
|           url: props?.extra?.url, | ||||
|           msg_type: props?.data?.msg_type, | ||||
|         }), | ||||
|       )}')`, | ||||
|     ) | ||||
|     uni.reLaunch({ | ||||
|       url: '/pages/index/index', | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="sys-message-push" @click="handleSysMessagePush"> | ||||
|     <div class="sys-message-push-card"> | ||||
|       <span class="sys-message-push-card-title">{{ props.extra.title }}</span> | ||||
|       <span | ||||
|         class="sys-message-push-card-message" | ||||
|         v-html="props.extra.message" | ||||
|       ></span> | ||||
|       <img :src="getMessageCard" alt="" /> | ||||
|     </div> | ||||
|     <div class="sys-message-push-card-btn"> | ||||
|       <span :style="{ color: getTextColor }">{{ getTextContent }}</span> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .sys-message-push { | ||||
|   border-radius: 0 16rpx 16rpx 16rpx; | ||||
|   overflow: hidden; | ||||
|   box-shadow: 0 6px 12px 2px rgba(188, 188, 188, 0.08); | ||||
|   .sys-message-push-card { | ||||
|     position: relative; | ||||
|     width: 486rpx; | ||||
|     height: 270rpx; | ||||
|     .sys-message-push-card-title { | ||||
|       position: absolute; | ||||
|       top: 28rpx; | ||||
|       left: 30rpx; | ||||
|       font-size: 24rpx; | ||||
|       line-height: 34rpx; | ||||
|       color: rgba(255, 255, 255, 0.73); | ||||
|       font-weight: 500; | ||||
|     } | ||||
|     .sys-message-push-card-message { | ||||
|       position: absolute; | ||||
|       bottom: 44rpx; | ||||
|       left: 30rpx; | ||||
|       font-size: 40rpx; | ||||
|       line-height: 58rpx; | ||||
|       color: #fff; | ||||
|       font-weight: bold; | ||||
|       min-width: 326rpx; | ||||
|       max-width: 100%; | ||||
|     } | ||||
|     img { | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|     } | ||||
|   } | ||||
|   .sys-message-push-card-btn { | ||||
|     background-color: #fff; | ||||
|     width: 486rpx; | ||||
|     height: 78rpx; | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     justify-content: center; | ||||
|     align-items: center; | ||||
|     text { | ||||
|       color: #000; | ||||
|       font-size: 28rpx; | ||||
|       line-height: 40rpx; | ||||
|       font-weight: 500; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -17,7 +17,7 @@ export const ChatMsgSysText = 1000 // 系统文本消息 | ||||
| export const ChatMsgSysGroupCreate = 1101 // 创建群聊消息
 | ||||
| export const ChatMsgSysGroupMemberJoin = 1102 // 加入群聊消息
 | ||||
| export const ChatMsgSysGroupMemberQuit = 1103 // 群成员退出群消息
 | ||||
| export const ChatMsgSysGroupMemberKicked = 1104 // 移出群成员消息(普通群、项目群被踢)
 | ||||
| export const ChatMsgSysGroupMemberKicked = 1104 // 移出群成员消息
 | ||||
| export const ChatMsgSysGroupMessageRevoke = 1105 // 管理员撤回成员消息
 | ||||
| export const ChatMsgSysGroupDismissed = 1106 // 群解散
 | ||||
| export const ChatMsgSysGroupMuted = 1107 // 群禁言
 | ||||
| @ -27,16 +27,6 @@ export const ChatMsgSysGroupMemberCancelMuted = 1110 // 群成员解除禁言 | ||||
| export const ChatMsgSysGroupNotice = 1111 // 编辑群公告
 | ||||
| export const ChatMsgSysGroupTransfer = 1113 // 变更群主
 | ||||
| export const ChatMsgSysGroupAdmin = 1114 // 设置管理员
 | ||||
| export const ChatMsgSysGroupMemberRemoved = 1115 // 移出群成员消息(部门群、公司群自动移出)
 | ||||
| export const ChatMsgSysGroupInfoChange = 1116 // 管理员更新了群信息
 | ||||
| export const ChatMsgSysPush_PromptReminder = 1117 // 系统推送消息-催办提醒
 | ||||
| export const ChatMsgSysPush_NoClockReminderBeforeWork = 1118 // 系统推送消息-上班未打卡
 | ||||
| export const ChatMsgSysPush_NoClockReminderAfterWork = 1119 // 系统推送消息-下班未打卡
 | ||||
| export const ChatMsgSysPush_ApprovalReminder = 1120 // 系统推送消息-待审批提醒
 | ||||
| export const ChatMsgSysPush_CopyReminder = 1121 // 系统推送消息-抄送提醒
 | ||||
| export const ChatMsgSysPush_AbsentReminder = 1122 // 系统推送消息-缺卡提醒
 | ||||
| export const ChatMsgSysPush_LateReminder = 1123 // 系统推送消息-迟到提醒
 | ||||
| export const ChatMsgSysPush_EarlyReminder = 1124 // 系统推送消息-早退提醒
 | ||||
| 
 | ||||
| export const ChatMsgTypeMapping = { | ||||
|   [ChatMsgTypeText]: '[文本消息]', | ||||
| @ -66,17 +56,7 @@ export const ChatMsgTypeMapping = { | ||||
|   [ChatMsgSysGroupMemberCancelMuted]: '[群成员解除禁言消息]', | ||||
|   [ChatMsgSysGroupNotice]: '[群公告]', | ||||
|   [ChatMsgSysGroupTransfer]: '[转让群主]', | ||||
|   [ChatMsgSysGroupAdmin]: '[设置管理员]', | ||||
|   [ChatMsgSysGroupMemberRemoved]: '[移出群成员消息]', | ||||
|   [ChatMsgSysGroupInfoChange]: '[群信息更新]', | ||||
|   [ChatMsgSysPush_PromptReminder]: '[催办提醒]', | ||||
|   [ChatMsgSysPush_NoClockReminderBeforeWork]: '[未打卡提醒]', | ||||
|   [ChatMsgSysPush_NoClockReminderAfterWork]: '[未打卡提醒]', | ||||
|   [ChatMsgSysPush_ApprovalReminder]: '[审批提醒]', | ||||
|   [ChatMsgSysPush_CopyReminder]: '[抄送提醒]', | ||||
|   [ChatMsgSysPush_AbsentReminder]: '[异常卡提醒]', | ||||
|   [ChatMsgSysPush_LateReminder]: '[异常卡提醒]', | ||||
|   [ChatMsgSysPush_EarlyReminder]: '[异常卡提醒]' | ||||
|   [ChatMsgSysGroupAdmin]: '[设置管理员]' | ||||
| } | ||||
| 
 | ||||
| // 消息类型 - 消息组件 映射关系
 | ||||
| @ -101,24 +81,13 @@ export const MessageComponents = { | ||||
|   [ChatMsgSysGroupMemberQuit]: 'sys-group-member-quit-message', | ||||
|   [ChatMsgSysGroupMemberKicked]: 'sys-group-member-kicked-message', | ||||
|   // [ChatMsgSysGroupMessageRevoke]: '[撤回消息]',
 | ||||
|   [ChatMsgSysGroupDismissed]: 'sys-group-dismissed', | ||||
|   // [ChatMsgSysGroupDismissed]: '[群解散消息]',
 | ||||
|   [ChatMsgSysGroupMuted]: 'sys-group-muted-message', | ||||
|   [ChatMsgSysGroupCancelMuted]: 'sys-group-cancel-muted-message', | ||||
|   [ChatMsgSysGroupMemberMuted]: 'sys-group-member-muted-message', | ||||
|   [ChatMsgSysGroupMemberCancelMuted]: 'sys-group-member-cancel-muted-message', | ||||
|   [ChatMsgSysGroupTransfer]: 'sys-group-transfer-message', | ||||
|   [ChatMsgSysGroupAdmin]:'sys-group-admin-message', | ||||
|   [ChatMsgSysGroupMemberRemoved]:'sys-group-member-removed-message', | ||||
|   [ChatMsgSysGroupInfoChange]:'sys-group-info-change-message', | ||||
|   // 智能助手推送的系统消息采用相同的模版
 | ||||
|   [ChatMsgSysPush_PromptReminder]:'sys-push-message', | ||||
|   [ChatMsgSysPush_NoClockReminderBeforeWork]:'sys-push-message', | ||||
|   [ChatMsgSysPush_NoClockReminderAfterWork]:'sys-push-message', | ||||
|   [ChatMsgSysPush_ApprovalReminder]:'sys-push-message', | ||||
|   [ChatMsgSysPush_CopyReminder]:'sys-push-message', | ||||
|   [ChatMsgSysPush_AbsentReminder]:'sys-push-message', | ||||
|   [ChatMsgSysPush_LateReminder]:'sys-push-message', | ||||
|   [ChatMsgSysPush_EarlyReminder]:'sys-push-message' | ||||
|   [ChatMsgSysGroupAdmin]:'sys-group-admin-message' | ||||
| } | ||||
| 
 | ||||
| // 可转发的消息类型
 | ||||
| @ -126,11 +95,10 @@ export const ForwardableMessageType = [ | ||||
|   ChatMsgTypeText, | ||||
|   ChatMsgTypeCode, | ||||
|   ChatMsgTypeImage, | ||||
|   // ChatMsgTypeAudio,
 | ||||
|   ChatMsgTypeAudio, | ||||
|   ChatMsgTypeVideo, | ||||
|   ChatMsgTypeFile, | ||||
|   ChatMsgTypeLocation, | ||||
|   ChatMsgTypeCard, | ||||
|   ChatMsgTypeLink, | ||||
|   ChatMsgTypeForward | ||||
|   ChatMsgTypeLink | ||||
| ] | ||||
|  | ||||
| @ -5,17 +5,8 @@ import { parseTime } from '@/utils/datetime' | ||||
| import * as message from '@/constant/message' | ||||
| import { formatTalkItem, palyMusic, formatTalkRecord } from '@/utils/talk' | ||||
| // import { isElectronMode } from '@/utils/common'
 | ||||
| import { | ||||
|   ServeClearTalkUnreadNum, | ||||
|   ServeCreateTalkList, | ||||
| } from '@/api/chat/index.js' | ||||
| import { | ||||
|   useTalkStore, | ||||
|   useDialogueStore, | ||||
|   useDialogueListStore, | ||||
|   useGroupStore, | ||||
| } from '@/store' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| import { ServeClearTalkUnreadNum, ServeCreateTalkList } from '@/api/chat/index.js' | ||||
| import { useTalkStore, useDialogueStore,useDialogueListStore } from '@/store' | ||||
| 
 | ||||
| /** | ||||
|  * 好友状态事件 | ||||
| @ -58,7 +49,7 @@ class Talk extends Base { | ||||
|     this.receiver_id = resource.receiver_id | ||||
|     this.talk_type = resource.talk_type | ||||
|     // this.fileNum = resource.file_num
 | ||||
|     if (resource.file_num) { | ||||
|     if(resource.file_num){ | ||||
|       resource.data.file_num = resource.file_num | ||||
|     } | ||||
|     this.resource = resource.data | ||||
| @ -94,11 +85,7 @@ class Talk extends Base { | ||||
|    */ | ||||
|   getTalkText() { | ||||
|     let text = '' | ||||
|     if ( | ||||
|       this.resource.msg_type != message.ChatMsgTypeText && | ||||
|       //智能助手发送的系统消息,也直接显示内容
 | ||||
|       this.resource.user_id !== 2 | ||||
|     ) { | ||||
|     if (this.resource.msg_type != message.ChatMsgTypeText) { | ||||
|       text = message.ChatMsgTypeMapping[this.resource.msg_type] | ||||
|     } else { | ||||
|       text = this.resource.extra.content.replace(/<img .*?>/g, '') | ||||
| @ -111,10 +98,11 @@ class Talk extends Base { | ||||
|   play() { | ||||
|     // 客户端有消息提示
 | ||||
|     // if (isElectronMode()) return
 | ||||
| 
 | ||||
|     // useSettingsStore().isPromptTone && palyMusic()
 | ||||
|   } | ||||
| 
 | ||||
|   async handle() { | ||||
|   handle() { | ||||
|     // 不是自己发送的消息则需要播放提示音
 | ||||
|     if (!this.isCurrSender()) { | ||||
|       this.play() | ||||
| @ -122,21 +110,7 @@ class Talk extends Base { | ||||
| 
 | ||||
|     // 判断会话列表是否存在,不存在则创建
 | ||||
|     if (useTalkStore().findTalkIndex(this.getIndexName()) == -1) { | ||||
|       if (this.resource.msg_type == 1102) { | ||||
|         //被邀请进入群聊时,需要热更新会话列表
 | ||||
|         await useTalkStore().loadTalkList() | ||||
|       } else if (this.resource.msg_type == 1106) { | ||||
|         //群解散时,需要热更新会话列表
 | ||||
|         await useTalkStore().loadTalkList() | ||||
|       } else if ( | ||||
|         this.resource.msg_type == 1104 || | ||||
|         this.resource.msg_type == 1115 | ||||
|       ) { | ||||
|         //群成员被移出时,需要热更新会话列表
 | ||||
|         await useTalkStore().loadTalkList() | ||||
|       } else { | ||||
|         return this.addTalkItem() | ||||
|       } | ||||
|       return this.addTalkItem() | ||||
|     } | ||||
| 
 | ||||
|     // 判断当前是否正在和好友对话
 | ||||
| @ -144,24 +118,9 @@ class Talk extends Base { | ||||
|       this.insertTalkRecord() | ||||
|     } else { | ||||
|       this.updateTalkItem() | ||||
|       if ( | ||||
|         !useTalkStore().items[useTalkStore().findTalkIndex(this.getIndexName())] | ||||
|           ?.is_disturb && | ||||
|         !( | ||||
|           useTalkStore().findTalkIndex(this.getIndexName()) == -1 && | ||||
|           (this.resource.msg_type == 1104 || this.resource.msg_type == 1115) | ||||
|         ) | ||||
|       ) { | ||||
|         this.updateUnreadMsgNumAdd() | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   //更新未读数量+1
 | ||||
|   updateUnreadMsgNumAdd() { | ||||
|     handleFindWebview(`updateUnreadMsgNumAdd()`) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * 显示消息提示 | ||||
|    * @returns | ||||
| @ -173,6 +132,7 @@ class Talk extends Base { | ||||
|     //     lang: 'zh-CN',
 | ||||
|     //     body: '您有新的消息请注意查收'
 | ||||
|     //   })
 | ||||
| 
 | ||||
|     //   notification.onclick = () => {
 | ||||
|     //     notification.close()
 | ||||
|     //   }
 | ||||
| @ -200,16 +160,12 @@ class Talk extends Base { | ||||
| 
 | ||||
|     ServeCreateTalkList({ | ||||
|       talk_type, | ||||
|       receiver_id, | ||||
|     }).then(async ({ code, data }) => { | ||||
|       receiver_id | ||||
|     }).then(({ code, data }) => { | ||||
|       if (code == 200) { | ||||
|         let item = formatTalkItem(data) | ||||
|         if (!item?.is_disturb) { | ||||
|           item.unread_num = 1 | ||||
|           this.updateUnreadMsgNumAdd() | ||||
|         } | ||||
|         item.unread_num = 1 | ||||
|         useTalkStore().addItem(item) | ||||
|         await useTalkStore().loadTalkList() | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| @ -219,76 +175,24 @@ class Talk extends Base { | ||||
|    */ | ||||
|   insertTalkRecord() { | ||||
|     let record = this.resource | ||||
|     let newRecord = formatTalkRecord(this.getAccountId(), this.resource) | ||||
|     const { addDialogueRecord, addChatRecord } = useDialogueListStore() | ||||
|     let newRecord = formatTalkRecord(this.getAccountId(), this.resource); | ||||
|     const {addDialogueRecord,addChatRecord} = useDialogueListStore() | ||||
|     // 群成员变化的消息,需要更新群成员列表
 | ||||
|     if ([1102, 1103, 1104, 1115].includes(record.msg_type)) { | ||||
|     if ([1102, 1103, 1104].includes(record.msg_type)) { | ||||
|       useDialogueStore().updateGroupMembers() | ||||
|     } | ||||
| 
 | ||||
|     //群解散时,需要更新群成员权限
 | ||||
|     if ([1106].includes(record.msg_type)) { | ||||
|       useDialogueStore().updateDismiss() | ||||
|     } | ||||
| 
 | ||||
|     //群成员被移出时,需要更新群成员权限
 | ||||
|     if ([1104, 1115].includes(record.msg_type)) { | ||||
|       console.error(this.resource.extra.members, 'this.resource.extra.members') | ||||
|       if (this.resource?.extra?.members?.length > 0) { | ||||
|         const isMeQuit = this.resource.extra.members.find( | ||||
|           (item) => item.user_id === this.getAccountId(), | ||||
|         ) | ||||
|         if (isMeQuit) { | ||||
|           useDialogueStore().updateQuit() | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     //群禁言变化时,需要更新群禁言状态——即更新群成员列表
 | ||||
|     if ([1107, 1108, 1109, 1110].includes(record.msg_type)) { | ||||
|       useDialogueStore().updateGroupMembers() | ||||
|     } | ||||
|     //群公告变化时,需要更新群公告(新增和修改有热更新,删除没有)
 | ||||
|     if ([13].includes(record.msg_type)) { | ||||
|       useGroupStore().ServeGetGroupNotices() | ||||
|     } | ||||
|     //群管理员变化时,需要更新群管理员列表——即更新群成员列表,同时更新群信息
 | ||||
|     if ([1114].includes(record.msg_type)) { | ||||
|       useDialogueStore().updateGroupMembers() | ||||
|       useGroupStore().ServeGroupDetail() | ||||
|     } | ||||
|     if ([1116].includes(record.msg_type)) { | ||||
|       // 更新会话信息
 | ||||
|       useDialogueStore().setDialogue({ | ||||
|         name: record.extra.group_name, | ||||
|         talk_type: record.talk_type, | ||||
|         receiver_id: record.receiver_id, | ||||
|       }) | ||||
|       // 更新群聊信息
 | ||||
|       useGroupStore().updateGroupInfo({ | ||||
|         group_name: record.extra.group_name, | ||||
|         avatar: record.extra.group_avatar, | ||||
|       }) | ||||
|       // 更新会话列表中的会话信息
 | ||||
|       const dialogue = useDialogueListStore().getDialogueList( | ||||
|         `${record.talk_type}_${record.receiver_id}`, | ||||
|       ) | ||||
|       if (dialogue) { | ||||
|         dialogue.talk.username = record.extra.group_name | ||||
|       } | ||||
|     } | ||||
|     addDialogueRecord([newRecord], 'add') | ||||
|     addChatRecord(this.getIndexName(), newRecord) | ||||
|     addDialogueRecord([newRecord],'add') | ||||
|     addChatRecord(this.getIndexName(),newRecord) | ||||
|     useDialogueStore().addDialogueRecord(newRecord) | ||||
| 
 | ||||
|     if (!this.isCurrSender()) { | ||||
|       // 推送已读消息
 | ||||
|       // setTimeout(() => {
 | ||||
|       //   ws.emit('im.message.read', {
 | ||||
|       //     receiver_id: this.sender_id,
 | ||||
|       //     msg_ids: [this.resource.msg_id],
 | ||||
|       //   })
 | ||||
|       // }, 1000)
 | ||||
|       setTimeout(() => { | ||||
|         ws.emit('im.message.read', { | ||||
|           receiver_id: this.sender_id, | ||||
|           msg_ids: [this.resource.msg_id] | ||||
|         }) | ||||
|       }, 1000) | ||||
|     } | ||||
| 
 | ||||
|     // 获取聊天面板元素节点
 | ||||
| @ -296,8 +200,7 @@ class Talk extends Base { | ||||
|     if (!el) return | ||||
| 
 | ||||
|     // 判断的滚动条是否在底部
 | ||||
|     const isBottom = | ||||
|       Math.ceil(el.scrollTop) + el.clientHeight >= el.scrollHeight | ||||
|     const isBottom = Math.ceil(el.scrollTop) + el.clientHeight >= el.scrollHeight | ||||
| 
 | ||||
|     if (isBottom || record.user_id == this.getAccountId()) { | ||||
|       nextTick(() => { | ||||
| @ -310,15 +213,14 @@ class Talk extends Base { | ||||
|     useTalkStore().updateItem({ | ||||
|       index_name: this.getIndexName(), | ||||
|       msg_text: this.getTalkText(), | ||||
|       updated_at: parseTime(new Date()), | ||||
|       updated_at: parseTime(new Date()) | ||||
|     }) | ||||
| 
 | ||||
|     if (this.talk_type == 1 && this.getAccountId() !== this.sender_id) { | ||||
|       //不在此处维护未读消息数量
 | ||||
|       // ServeClearTalkUnreadNum({
 | ||||
|       //   talk_type: 1,
 | ||||
|       //   receiver_id: this.sender_id,
 | ||||
|       // })
 | ||||
|       ServeClearTalkUnreadNum({ | ||||
|         talk_type: 1, | ||||
|         receiver_id: this.sender_id | ||||
|       }) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| @ -329,31 +231,8 @@ class Talk extends Base { | ||||
|     useTalkStore().updateMessage({ | ||||
|       index_name: this.getIndexName(), | ||||
|       msg_text: this.getTalkText(), | ||||
|       updated_at: parseTime(new Date()), | ||||
|       updated_at: parseTime(new Date()) | ||||
|     }) | ||||
|     //收到新消息时,同时判断是否有人@我
 | ||||
|     if ( | ||||
|       this.resource.msg_type === 1 && | ||||
|       this.resource?.extra?.mentions?.length > 0 | ||||
|     ) { | ||||
|       const findMention = this.resource?.extra?.mentions?.find( | ||||
|         (mention) => mention === this.getAccountId(), | ||||
|       ) | ||||
|       //有人@我或者@所有人,则更新会话列表
 | ||||
|       if (findMention || this.resource?.extra?.mentions?.includes(0)) { | ||||
|         useTalkStore().loadTalkList() | ||||
|       } | ||||
|     } | ||||
|     if (this.resource.msg_type == 1116) { | ||||
|       // 更新会话列表中的会话信息
 | ||||
|       const dialogue = useDialogueListStore().getDialogueList( | ||||
|         `${this.resource.talk_type}_${this.resource.receiver_id}`, | ||||
|       ) | ||||
|       if (dialogue) { | ||||
|         dialogue.talk.username = this.resource.extra.group_name | ||||
|       } | ||||
|       useTalkStore().loadTalkList() | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -11,12 +11,7 @@ import { reactive, nextTick, computed, h, inject } from 'vue' | ||||
| //   EditTwo,
 | ||||
| //   IdCard
 | ||||
| // } from '@icon-park/vue-next'
 | ||||
| import { | ||||
|   ServeTopTalkList, | ||||
|   ServeDeleteTalkList, | ||||
|   ServeSetNotDisturb, | ||||
|   ServeClearTalkUnreadNum, | ||||
| } from '@/api/chat' | ||||
| import { ServeTopTalkList, ServeDeleteTalkList, ServeSetNotDisturb } from '@/api/chat' | ||||
| import { useDialogueStore, useTalkStore, useDialogueListStore } from '@/store' | ||||
| import { ServeSecedeGroup } from '@/api/group' | ||||
| // import { ServeDeleteContact, ServeEditContactRemark } from '@/api/contact'
 | ||||
| @ -28,7 +23,7 @@ export function useSessionMenu() { | ||||
|     show: false, | ||||
|     x: 0, | ||||
|     y: 0, | ||||
|     item: {}, | ||||
|     item: {} | ||||
|   }) | ||||
| 
 | ||||
|   const dialogueStore = useDialogueStore() | ||||
| @ -123,22 +118,10 @@ export function useSessionMenu() { | ||||
|   // 移除会话
 | ||||
|   const onRemoveTalk = (item) => { | ||||
|     ServeDeleteTalkList({ | ||||
|       list_id: item.id, | ||||
|       list_id: item.id | ||||
|     }).then(({ code }) => { | ||||
|       if (code == 200) { | ||||
|         onDeleteTalk(item.index_name) | ||||
|         console.error(item, 'item') | ||||
|         if (item.unread_num > 0) { | ||||
|           //同时已读
 | ||||
|           ServeClearTalkUnreadNum( | ||||
|             { | ||||
|               talk_type: item.talk_type, | ||||
|               receiver_id: item.receiver_id, | ||||
|             }, | ||||
|             item.unread_num, | ||||
|           ).then(() => { | ||||
|           }) | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| @ -148,13 +131,13 @@ export function useSessionMenu() { | ||||
|     ServeSetNotDisturb({ | ||||
|       talk_type: item.talk_type, | ||||
|       receiver_id: item.receiver_id, | ||||
|       is_disturb: item.is_disturb == 0 ? 1 : 0, | ||||
|       is_disturb: item.is_disturb == 0 ? 1 : 0 | ||||
|     }).then(({ code, message }) => { | ||||
|       if (code == 200) { | ||||
|         message.success('设置成功!') | ||||
|         talkStore.updateItem({ | ||||
|           index_name: item.index_name, | ||||
|           is_disturb: item.is_disturb == 0 ? 1 : 0, | ||||
|           is_disturb: item.is_disturb == 0 ? 1 : 0 | ||||
|         }) | ||||
|       } else { | ||||
|         message.error(message) | ||||
| @ -170,12 +153,12 @@ export function useSessionMenu() { | ||||
| 
 | ||||
|     ServeTopTalkList({ | ||||
|       list_id: item.id, | ||||
|       type: item.is_top == 0 ? 1 : 2, | ||||
|       type: item.is_top == 0 ? 1 : 2 | ||||
|     }).then(({ code, message }) => { | ||||
|       if (code == 200) { | ||||
|         talkStore.updateItem({ | ||||
|           index_name: item.index_name, | ||||
|           is_top: item.is_top == 0 ? 1 : 0, | ||||
|           is_top: item.is_top == 0 ? 1 : 0 | ||||
|         }) | ||||
|       } else { | ||||
|         message.error(message) | ||||
| @ -218,7 +201,7 @@ export function useSessionMenu() { | ||||
|       negativeText: '取消', | ||||
|       onPositiveClick: () => { | ||||
|         ServeSecedeGroup({ | ||||
|           group_id: item.receiver_id, | ||||
|           group_id: item.receiver_id | ||||
|         }).then(({ code, message }) => { | ||||
|           if (code == 200) { | ||||
|             message.success('已退出群聊') | ||||
| @ -227,7 +210,7 @@ export function useSessionMenu() { | ||||
|             message.error(message) | ||||
|           } | ||||
|         }) | ||||
|       }, | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
| @ -276,18 +259,12 @@ export function useSessionMenu() { | ||||
|       disturb: onSetDisturb, | ||||
|       signout_group: onSignOutGroup, | ||||
|       delete_contact: onDeleteContact, | ||||
|       remark: onChangeRemark, | ||||
|       remark: onChangeRemark | ||||
|     } | ||||
| 
 | ||||
|     dropdown.show = false | ||||
|     evnets[key] && evnets[key](dropdown.item) | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     dropdown, | ||||
|     onCloseContextMenu, | ||||
|     onContextMenuTalkHandle, | ||||
|     onToTopTalk, | ||||
|     onRemoveTalk, | ||||
|   } | ||||
|   return { dropdown, onCloseContextMenu, onContextMenuTalkHandle, onToTopTalk, onRemoveTalk } | ||||
| } | ||||
|  | ||||
| @ -74,7 +74,7 @@ export const useTalkRecord = (uid) => { | ||||
|   // 加载数据列表
 | ||||
|   const load = async (params) => { | ||||
|     const request = { | ||||
|       limit: 30, | ||||
|       limit:10, | ||||
|       ...params, | ||||
|       talk_type: params.talk_type, | ||||
|       receiver_id: params.receiver_id, | ||||
|  | ||||
							
								
								
									
										77
									
								
								src/main.js
									
									
									
									
									
								
							
							
						
						| @ -10,21 +10,11 @@ import tmui from '@/uni_modules/tmui' | ||||
| import { config } from '@/config/tmui/index.js' | ||||
| import 'dayjs/locale/zh-cn' | ||||
| import xLoaderror from '@/components/x-loaderror/index.vue' | ||||
| import asyncLoading from '@/components/async-loading/index.vue' | ||||
| import asyncError from '@/components/async-error/index.vue' | ||||
| import { vLoading } from '@/components/x-loading/index.js' | ||||
| import messagePopup from '@/components/x-message/useMessagePopup' | ||||
| import pageAnimation from '@/components/page-animation/index.vue' | ||||
| import * as plugins from './plugins' | ||||
| import { | ||||
|   useDialogueStore, | ||||
|   useTalkStore, | ||||
|   useUserStore, | ||||
|   useDialogueListStore, | ||||
| } from '@/store' | ||||
| import { uniStorage } from '@/utils/uniStorage.js' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| 
 | ||||
| import { useDialogueStore, useTalkStore } from '@/store' | ||||
| const { showMessage } = messagePopup() | ||||
| dayjs.locale('zh-cn') | ||||
| if (import.meta.env.VITE_SHOW_CONSOLE === 'true') { | ||||
| @ -39,8 +29,6 @@ export function createApp() { | ||||
|   app.mixin(pageAnimation) | ||||
|   app.component('customNavbar', customNavbar) | ||||
|   app.component('x-loaderror', xLoaderror) | ||||
|   app.component('AsyncLoading', asyncLoading) | ||||
|   app.component('AsyncError', asyncError) | ||||
|   app.directive('no-space', { | ||||
|     mounted(el) { | ||||
|       el.addEventListener('input', (e) => { | ||||
| @ -72,75 +60,22 @@ export function createApp() { | ||||
|     ) { | ||||
|       return | ||||
|     } | ||||
|     console.log('===准备创建本地通知栏消息') | ||||
|     handleFindWebview(`doCreatePushMessage('${msg}')`) | ||||
|     let OAWebView = plus.webview.all() | ||||
|     //all里面第一个是入口webview
 | ||||
|     OAWebView[0].evalJS(`doCreatePushMessage('${msg}')`) | ||||
|   } | ||||
| 
 | ||||
|   //处理聊天推送弹窗点开
 | ||||
|   window.openUniPushMsg = (msg) => { | ||||
|     console.log('=====点击通知栏消息') | ||||
|     console.log("=====点击通知栏消息") | ||||
|     let pushMsg = JSON.parse(decodeURIComponent(msg)) | ||||
|     console.log('=====pushMsg', pushMsg) | ||||
|     console.log("=====pushMsg",pushMsg) | ||||
|     //由于弹窗前处理了不该弹窗的场景,因此这里弹窗可以一并处理
 | ||||
|     //也就是都跳转到聊天页面
 | ||||
|     const talkStore = useTalkStore() | ||||
|     talkStore.toTalk(pushMsg?.payload?.talk_type, pushMsg?.payload?.receiver_id) | ||||
|   } | ||||
| 
 | ||||
|   //处理当用户信息发生变化时,更新用户信息
 | ||||
|   window.updateUserInfo = () => { | ||||
|     useUserStore().loadSetting() | ||||
|   } | ||||
|   // 通讯录跳转
 | ||||
|   window.handleContacts = () => { | ||||
|     // 旧版本-按组织架构树的通讯录
 | ||||
|     uni.navigateTo({ | ||||
|       url: '/pages/chooseByDeps/index?chooseMode=3&type=true' | ||||
|     }); | ||||
| 
 | ||||
|     // 新版本-按公司别、好友、群组的通讯录
 | ||||
|     // uni.navigateTo({
 | ||||
|     //   url: '/pages/addressBook/index?type=true',
 | ||||
|     // })
 | ||||
|   } | ||||
| 
 | ||||
|   //处理OA、墨册强制刷新时,聊天同步强制刷新
 | ||||
|   window.doLocationRefresh = () => { | ||||
|     uniStorage.removeItem('dialogueList') | ||||
|     uniStorage.removeItem('dialogue') | ||||
|     useUserStore().loadSetting() | ||||
|     useDialogueListStore().dialogueList.value = [] | ||||
|     // location.reload(true)
 | ||||
|   } | ||||
| 
 | ||||
|   //检查聊天页面是否可用
 | ||||
|   window.checkChatWebviewAvailable = () => { | ||||
|     handleFindWebview(`doneCheckChatWebviewAvailable()`) | ||||
|   } | ||||
| 
 | ||||
|   //获取从base传来的多选视频列表
 | ||||
|   window.getBaseMulVideo = (videoList) => { | ||||
|     const videos = JSON.parse(decodeURIComponent(videoList)) | ||||
|     console.error('=====videos', videos) | ||||
|     if (videos.length > 0) { | ||||
|       const videoUri = videos[0] | ||||
|       console.error('=====videoUri', videoUri) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   //检查是否是特殊测试用户,开启控制台
 | ||||
|   const checkTestUser = () => { | ||||
|     const userStore = useUserStore() | ||||
|     if ( | ||||
|       import.meta.env.VITE_SHOW_CONSOLE === 'false' && | ||||
|       userStore.mobile == '18100591363' | ||||
|     ) { | ||||
|       new VConsole() | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   checkTestUser() | ||||
| 
 | ||||
|   window.message = ['success', 'error', 'warning'].reduce((acc, type) => { | ||||
|     acc[type] = (message) => { | ||||
|       if (typeof message === 'string') { | ||||
|  | ||||
| @ -5,7 +5,8 @@ | ||||
|       "^tm-(.*)": "@/tmui/components/tm-$1/tm-$1.vue" | ||||
|     } | ||||
|   }, | ||||
|   "pages": [{ | ||||
|   "pages": [ | ||||
|     { | ||||
|       "path": "pages/index/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
| @ -177,38 +178,6 @@ | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/chatSettings/groupManage/manageGroupDeps", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/complaintReport/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/addressBook/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/addressBook/addFriend/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "globalStyle": { | ||||
|  | ||||
| @ -1,398 +0,0 @@ | ||||
| <template> | ||||
|   <div class="add-friend-page"> | ||||
|     <zPaging ref="zPaging" :show-scrollbar="false" @scrolltolower="doLoadMore"> | ||||
|       <template #top> | ||||
|         <div :class="'top_bg'"> | ||||
|           <customNavbar | ||||
|             :class="'index_top_navbar'" | ||||
|             :title="$t('addFriend.pageTitle')" | ||||
|           ></customNavbar> | ||||
|           <div class="pl-[32rpx] pr-[32rpx] pt-[32rpx] pb-[32rpx]"> | ||||
|             <customInput | ||||
|               :searchText="searchVal" | ||||
|               @inputSearchText="inputSearchText" | ||||
|             ></customInput> | ||||
|           </div> | ||||
|         </div> | ||||
|       </template> | ||||
|       <div class="add-friend"> | ||||
|         <div class="add-friend-list" v-if="state.friendsList.length > 0"> | ||||
|           <div | ||||
|             class="members-list-each" | ||||
|             v-for="(item, index) in state.friendsList" | ||||
|             :key="index" | ||||
|             @click="toUserDetail(item)" | ||||
|           > | ||||
|             <div class="members-info"> | ||||
|               <avatarModule | ||||
|                 :mode="1" | ||||
|                 :avatar="item.avatar" | ||||
|                 :groupType="0" | ||||
|                 :userName="item.nickname" | ||||
|                 :customStyle="{ width: '72rpx', height: '72rpx' }" | ||||
|                 :customTextStyle="{ | ||||
|                   fontSize: '32rpx', | ||||
|                   fontWeight: 'bold', | ||||
|                   color: '#fff', | ||||
|                   lineHeight: '44rpx', | ||||
|                 }" | ||||
|               ></avatarModule> | ||||
|               <div class="members-info-area"> | ||||
|                 <div | ||||
|                   class="members-info-basic" | ||||
|                   :style="{ padding: item.company_name ? 0 : '0 0 24rpx' }" | ||||
|                 > | ||||
|                   <span class="members-name"> | ||||
|                     {{ item.nickname }} | ||||
|                   </span> | ||||
|                   <span class="members-jobNum"> | ||||
|                     {{ item.job_num }} | ||||
|                   </span> | ||||
|                   <!-- <span class="members-company">{{ item.company_name }}</span> --> | ||||
|                 </div> | ||||
|                 <div class="members-positions-area"> | ||||
|                   <tm-popover position="bc"> | ||||
|                     <tm-scrolly :refresher="false" :height="84"> | ||||
|                       <div class="members-positions"> | ||||
|                         <div | ||||
|                           class="members-positions-each" | ||||
|                           v-for="(postionItem, | ||||
|                           positionIndex) in item.user_position" | ||||
|                           :key="positionIndex" | ||||
|                         > | ||||
|                           {{ postionItem.position_name }} | ||||
|                         </div> | ||||
|                       </div> | ||||
|                     </tm-scrolly> | ||||
|                     <template v-slot:label> | ||||
|                       <tm-scrolly | ||||
|                         :refresher="false" | ||||
|                         :height=" | ||||
|                           item.user_position.length >= 4 | ||||
|                             ? 180 | ||||
|                             : item.user_position.length === 3 | ||||
|                             ? 140 | ||||
|                             : item.user_position.length === 2 | ||||
|                             ? 100 | ||||
|                             : item.user_position.length === 1 | ||||
|                             ? 60 | ||||
|                             : 0 | ||||
|                         " | ||||
|                       > | ||||
|                         <div class="members-positions-popover-box"> | ||||
|                           <div | ||||
|                             class="members-positions-popover" | ||||
|                             v-for="(postionItem, | ||||
|                             positionIndex) in item.user_position" | ||||
|                             :key="positionIndex" | ||||
|                           > | ||||
|                             {{ postionItem.position_name }} | ||||
|                           </div> | ||||
|                         </div> | ||||
|                       </tm-scrolly> | ||||
|                     </template> | ||||
|                   </tm-popover> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="company-infos" v-if="item.company_name"> | ||||
|               <div class="company-each"> | ||||
|                 <span>{{ item.company_name }}</span> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="addressBook-noData" v-if="state.friendsList.length === 0"> | ||||
|           <img src="@/static/image/search/search-no-data.png" /> | ||||
|           <span> | ||||
|             {{ | ||||
|               searchVal | ||||
|                 ? $t('addFriend.message.notFindData') | ||||
|                 : $t('search.hint') | ||||
|             }} | ||||
|           </span> | ||||
|         </div> | ||||
|       </div> | ||||
|     </zPaging> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import customInput from '@/components/custom-input/custom-input.vue' | ||||
| import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import avatarModule from '@/components/avatar-module/index.vue' | ||||
| import { ServeFriendSearch } from '@/api/addressBook/index' | ||||
| import { handleSetWebviewStyle } from '@/utils/common' | ||||
| 
 | ||||
| import { ref, onMounted, reactive } from 'vue' | ||||
| 
 | ||||
| import { useI18n } from 'vue-i18n' | ||||
| const { t } = useI18n() | ||||
| 
 | ||||
| const searchVal = ref('') | ||||
| 
 | ||||
| const state = reactive({ | ||||
|   friendsListPage: 1, //当前查询的可添加好友列表分页 | ||||
|   friendsListPageSize: 10, //可添加好友列表单页数量 | ||||
|   friendsList: [], //可添加好友列表 | ||||
|   hasMoreFriends: true, //是否还有更多可添加好友数据 | ||||
| }) | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   getFriendsList() | ||||
|   handleSetWebviewStyle() | ||||
| }) | ||||
| 
 | ||||
| //输入搜索内容 | ||||
| const inputSearchText = (e) => { | ||||
|   searchVal.value = e | ||||
| } | ||||
| 
 | ||||
| //搜索可添加好友(精确搜索) | ||||
| const getFriendsList = () => { | ||||
|   let params = { | ||||
|     name: searchVal.value, | ||||
|   } | ||||
|   ServeFriendSearch(params) | ||||
|     .then((res) => { | ||||
|       console.error(res) | ||||
|       if (res?.code === 200) { | ||||
|         if (state.friendsListPage === 1) { | ||||
|           state.friendsList = res.data?.user_list || [] | ||||
|         } else { | ||||
|           state.friendsList = state.friendsList.concat( | ||||
|             res.data?.user_list || [], | ||||
|           ) | ||||
|         } | ||||
|         if (state.friendsList.length < res.data?.count) { | ||||
|           state.hasMoreFriends = true | ||||
|         } else { | ||||
|           state.hasMoreFriends = false | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|     .catch((err) => { | ||||
|       console.log(err) | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| //点击进入用户详情页面 | ||||
| const toUserDetail = (userInfo) => { | ||||
|   uni.navigateTo({ | ||||
|     url: | ||||
|       '/pages/dialog/dialogDetail/userDetail??erpUserId=' + | ||||
|       userInfo.erp_user_id, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //加载更多数据 | ||||
| const doLoadMore = (e) => { | ||||
|   state.friendsListPage += 1 | ||||
|   getFriendsList() | ||||
| } | ||||
| 
 | ||||
| watch( | ||||
|   () => searchVal.value, | ||||
|   (newVal) => { | ||||
|     state.friendsListPage = 1 | ||||
|     getFriendsList() | ||||
|   }, | ||||
| ) | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| ::v-deep .zp-paging-container-content { | ||||
|   height: 100%; | ||||
|   display: flex; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .tmicon-angle-left { | ||||
|   color: #fff !important; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .text-weight-b { | ||||
|   color: #fff !important; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .statusHeightTop > .noNvueBorder:first-child { | ||||
|   background: transparent !important; | ||||
|   border: none !important; | ||||
| } | ||||
| 
 | ||||
| .top_bg { | ||||
|   background: url('@/static/image/mine/page_top.png') no-repeat; | ||||
|   background-size: cover; | ||||
|   background-position: bottom center; | ||||
| } | ||||
| 
 | ||||
| :deep(.animateAll_tabs_tmui) { | ||||
|   width: 120rpx !important; | ||||
|   height: 10rpx !important; | ||||
| } | ||||
| 
 | ||||
| .add-friend-page { | ||||
|   .add-friend { | ||||
|     flex: 1; | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|     background-size: cover; | ||||
|     background-position: bottom center; | ||||
|     background-attachment: fixed; | ||||
|     width: 100%; | ||||
| 
 | ||||
|     .add-friend-list { | ||||
|       margin: 30rpx 24rpx; | ||||
|       overflow: hidden; | ||||
|       flex: 1; | ||||
|       gap: 30rpx 0; | ||||
| 
 | ||||
|       .members-list-each { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: flex-start; | ||||
|         justify-content: center; | ||||
|         background-color: #fff; | ||||
|         width: 100%; | ||||
|         border-radius: 8rpx; | ||||
| 
 | ||||
|         .swipe-action { | ||||
|           width: 100%; | ||||
|           overflow: visible !important; | ||||
| 
 | ||||
|           :deep(.wd-swipe-action__right) { | ||||
|             display: flex; | ||||
|             flex-direction: row; | ||||
|             align-items: center; | ||||
|             justify-content: center; | ||||
|             background-color: #cf3050; | ||||
|             padding: 0 48rpx; | ||||
|             border-radius: 0 8rpx 8rpx 0; | ||||
|             span { | ||||
|               color: #fff; | ||||
|               line-height: 1; | ||||
|               font-size: 30rpx; | ||||
|               font-weight: 400; | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| 
 | ||||
|         .members-info { | ||||
|           display: flex; | ||||
|           flex-direction: row; | ||||
|           align-items: flex-start; | ||||
|           justify-content: flex-start; | ||||
|           gap: 16rpx; | ||||
|           padding: 24rpx 24rpx 0; | ||||
| 
 | ||||
|           .members-info-area { | ||||
|             display: flex; | ||||
|             flex-direction: row; | ||||
|             align-items: flex-start; | ||||
|             justify-content: flex-start; | ||||
|             gap: 16rpx; | ||||
| 
 | ||||
|             .members-info-basic { | ||||
|               display: flex; | ||||
|               flex-direction: column; | ||||
|               align-items: flex-start; | ||||
|               justify-content: center; | ||||
| 
 | ||||
|               .members-name { | ||||
|                 font-size: 30rpx; | ||||
|                 font-weight: 500; | ||||
|                 width: 150rpx; | ||||
|                 word-break: break-all; | ||||
|               } | ||||
| 
 | ||||
|               .members-jobNum { | ||||
|                 font-size: 26rpx; | ||||
|                 font-weight: 400; | ||||
|                 color: #999; | ||||
|                 word-break: break-all; | ||||
|                 margin: 10rpx 0 0; | ||||
|               } | ||||
| 
 | ||||
|               .members-company { | ||||
|                 font-size: 24rpx; | ||||
|                 font-weight: 400; | ||||
|                 line-height: 1; | ||||
|                 color: #999; | ||||
|               } | ||||
|             } | ||||
|             .members-positions-area { | ||||
|               .members-positions { | ||||
|                 display: flex; | ||||
|                 flex-direction: row; | ||||
|                 align-items: center; | ||||
|                 justify-content: flex-start; | ||||
|                 flex-wrap: wrap; | ||||
|                 gap: 10rpx; | ||||
|                 padding: 4rpx 0 0; | ||||
|                 max-height: 84rpx; | ||||
|                 overflow: hidden; | ||||
|                 -webkit-box-orient: vertical; | ||||
|                 -webkit-line-clamp: 2; | ||||
|                 line-clamp: 2; | ||||
| 
 | ||||
|                 .members-positions-each { | ||||
|                   background-color: #eee9f8; | ||||
|                   line-height: 1; | ||||
|                   font-size: 24rpx; | ||||
|                   padding: 4rpx 16rpx; | ||||
|                   color: #46299d; | ||||
|                 } | ||||
|               } | ||||
|               .members-positions-popover-box { | ||||
|                 display: flex; | ||||
|                 flex-direction: column; | ||||
|                 align-items: center; | ||||
|                 justify-content: center; | ||||
|                 gap: 10rpx 0; | ||||
| 
 | ||||
|                 .members-positions-popover { | ||||
|                   background-color: #eee9f8; | ||||
|                   line-height: 1; | ||||
|                   font-size: 24rpx; | ||||
|                   padding: 8rpx 16rpx; | ||||
|                   color: #46299d; | ||||
|                   width: 100%; | ||||
|                 } | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|         .company-infos { | ||||
|           margin: 0 0 0 88rpx; | ||||
|           padding: 0 0 24rpx 24rpx; | ||||
|           .company-each { | ||||
|             span { | ||||
|               font-size: 24rpx; | ||||
|               font-weight: 400; | ||||
|               line-height: 1; | ||||
|               color: #999; | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     .addressBook-noData { | ||||
|       margin: 30rpx 24rpx; | ||||
|       overflow: hidden; | ||||
|       flex: 1; | ||||
|       gap: 30rpx 0; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       img { | ||||
|         width: 500rpx; | ||||
|       } | ||||
|       span { | ||||
|         font-size: 30rpx; | ||||
|         font-weight: 400; | ||||
|         line-height: 1; | ||||
|         color: #999; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -1,944 +0,0 @@ | ||||
| <template> | ||||
|   <div class="address-book-page"> | ||||
|     <zPaging ref="zPaging" :show-scrollbar="false" @scrolltolower="doLoadMore"> | ||||
|       <template #top> | ||||
|         <div :class="'top_bg'"> | ||||
|           <customNavbar | ||||
|             :class="'index_top_navbar'" | ||||
|             :title="$t('index.mine.addressBook')" | ||||
|             :hideHome="navshow" | ||||
|             :hideBack="navshow" | ||||
|           > | ||||
|             <template #left> | ||||
|               <tm-icon | ||||
|                 @click="goWebHome" | ||||
|                 v-if="navshow" | ||||
|                 name="tmicon-angle-left" | ||||
|                 style="padding-left: 30rpx;" | ||||
|               ></tm-icon> | ||||
|             </template> | ||||
|           </customNavbar> | ||||
|           <div class="pl-[32rpx] pr-[32rpx] pt-[32rpx] pb-[32rpx]"> | ||||
|             <customInput | ||||
|               :searchText="searchVal" | ||||
|               @inputSearchText="inputSearchText" | ||||
|             ></customInput> | ||||
|           </div> | ||||
|         </div> | ||||
|         <tm-tabs | ||||
|           :list="state.addressBookTabs" | ||||
|           align="center" | ||||
|           :width="750" | ||||
|           :height="300" | ||||
|           :itemWidth="250" | ||||
|           default-name="company" | ||||
|           activeColor="#46299d" | ||||
|           activeFontColor="#46299d" | ||||
|           tabs-line-ani-color="#46299d" | ||||
|           :showTabsLineAni="true" | ||||
|           :showTabsLine="false" | ||||
|           :activeFontSize="32" | ||||
|           :itemFontSize="30" | ||||
|           @update:activeName="updateAddressBookTab" | ||||
|         ></tm-tabs> | ||||
|       </template> | ||||
|       <div class="address-book"> | ||||
|         <div class="address-book-tabs-panes-list"> | ||||
|           <div | ||||
|             class="tabs-panes-each address-book-company" | ||||
|             v-if=" | ||||
|               state.addressBookActiveTab === 'company' && | ||||
|               state.myContractList.length > 0 | ||||
|             " | ||||
|           > | ||||
|             <div class="address-book-company-name"> | ||||
|               <span>{{ state.myCompany }}</span> | ||||
|             </div> | ||||
|             <div | ||||
|               class="members-list-each" | ||||
|               v-for="(item, index) in state.myContractList" | ||||
|               :key="index" | ||||
|             > | ||||
|               <div class="members-info" @click="toUserDetail(item)"> | ||||
|                 <avatarModule | ||||
|                   :mode="1" | ||||
|                   :avatar="item.avatar" | ||||
|                   :groupType="0" | ||||
|                   :userName="item.nickname" | ||||
|                   :customStyle="{ width: '72rpx', height: '72rpx' }" | ||||
|                   :customTextStyle="{ | ||||
|                     fontSize: '32rpx', | ||||
|                     fontWeight: 'bold', | ||||
|                     color: '#fff', | ||||
|                     lineHeight: '44rpx', | ||||
|                   }" | ||||
|                 ></avatarModule> | ||||
|                 <div class="members-info-area"> | ||||
|                   <div class="members-info-basic" style="padding: 0 0 24rpx;"> | ||||
|                     <span class="members-name"> | ||||
|                       {{ item.nickname }} | ||||
|                     </span> | ||||
|                     <span class="members-jobNum"> | ||||
|                       {{ item.job_num }} | ||||
|                     </span> | ||||
|                   </div> | ||||
|                   <div class="members-positions-area"> | ||||
|                     <tm-popover position="bc"> | ||||
|                       <tm-scrolly :refresher="false" :height="84"> | ||||
|                         <div class="members-positions"> | ||||
|                           <div | ||||
|                             class="members-positions-each" | ||||
|                             v-for="(postionItem, | ||||
|                             positionIndex) in item.user_position" | ||||
|                             :key="positionIndex" | ||||
|                           > | ||||
|                             {{ postionItem.position_name }} | ||||
|                           </div> | ||||
|                         </div> | ||||
|                       </tm-scrolly> | ||||
|                       <template v-slot:label> | ||||
|                         <tm-scrolly | ||||
|                           :refresher="false" | ||||
|                           :height=" | ||||
|                             item.user_position.length >= 4 | ||||
|                               ? 180 | ||||
|                               : item.user_position.length === 3 | ||||
|                               ? 140 | ||||
|                               : item.user_position.length === 2 | ||||
|                               ? 100 | ||||
|                               : item.user_position.length === 1 | ||||
|                               ? 60 | ||||
|                               : 0 | ||||
|                           " | ||||
|                         > | ||||
|                           <div class="members-positions-popover-box"> | ||||
|                             <div | ||||
|                               class="members-positions-popover" | ||||
|                               v-for="(postionItem, | ||||
|                               positionIndex) in item.user_position" | ||||
|                               :key="positionIndex" | ||||
|                             > | ||||
|                               {{ postionItem.position_name }} | ||||
|                             </div> | ||||
|                           </div> | ||||
|                         </tm-scrolly> | ||||
|                       </template> | ||||
|                     </tm-popover> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div | ||||
|             class="tabs-panes-each address-book-friends" | ||||
|             v-if=" | ||||
|               state.addressBookActiveTab === 'friends' && | ||||
|               state.myFriendsList.length > 0 | ||||
|             " | ||||
|           > | ||||
|             <div | ||||
|               class="members-list-each" | ||||
|               v-for="(item, index) in state.myFriendsList" | ||||
|               :key="index" | ||||
|             > | ||||
|               <wd-swipe-action | ||||
|                 class="swipe-action" | ||||
|                 @click="showDeleteModal(item, index)" | ||||
|                 v-if=" | ||||
|                   !item.company_name || | ||||
|                   (item.company_name && item.company_name !== state.myCompany) | ||||
|                 " | ||||
|               > | ||||
|                 <div class="members-info" @click="toUserDetail(item)"> | ||||
|                   <avatarModule | ||||
|                     :mode="1" | ||||
|                     :avatar="item.avatar" | ||||
|                     :groupType="0" | ||||
|                     :userName="item.nickname" | ||||
|                     :customStyle="{ width: '72rpx', height: '72rpx' }" | ||||
|                     :customTextStyle="{ | ||||
|                       fontSize: '32rpx', | ||||
|                       fontWeight: 'bold', | ||||
|                       color: '#fff', | ||||
|                       lineHeight: '44rpx', | ||||
|                     }" | ||||
|                   ></avatarModule> | ||||
|                   <div class="members-info-area"> | ||||
|                     <div | ||||
|                       class="members-info-basic" | ||||
|                       :style="{ padding: item.company_name ? 0 : '0 0 24rpx' }" | ||||
|                     > | ||||
|                       <span class="members-name"> | ||||
|                         {{ item.nickname }} | ||||
|                       </span> | ||||
|                       <span class="members-jobNum"> | ||||
|                         {{ item.job_num }} | ||||
|                       </span> | ||||
|                       <!-- <span class="members-company">{{ item.company_name }}</span> --> | ||||
|                     </div> | ||||
|                     <div class="members-positions-area"> | ||||
|                       <tm-popover position="bc"> | ||||
|                         <tm-scrolly :refresher="false" :height="84"> | ||||
|                           <div class="members-positions"> | ||||
|                             <div | ||||
|                               class="members-positions-each" | ||||
|                               v-for="(postionItem, | ||||
|                               positionIndex) in item.user_position" | ||||
|                               :key="positionIndex" | ||||
|                             > | ||||
|                               {{ postionItem.position_name }} | ||||
|                             </div> | ||||
|                           </div> | ||||
|                         </tm-scrolly> | ||||
|                         <template v-slot:label> | ||||
|                           <tm-scrolly | ||||
|                             :refresher="false" | ||||
|                             :height=" | ||||
|                               item.user_position.length >= 4 | ||||
|                                 ? 180 | ||||
|                                 : item.user_position.length === 3 | ||||
|                                 ? 140 | ||||
|                                 : item.user_position.length === 2 | ||||
|                                 ? 100 | ||||
|                                 : item.user_position.length === 1 | ||||
|                                 ? 60 | ||||
|                                 : 0 | ||||
|                             " | ||||
|                           > | ||||
|                             <div class="members-positions-popover-box"> | ||||
|                               <div | ||||
|                                 class="members-positions-popover" | ||||
|                                 v-for="(postionItem, | ||||
|                                 positionIndex) in item.user_position" | ||||
|                                 :key="positionIndex" | ||||
|                               > | ||||
|                                 {{ postionItem.position_name }} | ||||
|                               </div> | ||||
|                             </div> | ||||
|                           </tm-scrolly> | ||||
|                         </template> | ||||
|                       </tm-popover> | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                 <div class="company-infos" v-if="item.company_name"> | ||||
|                   <div class="company-each"> | ||||
|                     <span>{{ item.company_name }}</span> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                 <template #right> | ||||
|                   <div | ||||
|                     v-for="(swipeActionItem, | ||||
|                     swipeActionIndex) in state.swipeAction" | ||||
|                     :key="swipeActionIndex" | ||||
|                   > | ||||
|                     <span>{{ swipeActionItem.text }}</span> | ||||
|                   </div> | ||||
|                 </template> | ||||
|               </wd-swipe-action> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div | ||||
|             class="tabs-panes-each address-book-groups" | ||||
|             v-if=" | ||||
|               state.addressBookActiveTab === 'groups' && | ||||
|               state.myGroupsList.length > 0 | ||||
|             " | ||||
|           > | ||||
|             <div | ||||
|               class="groups-list-each" | ||||
|               v-for="(item, index) in state.myGroupsList" | ||||
|               :key="index" | ||||
|             > | ||||
|               <div class="groups-info"> | ||||
|                 <avatarModule | ||||
|                   :mode="2" | ||||
|                   :avatar="item?.avatar" | ||||
|                   :groupType="Number(item?.group_type)" | ||||
|                   :customStyle="{ width: '72rpx', height: '72rpx' }" | ||||
|                 ></avatarModule> | ||||
|                 <span class="groups-name"> | ||||
|                   {{ item?.group_name }} | ||||
|                 </span> | ||||
|                 <span | ||||
|                   class="groups-type" | ||||
|                   :style="{ | ||||
|                     color: | ||||
|                       groupTypeMapping[item?.group_type]?.result_type_color, | ||||
|                     border: | ||||
|                       '1px solid' + | ||||
|                       groupTypeMapping[item?.group_type]?.result_type_color, | ||||
|                   }" | ||||
|                 > | ||||
|                   {{ groupTypeMapping[item?.group_type]?.result_type }} | ||||
|                 </span> | ||||
|               </div> | ||||
|               <div class="groups-btns"> | ||||
|                 <div class="groups-btns-each" @click="toGroupChat(item)"> | ||||
|                   <span>{{ $t('addressBook.btns.enterGroup') }}</span> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div | ||||
|             class="addressBook-noData" | ||||
|             v-if=" | ||||
|               !state.isLoadingData && | ||||
|               ((state.addressBookActiveTab === 'company' && | ||||
|                 state.myContractList.length === 0) || | ||||
|                 (state.addressBookActiveTab === 'friends' && | ||||
|                   state.myFriendsList.length === 0) || | ||||
|                 (state.addressBookActiveTab === 'groups' && | ||||
|                   state.myGroupsList.length === 0)) | ||||
|             " | ||||
|           > | ||||
|             <img src="@/static/image/search/search-no-data.png" /> | ||||
|             <span> | ||||
|               {{ | ||||
|                 searchVal | ||||
|                   ? $t('addFriend.message.notFindData') | ||||
|                   : $t('search.hint') | ||||
|               }} | ||||
|             </span> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <tm-modal | ||||
|         class="friendDeleteModal" | ||||
|         ref="friendDeleteModalRef" | ||||
|         :mask="true" | ||||
|         :okText="$t('ok')" | ||||
|         okColor="#46299d" | ||||
|         cancelColor="#999" | ||||
|         @ok="doDeleteFriend" | ||||
|         @cancel="hideDeleteModal" | ||||
|         :teleport="false" | ||||
|         titleStyle="font-size: 40rpx;font-weight: 600;line-height: 1;" | ||||
|         :height="300" | ||||
|         :round="4" | ||||
|       > | ||||
|         <div class="friendDeleteModal-content"> | ||||
|           <span>{{ $t('addressBook.message.doOrNotDeleteFriend') }}</span> | ||||
|         </div> | ||||
|       </tm-modal> | ||||
|     </zPaging> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import customInput from '@/components/custom-input/custom-input.vue' | ||||
| import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import avatarModule from '@/components/avatar-module/index.vue' | ||||
| 
 | ||||
| import { ref, onMounted, reactive, watch } from 'vue' | ||||
| import { onLoad } from '@dcloudio/uni-app' | ||||
| import { handleSetWebviewStyle, handleFindWebview } from '@/utils/common' | ||||
| import { ServeUserGroupChatList, ServeCreateTalkList } from '@/api/chat/index' | ||||
| import { ServeGetSessionId } from '@/api/search/index' | ||||
| import { formatTalkItem } from '@/utils/talk' | ||||
| import { useDialogueStore, useTalkStore } from '@/store' | ||||
| import { | ||||
|   ServeQueryFriendsList, | ||||
|   ServeDeleteFriend, | ||||
| } from '@/api/addressBook/index' | ||||
| 
 | ||||
| import { useI18n } from 'vue-i18n' | ||||
| const { t } = useI18n() | ||||
| 
 | ||||
| const navshow = ref(false) | ||||
| const searchVal = ref('') | ||||
| const friendDeleteModalRef = ref(null) | ||||
| 
 | ||||
| const state = reactive({ | ||||
|   addressBookTabs: [], //tab页 | ||||
|   addressBookActiveTab: 'company', //当前选中的通讯录tab | ||||
|   myContractListPage: 1, //当前查询的我的组织架构通讯录列表分页 | ||||
|   myContractListPageSize: 10, //我的组织架构通讯录列表单页数量 | ||||
|   myContractList: [], //我的组织架构通讯录列表 | ||||
|   hasMoreContracts: true, //是否还有更多我的组织架构通讯录数据 | ||||
|   myFriendsListPage: 1, //当前查询的我的好友列表分页 | ||||
|   myFriendsListPageSize: 10, //我的好友列表单页数量 | ||||
|   myFriendsList: [], //我的好友列表 | ||||
|   hasMoreFriends: true, //是否还有更多我的好友数据 | ||||
|   myGroupsListPage: 1, //当前查询的群组列表分页 | ||||
|   myGroupsListPageSize: 10, //群组列表单页数量 | ||||
|   myGroupsList: [], //我的群组列表 | ||||
|   hasMoreGroups: true, //是否还有更多我的群组数据 | ||||
|   myCompany: '', //当前登录人公司别 | ||||
|   swipeAction: [], //左滑操作栏按钮组 | ||||
|   isLoadingData: true, //是否正在加载数据 | ||||
| }) | ||||
| 
 | ||||
| onLoad((options) => { | ||||
|   if (options.type) { | ||||
|     navshow.value = true | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const goWebHome = () => { | ||||
|   uni.navigateBack() | ||||
|   handleFindWebview(`handleBackHost()`) | ||||
| } | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   state.swipeAction = [ | ||||
|     { | ||||
|       text: t('addressBook.btns.delete'), //操作按钮的文本, | ||||
|     }, | ||||
|   ] | ||||
|   state.addressBookTabs = [ | ||||
|     { | ||||
|       key: 'company', | ||||
|       title: t('addressBook.tabs.company'), | ||||
|     }, | ||||
|     { | ||||
|       key: 'friends', | ||||
|       title: t('addressBook.tabs.friends'), | ||||
|     }, | ||||
|     { | ||||
|       key: 'groups', | ||||
|       title: t('addressBook.tabs.groups'), | ||||
|     }, | ||||
|   ] | ||||
|   handleSetWebviewStyle() | ||||
|   getMyContractList() | ||||
| }) | ||||
| 
 | ||||
| //输入搜索内容 | ||||
| const inputSearchText = (e) => { | ||||
|   searchVal.value = e | ||||
| } | ||||
| 
 | ||||
| //切换通讯录tab | ||||
| const updateAddressBookTab = (e) => { | ||||
|   state.addressBookActiveTab = e | ||||
| } | ||||
| 
 | ||||
| //获取“组织架构”通讯录列表 | ||||
| const getMyContractList = () => { | ||||
|   let params = { | ||||
|     type: 'addressBook', //查我的通讯录的时候写死addressBook | ||||
|     page: state.myContractListPage, | ||||
|     page_size: state.myContractListPageSize, | ||||
|     name: searchVal.value, | ||||
|   } | ||||
|   // console.log(params) | ||||
|   state.isLoadingData = true | ||||
|   ServeQueryFriendsList(params) | ||||
|     .then((res) => { | ||||
|       // console.log(res) | ||||
|       state.isLoadingData = false | ||||
|       if (res?.code === 200) { | ||||
|         state.myCompany = res.data?.company_name | ||||
|         if (state.myContractListPage === 1) { | ||||
|           state.myContractList = res.data?.user_list || [] | ||||
|         } else { | ||||
|           state.myContractList = state.myContractList.concat( | ||||
|             res.data?.user_list || [], | ||||
|           ) | ||||
|         } | ||||
|         if (state.myContractList.length < res.data?.count) { | ||||
|           state.hasMoreContracts = true | ||||
|         } else { | ||||
|           state.hasMoreContracts = false | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|     .catch((err) => { | ||||
|       state.isLoadingData = false | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| //获取“我的好友”列表 | ||||
| const getMyFriendsList = () => { | ||||
|   let params = { | ||||
|     type: 'myFriends', //查我得好友的时候写死myFriends | ||||
|     page: state.myFriendsListPage, | ||||
|     page_size: state.myFriendsListPageSize, | ||||
|     name: searchVal.value, | ||||
|   } | ||||
|   console.error(params) | ||||
|   state.isLoadingData = true | ||||
|   ServeQueryFriendsList(params) | ||||
|     .then((res) => { | ||||
|       console.log(res) | ||||
|       state.isLoadingData = false | ||||
|       if (res?.code === 200) { | ||||
|         if (state.myFriendsListPage === 1) { | ||||
|           state.myFriendsList = res.data?.user_list || [] | ||||
|         } else { | ||||
|           state.myFriendsList = state.myFriendsList.concat( | ||||
|             res.data?.user_list || [], | ||||
|           ) | ||||
|         } | ||||
|         if (state.myFriendsList.length < res.data?.count) { | ||||
|           state.hasMoreFriends = true | ||||
|         } else { | ||||
|           state.hasMoreFriends = false | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|     .catch((err) => { | ||||
|       state.isLoadingData = false | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| //点击进入用户详情页面 | ||||
| const toUserDetail = (userInfo) => { | ||||
|   uni.navigateTo({ | ||||
|     url: | ||||
|       '/pages/dialog/dialogDetail/userDetail??erpUserId=' + | ||||
|       userInfo.erp_user_id, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //获取“我的群组”列表 | ||||
| const getMyGroupsList = () => { | ||||
|   let params = { | ||||
|     page: state.myGroupsListPage, | ||||
|     page_size: state.myGroupsListPageSize, | ||||
|     group_name: searchVal.value, | ||||
|   } | ||||
|   state.isLoadingData = true | ||||
|   ServeUserGroupChatList(params) | ||||
|     .then((res) => { | ||||
|       console.log(res) | ||||
|       state.isLoadingData = false | ||||
|       if (res?.code === 200) { | ||||
|         if (state.myGroupsListPage === 1) { | ||||
|           state.myGroupsList = res.data?.items || [] | ||||
|         } else { | ||||
|           state.myGroupsList = state.myGroupsList.concat(res.data?.items || []) | ||||
|         } | ||||
|         if (state.myGroupsList.length < res.data?.total) { | ||||
|           state.hasMoreGroups = true | ||||
|         } else { | ||||
|           state.hasMoreGroups = false | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
|     .catch((err) => { | ||||
|       state.isLoadingData = false | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| //加载更多数据 | ||||
| const doLoadMore = (e) => { | ||||
|   if (state.addressBookActiveTab === 'company' && state.hasMoreContracts) { | ||||
|     state.myContractListPage += 1 | ||||
|     getMyContractList() | ||||
|   } else if (state.addressBookActiveTab === 'friends' && state.hasMoreFriends) { | ||||
|     state.myFriendsListPage += 1 | ||||
|     getMyFriendsList() | ||||
|   } else if (state.addressBookActiveTab === 'groups' && state.hasMoreGroups) { | ||||
|     state.myGroupsListPage += 1 | ||||
|     getMyGroupsList() | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| //获取会话Id | ||||
| const getSessionId = (talk_type, receiver_id) => { | ||||
|   return new Promise((resolve, reject) => { | ||||
|     let params = { | ||||
|       talkType: talk_type, | ||||
|       receiverId: receiver_id, | ||||
|     } | ||||
|     const resp = ServeGetSessionId(params) | ||||
|     console.log(resp) | ||||
|     resp.then(({ code, data }) => { | ||||
|       console.log(data) | ||||
|       if (code == 200) { | ||||
|         resolve(data?.sessionId) | ||||
|       } else { | ||||
|       } | ||||
|     }) | ||||
|     resp.catch(() => {}) | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //点击跳转进入我的指定群聊 | ||||
| const toGroupChat = async (groupInfo) => { | ||||
|   let talk_type = 2 | ||||
|   let receiver_id = groupInfo.id | ||||
|   const sessionId = await getSessionId(talk_type, receiver_id) | ||||
|   if (useTalkStore().findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { | ||||
|     ServeCreateTalkList({ | ||||
|       talk_type, | ||||
|       receiver_id, | ||||
|     }).then(async ({ code, data }) => { | ||||
|       if (code == 200) { | ||||
|         let item = formatTalkItem(data) | ||||
|         useTalkStore().addItem(item) | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
|   useDialogueStore().setDialogue({ | ||||
|     name: groupInfo.group_name, | ||||
|     talk_type: 2, | ||||
|     receiver_id: receiver_id, | ||||
|   }) | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/dialog/index?sessionId=' + sessionId, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //显示删除好友确认框 | ||||
| const showDeleteModal = (userInfo, friendIndex) => { | ||||
|   friendDeleteModalRef.value.open({ | ||||
|     userInfo, | ||||
|     friendIndex, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //关闭删除好友确认框 | ||||
| const hideDeleteModal = () => { | ||||
|   friendDeleteModalRef.value.close() | ||||
| } | ||||
| 
 | ||||
| //删除好友 | ||||
| const doDeleteFriend = (args) => { | ||||
|   console.log(args.userInfo, args.friendIndex) | ||||
|   let params = { | ||||
|     receiver_id: args.userInfo.id, //聊天的用户id | ||||
|     talk_type: 1, | ||||
|   } | ||||
|   ServeDeleteFriend(params).then((res) => { | ||||
|     console.log(res) | ||||
|     if (res?.code === 200) { | ||||
|       message.success(t('addressBook.message.deleteSuccess') + ' !') | ||||
|       state.myFriendsList.splice(args.friendIndex, 1) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| // 映射表-根据groupType设置对应值 | ||||
| const groupTypeMapping = { | ||||
|   0: {}, | ||||
|   1: {}, | ||||
|   2: { | ||||
|     result_type: t('index.mine.department'), | ||||
|     result_type_color: '#377EC6', | ||||
|   }, | ||||
|   3: { | ||||
|     result_type: t('index.mine.project'), | ||||
|     result_type_color: '#C1681C', | ||||
|   }, | ||||
|   4: { | ||||
|     result_type: t('index.type.company'), | ||||
|     result_type_color: '#7A58DE', | ||||
|   }, | ||||
| } | ||||
| 
 | ||||
| watch( | ||||
|   () => state.addressBookActiveTab, | ||||
|   (newVal) => { | ||||
|     if (newVal === 'company') { | ||||
|       state.myContractListPage = 1 | ||||
|       getMyContractList() | ||||
|     } else if (newVal === 'friends') { | ||||
|       state.myFriendsListPage = 1 | ||||
|       getMyFriendsList() | ||||
|     } else if (newVal === 'groups') { | ||||
|       state.myGroupsListPage = 1 | ||||
|       getMyGroupsList() | ||||
|     } | ||||
|   }, | ||||
| ) | ||||
| 
 | ||||
| watch( | ||||
|   () => searchVal.value, | ||||
|   (newVal) => { | ||||
|     if (state.addressBookActiveTab === 'company') { | ||||
|       state.myContractListPage = 1 | ||||
|       getMyContractList() | ||||
|     } else if (state.addressBookActiveTab === 'friends') { | ||||
|       state.myFriendsListPage = 1 | ||||
|       getMyFriendsList() | ||||
|     } else if (state.addressBookActiveTab === 'groups') { | ||||
|       state.myGroupsListPage = 1 | ||||
|       getMyGroupsList() | ||||
|     } | ||||
|   }, | ||||
| ) | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| ::v-deep .zp-paging-container-content { | ||||
|   height: 100%; | ||||
|   display: flex; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .tmicon-angle-left { | ||||
|   color: #fff !important; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .text-weight-b { | ||||
|   color: #fff !important; | ||||
| } | ||||
| 
 | ||||
| ::v-deep .index_top_navbar .statusHeightTop > .noNvueBorder:first-child { | ||||
|   background: transparent !important; | ||||
|   border: none !important; | ||||
| } | ||||
| 
 | ||||
| .top_bg { | ||||
|   background: url('@/static/image/mine/page_top.png') no-repeat; | ||||
|   background-size: cover; | ||||
|   background-position: bottom center; | ||||
| } | ||||
| 
 | ||||
| :deep(.animateAll_tabs_tmui) { | ||||
|   width: 120rpx !important; | ||||
|   height: 10rpx !important; | ||||
| } | ||||
| 
 | ||||
| .address-book-page { | ||||
|   .address-book { | ||||
|     flex: 1; | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|     background-size: cover; | ||||
|     background-position: bottom center; | ||||
|     background-attachment: fixed; | ||||
|     width: 100%; | ||||
| 
 | ||||
|     .address-book-tabs-panes-list { | ||||
|       margin: 30rpx 24rpx; | ||||
|       overflow: hidden; | ||||
|       flex: 1; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
| 
 | ||||
|       .tabs-panes-each { | ||||
|         gap: 30rpx 0; | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: flex-start; | ||||
|         justify-content: center; | ||||
|       } | ||||
|       .address-book-company, | ||||
|       .address-book-friends { | ||||
|         .address-book-company-name { | ||||
|           margin: 0 0 -16rpx; | ||||
|           span { | ||||
|             font-size: 26rpx; | ||||
|             font-weight: 400; | ||||
|             line-height: 1; | ||||
|             color: #999; | ||||
|           } | ||||
|         } | ||||
|         .members-list-each { | ||||
|           display: flex; | ||||
|           flex-direction: column; | ||||
|           align-items: flex-start; | ||||
|           justify-content: center; | ||||
|           background-color: #fff; | ||||
|           width: 100%; | ||||
|           border-radius: 8rpx; | ||||
| 
 | ||||
|           .swipe-action { | ||||
|             width: 100%; | ||||
|             overflow: visible !important; | ||||
| 
 | ||||
|             :deep(.wd-swipe-action__right) { | ||||
|               display: flex; | ||||
|               flex-direction: row; | ||||
|               align-items: center; | ||||
|               justify-content: center; | ||||
|               background-color: #cf3050; | ||||
|               padding: 0 48rpx; | ||||
|               border-radius: 0 8rpx 8rpx 0; | ||||
|               span { | ||||
|                 color: #fff; | ||||
|                 line-height: 1; | ||||
|                 font-size: 30rpx; | ||||
|                 font-weight: 400; | ||||
|               } | ||||
|             } | ||||
|           } | ||||
| 
 | ||||
|           .members-info { | ||||
|             display: flex; | ||||
|             flex-direction: row; | ||||
|             align-items: flex-start; | ||||
|             justify-content: flex-start; | ||||
|             gap: 16rpx; | ||||
|             padding: 24rpx 24rpx 0; | ||||
|             width: 100%; | ||||
| 
 | ||||
|             .members-info-area { | ||||
|               display: flex; | ||||
|               flex-direction: row; | ||||
|               align-items: flex-start; | ||||
|               justify-content: flex-start; | ||||
|               gap: 16rpx; | ||||
| 
 | ||||
|               .members-info-basic { | ||||
|                 display: flex; | ||||
|                 flex-direction: column; | ||||
|                 align-items: flex-start; | ||||
|                 justify-content: center; | ||||
| 
 | ||||
|                 .members-name { | ||||
|                   font-size: 30rpx; | ||||
|                   font-weight: 500; | ||||
|                   width: 150rpx; | ||||
|                   word-break: break-all; | ||||
|                 } | ||||
| 
 | ||||
|                 .members-jobNum { | ||||
|                   font-size: 26rpx; | ||||
|                   font-weight: 400; | ||||
|                   color: #999; | ||||
|                   word-break: break-all; | ||||
|                   margin: 10rpx 0 0; | ||||
|                 } | ||||
| 
 | ||||
|                 .members-company { | ||||
|                   font-size: 24rpx; | ||||
|                   font-weight: 400; | ||||
|                   line-height: 1; | ||||
|                   color: #999; | ||||
|                 } | ||||
|               } | ||||
|               .members-positions-area { | ||||
|                 .members-positions { | ||||
|                   display: flex; | ||||
|                   flex-direction: row; | ||||
|                   align-items: center; | ||||
|                   justify-content: flex-start; | ||||
|                   flex-wrap: wrap; | ||||
|                   gap: 10rpx; | ||||
|                   padding: 4rpx 0 0; | ||||
|                   max-height: 84rpx; | ||||
|                   overflow: hidden; | ||||
|                   -webkit-box-orient: vertical; | ||||
|                   -webkit-line-clamp: 2; | ||||
|                   line-clamp: 2; | ||||
| 
 | ||||
|                   .members-positions-each { | ||||
|                     background-color: #eee9f8; | ||||
|                     line-height: 1; | ||||
|                     font-size: 24rpx; | ||||
|                     padding: 4rpx 16rpx; | ||||
|                     color: #46299d; | ||||
|                   } | ||||
|                 } | ||||
|                 .members-positions-popover-box { | ||||
|                   display: flex; | ||||
|                   flex-direction: column; | ||||
|                   align-items: center; | ||||
|                   justify-content: center; | ||||
|                   gap: 10rpx 0; | ||||
| 
 | ||||
|                   .members-positions-popover { | ||||
|                     background-color: #eee9f8; | ||||
|                     line-height: 1; | ||||
|                     font-size: 24rpx; | ||||
|                     padding: 8rpx 16rpx; | ||||
|                     color: #46299d; | ||||
|                     width: 100%; | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|           .company-infos { | ||||
|             margin: 0 0 0 88rpx; | ||||
|             padding: 0 0 24rpx 24rpx; | ||||
|             .company-each { | ||||
|               span { | ||||
|                 font-size: 24rpx; | ||||
|                 font-weight: 400; | ||||
|                 line-height: 1; | ||||
|                 color: #999; | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       .address-book-groups { | ||||
|         .groups-list-each { | ||||
|           display: flex; | ||||
|           flex-direction: row; | ||||
|           align-items: center; | ||||
|           justify-content: space-between; | ||||
|           background-color: #fff; | ||||
|           width: 100%; | ||||
|           padding: 24rpx; | ||||
|           border-radius: 8rpx; | ||||
| 
 | ||||
|           .groups-info { | ||||
|             display: flex; | ||||
|             flex-direction: row; | ||||
|             align-items: center; | ||||
|             justify-content: flex-start; | ||||
|             gap: 16rpx; | ||||
|             .groups-name { | ||||
|               font-size: 30rpx; | ||||
|               font-weight: 500; | ||||
|             } | ||||
|             .groups-type { | ||||
|               font-size: 24rpx; | ||||
|               border-radius: 6rpx; | ||||
|               font-weight: bold; | ||||
|               padding: 6rpx 12rpx; | ||||
|               flex-shrink: 0; | ||||
|             } | ||||
|           } | ||||
|           .groups-btns { | ||||
|             flex-shrink: 0; | ||||
|             margin: 0 0 0 16rpx; | ||||
|             .groups-btns-each { | ||||
|               display: flex; | ||||
|               flex-direction: row; | ||||
|               align-items: center; | ||||
|               justify-content: center; | ||||
|               background-color: #46299d; | ||||
|               padding: 16rpx; | ||||
|               border-radius: 8rpx; | ||||
| 
 | ||||
|               span { | ||||
|                 color: #fff; | ||||
|                 font-size: 24rpx; | ||||
|                 font-weight: 400; | ||||
|                 line-height: 1; | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       .addressBook-noData { | ||||
|         margin: 30rpx 24rpx; | ||||
|         overflow: hidden; | ||||
|         flex: 1; | ||||
|         gap: 30rpx 0; | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         justify-content: center; | ||||
|         img { | ||||
|           width: 500rpx; | ||||
|         } | ||||
|         span { | ||||
|           font-size: 30rpx; | ||||
|           font-weight: 400; | ||||
|           line-height: 1; | ||||
|           color: #999; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   .friendDeleteModal { | ||||
|     :deep(.overlay) { | ||||
|       height: 100vh !important; | ||||
|     } | ||||
|     .friendDeleteModal-content { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -53,7 +53,7 @@ | ||||
|     > | ||||
|       <div class="group-member-each"> | ||||
|         <div class="group-member-avatar" :style="{ background: 'unset' }"> | ||||
|           <img src="@/static/image/chatSettings/add-member.png" /> | ||||
|           <img src="/src/static/image/chatSettings/add-member.png" /> | ||||
|         </div> | ||||
|         <div class="group-member-name"> | ||||
|           <span class="text-[24rpx] font-regular">添加</span> | ||||
| @ -71,7 +71,7 @@ | ||||
|     > | ||||
|       <div class="group-member-each"> | ||||
|         <div class="group-member-avatar" :style="{ background: 'unset' }"> | ||||
|           <img src="@/static/image/chatSettings/remove-member.png" /> | ||||
|           <img src="/src/static/image/chatSettings/remove-member.png" /> | ||||
|         </div> | ||||
|         <div class="group-member-name"> | ||||
|           <span class="text-[24rpx] font-regular">移除</span> | ||||
|  | ||||
| @ -49,7 +49,7 @@ | ||||
|           props?.memberItem?.is_mine && props?.manageType === 'removeMembers' | ||||
|         " | ||||
|       > | ||||
|         <img src="@/static/image/chatSettings/is-mine.png" /> | ||||
|         <img src="/src/static/image/chatSettings/is-mine.png" /> | ||||
|       </div> | ||||
|       <div | ||||
|         class="is-admin-tag" | ||||
|  | ||||
| @ -16,10 +16,9 @@ | ||||
|             props?.item?.hasPointer && | ||||
|             (props?.isManager || | ||||
|               (!props?.isManager && | ||||
|                 (props?.item?.label !== $t('chat.settings.groupName') || | ||||
|                   props?.item?.label !== $t('chat.settings.groupType')))) | ||||
|                 props?.item?.label !== $t('chat.settings.groupName'))) | ||||
|           " | ||||
|           src="@/static/image/chatSettings/pointer.png" | ||||
|           src="/src/static/image/chatSettings/pointer.png" | ||||
|         /> | ||||
|         <tm-switch | ||||
|           :width="88" | ||||
| @ -28,8 +27,8 @@ | ||||
|           barIcon="" | ||||
|           color="#46299D" | ||||
|           unCheckedColor="#EEEEEE" | ||||
|           :modelValue="props?.item?.label === t('chat.settings.openReminder') ? openReminderStatus : modelValue" | ||||
|           :defaultValue="props?.item?.label === t('chat.settings.openReminder') ? openReminderStatus : modelValue" | ||||
|           :modelValue="modelValue" | ||||
|           :defaultValue="modelValue" | ||||
|           @change="changeSwitch($event, props?.item)" | ||||
|         ></tm-switch> | ||||
|       </div> | ||||
| @ -46,8 +45,8 @@ | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import { defineProps, defineEmits, computed, ref, onMounted, watch } from 'vue' | ||||
| import { ServeContactRobotQuery } from '@/api/chat' | ||||
| import { defineProps, defineEmits, computed } from 'vue' | ||||
| 
 | ||||
| import { useI18n } from 'vue-i18n' | ||||
| const { t } = useI18n() | ||||
| 
 | ||||
| @ -103,32 +102,6 @@ const modelValue = computed(() => { | ||||
|   return switchStatus | ||||
| }) | ||||
| 
 | ||||
| // 仅用于“拒收消息”场景的本地状态 | ||||
| const openReminderStatus = ref(false) | ||||
| const loadRefuseStatus = async () => { | ||||
|   try { | ||||
|     const res = await ServeContactRobotQuery() | ||||
|     openReminderStatus.value = !!(res?.data?.Result) | ||||
|   } catch (err) { | ||||
|     openReminderStatus.value = false | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   if (props?.item?.label === t('chat.settings.openReminder')) { | ||||
|     loadRefuseStatus() | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| watch( | ||||
|   () => props?.item?.label, | ||||
|   (newLabel) => { | ||||
|     if (newLabel === t('chat.settings.openReminder')) { | ||||
|       loadRefuseStatus() | ||||
|     } | ||||
|   } | ||||
| ) | ||||
| 
 | ||||
| //切换开关选择 | ||||
| const changeSwitch = (e, item) => { | ||||
|   emits('changeSwitch', e, item.label) | ||||
|  | ||||
| @ -22,7 +22,6 @@ | ||||
|         <customBtn | ||||
|           :btnText="$t('button.text.edit')" | ||||
|           @click="editAvatar" | ||||
|           :isLoading="state.isLoading" | ||||
|         ></customBtn> | ||||
|       </ZPaging> | ||||
|     </div> | ||||
| @ -50,10 +49,6 @@ const dialogueParams = reactive({ | ||||
|   receiver_id: computed(() => dialogueStore.talk.receiver_id), | ||||
| }) | ||||
| 
 | ||||
| const state = reactive({ | ||||
|   isLoading: false, //是否正在加载中 | ||||
| }) | ||||
| 
 | ||||
| const onProgressFn = (progress, id) => { | ||||
|   console.log((progress.loaded / progress.total) * 100, 'progress') | ||||
| 
 | ||||
| @ -74,7 +69,6 @@ const editAvatar = () => { | ||||
|     count: 1, | ||||
|     success: async (res) => { | ||||
|       console.log(res, 'res') | ||||
|       state.isLoading = true | ||||
|       res.tempFiles.forEach(async (file) => { | ||||
|         console.log(file) | ||||
|         let image = new Image() | ||||
| @ -99,7 +93,6 @@ const editAvatar = () => { | ||||
|                 const resp = ServeEditGroup(params) | ||||
|                 resp.then(({ code }) => { | ||||
|                   if (code == 200) { | ||||
|                     state.isLoading = false | ||||
|                     groupStore.updateGroupInfo({ | ||||
|                       avatar: data.ori_url, | ||||
|                     }) | ||||
| @ -107,18 +100,13 @@ const editAvatar = () => { | ||||
|                     //   delta: 1, | ||||
|                     // }) | ||||
|                   } else { | ||||
|                     state.isLoading = false | ||||
|                   } | ||||
|                 }) | ||||
|                 resp.catch(() => { | ||||
|                   state.isLoading = false}) | ||||
|                 resp.catch(() => {}) | ||||
|               } else { | ||||
|                 state.isLoading = false | ||||
|               } | ||||
|             }, | ||||
|           ).catch(() => { | ||||
|             state.isLoading = false | ||||
|           }) | ||||
|           ) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
|  | ||||
| @ -35,7 +35,7 @@ | ||||
|               <img | ||||
|                 v-if="state.groupName" | ||||
|                 class="groupName-input-clearBtn" | ||||
|                 src="@/static/image/chatSettings/clear-btn.png" | ||||
|                 src="/src/static/image/chatSettings/clear-btn.png" | ||||
|                 @click="clearGroupNameInput" | ||||
|               /> | ||||
|             </div> | ||||
|  | ||||
| @ -46,7 +46,7 @@ | ||||
|               <div class="group-admin-list-each-btns"> | ||||
|                 <img | ||||
|                   v-if="item.is_mine" | ||||
|                   src="@/static/image/chatSettings/is-mine.png" | ||||
|                   src="/src/static/image/chatSettings/is-mine.png" | ||||
|                 /> | ||||
|                 <div | ||||
|                   class="group-admin-list-each-btns-each" | ||||
| @ -67,7 +67,7 @@ | ||||
|               margin: state?.groupAdminList.length == 0 ? '20rpx 0 0' : '', | ||||
|             }" | ||||
|           > | ||||
|             <img src="@/static/image/chatSettings/add-btn.png" /> | ||||
|             <img src="/src/static/image/chatSettings/add-btn.png" /> | ||||
|             <span class="text-[28rpx] font-medium"> | ||||
|               {{ $t('chat.manage.addAdmin') }} | ||||
|             </span> | ||||
|  | ||||
| @ -1,76 +0,0 @@ | ||||
| <template> | ||||
|   <div class="outer-layer manage-group-deps-page"> | ||||
|     <div class="root"> | ||||
|       <ZPaging | ||||
|         ref="zPaging" | ||||
|         :show-scrollbar="false" | ||||
|         :use-virtual-list="true" | ||||
|         :virtual-list-col="5" | ||||
|         :auto="false" | ||||
|         :refresher-enabled="false" | ||||
|         :loading-more-enabled="false" | ||||
|       > | ||||
|         <template #top> | ||||
|           <customNavbar :title="$t('pageTitle.view.deps')"></customNavbar> | ||||
|         </template> | ||||
|         <div class="group-deps-list"> | ||||
|           <div | ||||
|             class="group-deps-list-each" | ||||
|             v-for="item in groupParams.groupInfo.deptInfos" | ||||
|             :key="item.dept_id" | ||||
|           > | ||||
|             <span>{{ item.dept_name }}</span> | ||||
|           </div> | ||||
|         </div> | ||||
|       </ZPaging> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import settingFormItem from '../components/settingFormItem.vue' | ||||
| import ZPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import { onLoad } from '@dcloudio/uni-app' | ||||
| import { computed, onMounted, reactive } from 'vue' | ||||
| 
 | ||||
| import { useGroupStore } from '@/store' | ||||
| 
 | ||||
| import { useI18n } from 'vue-i18n' | ||||
| const { t } = useI18n() | ||||
| 
 | ||||
| const groupStore = useGroupStore() | ||||
| const groupParams = reactive({ | ||||
|   groupInfo: computed(() => groupStore.groupInfo), | ||||
| }) | ||||
| 
 | ||||
| const state = reactive({}) | ||||
| 
 | ||||
| onLoad((options) => { | ||||
|   console.log(options) | ||||
| }) | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   console.log(groupParams.groupInfo.deptInfos) | ||||
| }) | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .outer-layer { | ||||
|   flex: 1; | ||||
|   background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|   background-size: cover; | ||||
|   background-repeat: no-repeat; | ||||
| } | ||||
| .group-deps-list { | ||||
|   margin: 20rpx 32rpx; | ||||
|   background-color: #fff; | ||||
|   .group-deps-list-each { | ||||
|     padding: 34rpx 32rpx; | ||||
|     border-bottom: 1px solid $theme-border-color; | ||||
|     span { | ||||
|       font-size: 28rpx; | ||||
|       font-weight: 500; | ||||
|       color: #333; | ||||
|       line-height: 40rpx; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -42,7 +42,7 @@ | ||||
|             class="add-silence-member-btn chat-settings-card" | ||||
|             @click="toSelectMembersPage" | ||||
|           > | ||||
|             <img src="@/static/image/chatSettings/add-btn.png" /> | ||||
|             <img src="/src/static/image/chatSettings/add-btn.png" /> | ||||
|             <span class="text-[28rpx] font-medium"> | ||||
|               {{ $t('chat.manage.addSilenceMember') }} | ||||
|             </span> | ||||
|  | ||||
| @ -17,17 +17,10 @@ | ||||
|       </div> | ||||
|       <div class="avatarImg"> | ||||
|         <avatarModule | ||||
|           :mode="props?.data?.group_type === 0 ? 1 : 2" | ||||
|           :mode="2" | ||||
|           :avatar="props?.data?.avatar" | ||||
|           :groupType="props?.data?.group_type" | ||||
|           :userName="props?.data?.name" | ||||
|           :customStyle="{ width: '96rpx', height: '96rpx' }" | ||||
|           :customTextStyle="{ | ||||
|             fontSize: '32rpx', | ||||
|             fontWeight: 'bold', | ||||
|             color: '#fff', | ||||
|             lineHeight: '44rpx', | ||||
|           }" | ||||
|         ></avatarModule> | ||||
|       </div> | ||||
|       <div class="chatInfo"> | ||||
| @ -37,18 +30,16 @@ | ||||
|               class="text-[#171717] text-[32rpx] font-medium leading-[44rpx]" | ||||
|             > | ||||
|               <span>{{ props.data.name }}</span> | ||||
|               <span v-if="props.data.talk_type === 2"> | ||||
|                 ({{ props.data.group_member_num }}) | ||||
|               </span> | ||||
|               <span v-if="props.data.talk_type === 2">({{ props.data.group_member_num }})</span> | ||||
|               <span v-if="props.data.group_type === 2" class="depTag tag"> | ||||
|                 部门 | ||||
|               </span> | ||||
|               <span v-if="props.data.group_type === 3" class="projectTag tag"> | ||||
|                 项目 | ||||
|               </span> | ||||
|               <span v-if="props.data.group_type === 4" class="companyTag tag"> | ||||
|                 公司 | ||||
|               </span> | ||||
|                   部门 | ||||
|                 </span> | ||||
|                 <span v-if="props.data.group_type === 3" class="projectTag tag"> | ||||
|                   项目 | ||||
|                 </span> | ||||
|                 <span v-if="props.data.group_type === 4" class="companyTag tag"> | ||||
|                   公司 | ||||
|                 </span> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
| @ -136,7 +127,7 @@ const cellClick = () => { | ||||
|   opacity: 40%; | ||||
| } | ||||
| 
 | ||||
| .tag { | ||||
| .tag{ | ||||
|   display: inline-flex; | ||||
|   align-items: center; | ||||
|   text-align: center; | ||||
| @ -151,16 +142,16 @@ const cellClick = () => { | ||||
|   font-weight: bold; | ||||
| } | ||||
| .companyTag { | ||||
|   border: 1px solid #7a58de; | ||||
|   color: #7a58de; | ||||
| 	border: 1px solid #7a58de; | ||||
| 	color: #7a58de; | ||||
| } | ||||
| .depTag { | ||||
|   border: 1px solid #377ec6; | ||||
|   color: #377ec6; | ||||
| 	border: 1px solid #377ec6; | ||||
| 	color: #377ec6; | ||||
| } | ||||
| .projectTag { | ||||
|   border: 1px solid #c1681c; | ||||
|   color: #c1681c; | ||||
| 	border: 1px solid #c1681c; | ||||
| 	color: #c1681c; | ||||
| } | ||||
| 
 | ||||
| .textEllipsis { | ||||
|  | ||||
| @ -219,7 +219,7 @@ watch( | ||||
| onMounted(() => { | ||||
|   talkStore.loadTalkList() | ||||
|   console.log(talkStore.talkItems) | ||||
|   items.value = lodash.cloneDeep(talkStore.talkItems).filter(item=>item.is_dismiss === 0 && item.is_quit === 0 && ((item.talk_type === 1 && item.receiver_id !== 2) || item.talk_type !== 1)) | ||||
|   items.value = lodash.cloneDeep(talkStore.talkItems) | ||||
| }) | ||||
| onUnmounted(() => { | ||||
|   dialogueStore.setForwardType('') | ||||
|  | ||||
| @ -96,7 +96,6 @@ | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div | ||||
|                 v-if="isCreateProjecy" | ||||
|                 @click="groupActiveIndex = 2;depCheckedKeys = [];" | ||||
|                 class="groupCard thirdPanel" | ||||
|                 :class="groupActiveIndex === 2 ? 'activePanel' : ''" | ||||
| @ -145,7 +144,6 @@ onUnload(()=> { | ||||
| 
 | ||||
| }) | ||||
| const isHasPermission = ref(false) | ||||
| const isCreateProjecy = ref(false) | ||||
| onShow( async() =>{ | ||||
|   const isHasRes = await userHasPermission({ | ||||
|     erpUserId: userInfo?.value?.ID, | ||||
| @ -160,7 +158,6 @@ onShow( async() =>{ | ||||
|     } else{ | ||||
|       isHasPermission.value = false | ||||
|     } | ||||
|     isCreateProjecy.value = isHasRes.data.btn_rule_create_project_group | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
|  | ||||
| @ -404,7 +404,7 @@ const getCurrentMembers = async (depItem) => { | ||||
|     departmentId: depItem.ID, | ||||
|     status: 'notactive', | ||||
|   }) | ||||
|   if (res.code === 200) { | ||||
|   if (res.status === 0) { | ||||
|     currentMembers.value = res.data.data.length | ||||
|       ? res.data.data.map((v) => { | ||||
|           return { | ||||
| @ -628,7 +628,7 @@ const handleConfirm = async () => { | ||||
|     departmentIds: allCheckedList.value.map((v) => v.ID), | ||||
|     status: 'notactive', | ||||
|   }) | ||||
|   if (res.code == 200 && res.data?.data?.length) { | ||||
|   if (res.status == 0 && res.data?.data?.length) { | ||||
|     res.data?.data.forEach((v) => { | ||||
|       listT.push(v) | ||||
|     }) | ||||
|  | ||||
| @ -1,366 +0,0 @@ | ||||
| <template> | ||||
|   <div class="outer-layer user-detail-page"> | ||||
|     <div class="root"> | ||||
|       <ZPaging ref="zPaging" :show-scrollbar="false"> | ||||
|         <template #top> | ||||
|           <customNavbar :title="$t('complaint.title')"></customNavbar> | ||||
|         </template> | ||||
|         <!-- 投诉主体内容 --> | ||||
|         <view class="complaint-container"> | ||||
|           <!-- 投诉类型选择 --> | ||||
|           <view class="form-item"> | ||||
|             <text class="form-label">{{ $t('complaint.selectType') }}</text> | ||||
|             <picker mode="selector" :range="complaintTypes" range-key="label" @change="handleTypeChange"> | ||||
|               <view class="picker"> | ||||
|                 {{ selectedType.label || $t('complaint.selectPlaceholder') }} | ||||
|                 <uni-icons type="arrowright" size="16" color="#999"></uni-icons> | ||||
|               </view> | ||||
|             </picker> | ||||
|           </view> | ||||
| 
 | ||||
|           <!-- 图片证据上传 --> | ||||
|           <view class="form-item"> | ||||
|             <text class="form-label">{{ $t('complaint.imageEvidence') }}</text> | ||||
|             <view class="upload-area"> | ||||
|               <view v-for="(img, index) in imageList" :key="index" class="image-wrapper"> | ||||
|                 <image :src="img" mode="aspectFill" class="uploaded-image" @click="previewImage(index)" /> | ||||
|                 <uni-icons type="close" size="18" color="#fff" class="delete-icon" | ||||
|                   @click="removeImage(index)"></uni-icons> | ||||
|               </view> | ||||
|               <view v-if="imageList.length < 9" class="upload-btn" @click="chooseImage"> | ||||
|                 <uni-icons type="plusempty" size="28" color="#999"></uni-icons> | ||||
|                 <text class="upload-text">{{ $t('complaint.addImage') }}</text> | ||||
|               </view> | ||||
|             </view> | ||||
|           </view> | ||||
| 
 | ||||
|           <!-- 投诉内容 --> | ||||
|           <view class="form-item"> | ||||
|             <text class="form-label">{{ $t('complaint.complaintContent') }}</text> | ||||
|             <textarea v-model="complaintContent" :placeholder="$t('complaint.contentPlaceholder')" | ||||
|               class="content-textarea"></textarea> | ||||
|           </view> | ||||
| 
 | ||||
|           <!-- 投诉须知 --> | ||||
|           <view class="notice-box"> | ||||
|             <view class="notice-header" @click="toggleNotice"> | ||||
|               <text class="notice-title">{{ $t('complaint.noticeTitle') }}</text> | ||||
|               <text class="toggle-btn"> | ||||
|                 {{ isNoticeExpanded ? $t('complaint.collapse') : $t('complaint.expand') }} | ||||
|               </text> | ||||
|             </view> | ||||
| 
 | ||||
|             <!-- 折叠状态只显示前两项 --> | ||||
|             <view class="notice-content" v-if="!isNoticeExpanded"> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticeone') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticetwo') }}</text> | ||||
|               </view> | ||||
|             </view> | ||||
| 
 | ||||
|             <!-- 展开状态显示全部 --> | ||||
|             <view class="notice-content" v-else> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticeone') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticetwo') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticethree') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticefour') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticefive') }}</text> | ||||
|               </view> | ||||
|               <view class="notice-item"> | ||||
|                 <text>{{ $t('complaint.noticenoticeContent') }}</text> | ||||
|               </view> | ||||
|             </view> | ||||
|           </view> | ||||
|           <button class="submit-btn" :disabled="!selectedType.value" @click="handleSubmit"> | ||||
|             {{ $t('complaint.submit') }} | ||||
|           </button> | ||||
|         </view> | ||||
|       </ZPaging> | ||||
| 
 | ||||
|       <!-- 固定在底部的提交按钮 --> | ||||
| 
 | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| 
 | ||||
| <script setup> | ||||
|   import ZPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
|   import { | ||||
|     ref | ||||
|   } from 'vue'; | ||||
|   import { | ||||
|     onLoad | ||||
|   } from '@dcloudio/uni-app'; | ||||
|   import { | ||||
|     useI18n | ||||
|   } from 'vue-i18n'; | ||||
| 
 | ||||
|   const { | ||||
|     t | ||||
|   } = useI18n(); | ||||
| 
 | ||||
| 
 | ||||
|   // 投诉类型选项 | ||||
|   const complaintTypes = computed(() => [{ | ||||
|       label: t('complaint.typeOptions.porn'), | ||||
|       value: 'porn' | ||||
|     }, | ||||
|     { | ||||
|       label: t('complaint.typeOptions.illegal'), | ||||
|       value: 'illegal' | ||||
|     }, | ||||
|     { | ||||
|       label: t('complaint.typeOptions.gambling'), | ||||
|       value: 'gambling' | ||||
|     }, | ||||
|     { | ||||
|       label: t('complaint.typeOptions.violence'), | ||||
|       value: 'violence' | ||||
|     }, | ||||
|     { | ||||
|       label: t('complaint.typeOptions.selfHarm'), | ||||
|       value: 'selfHarm' | ||||
|     }, | ||||
|     { | ||||
|       label: t('complaint.typeOptions.other'), | ||||
|       value: 'other' | ||||
|     } | ||||
|   ]); | ||||
| 
 | ||||
|   // 表单数据 | ||||
|   const selectedType = ref({}); | ||||
|   const imageList = ref([]); | ||||
|   const complaintContent = ref(''); | ||||
|   const isNoticeExpanded = ref(false); | ||||
| 
 | ||||
|   // 切换投诉须知展开状态 | ||||
|   const toggleNotice = () => { | ||||
|     isNoticeExpanded.value = !isNoticeExpanded.value; | ||||
|   }; | ||||
| 
 | ||||
|   // 选择投诉类型 | ||||
|   const handleTypeChange = (e) => { | ||||
|     selectedType.value = complaintTypes.value[e.detail.value]; | ||||
|   }; | ||||
| 
 | ||||
|   // 选择图片 | ||||
|   const chooseImage = () => { | ||||
|     uni.chooseImage({ | ||||
|       count: 9 - imageList.value.length, | ||||
|       sizeType: ['compressed'], | ||||
|       sourceType: ['album', 'camera'], | ||||
|       success: (res) => { | ||||
|         imageList.value = [...imageList.value, ...res.tempFilePaths]; | ||||
|         if (imageList.value.length > 9) { | ||||
|           uni.showToast({ | ||||
|             title: `最多只能选择9张图片`, | ||||
|             icon: 'none' | ||||
|           }); | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|   }; | ||||
| 
 | ||||
|   // 删除图片 | ||||
|   const removeImage = (index) => { | ||||
|     imageList.value.splice(index, 1); | ||||
|   }; | ||||
| 
 | ||||
|   // 预览图片 | ||||
|   const previewImage = (index) => { | ||||
|     uni.previewImage({ | ||||
|       current: index, | ||||
|       urls: imageList.value | ||||
|     }); | ||||
|   }; | ||||
| 
 | ||||
|   // 提交投诉 | ||||
|   const handleSubmit = () => { | ||||
|     const formData = { | ||||
|       type: selectedType.value, | ||||
|       images: imageList.value, | ||||
|       content: complaintContent.value | ||||
|     }; | ||||
|     uni.showLoading({ | ||||
|       title: t('complaint.toast.submitting') // 使用国际化文本 | ||||
|     }); | ||||
|     setTimeout(() => { | ||||
|       uni.hideLoading(); | ||||
|       uni.showToast({ | ||||
|         title: t('complaint.toast.success'), | ||||
|         icon: 'success' | ||||
|       }); | ||||
|       selectedType.value = {}; | ||||
|       imageList.value = []; | ||||
|       complaintContent.value = ''; | ||||
|       // 返回上一页 | ||||
|       setTimeout(() => { | ||||
|         uni.navigateBack(); | ||||
|       }, 1500); | ||||
|     }, 2000); | ||||
|   }; | ||||
| 
 | ||||
|   onLoad(() => { | ||||
|     // 页面加载时初始化 | ||||
|   }); | ||||
| </script> | ||||
| 
 | ||||
| <style scoped lang="scss"> | ||||
|   ::v-deep .uni-picker-action-confirm { | ||||
|     color: #452aa1 !important; | ||||
|   } | ||||
| 
 | ||||
|   .outer-layer { | ||||
|     flex: 1; | ||||
|     background-image: url('@/static/image/mine/1111.png'); | ||||
|     background-size: cover; | ||||
|     background-repeat: no-repeat; | ||||
|   } | ||||
| 
 | ||||
|   .complaint-container { | ||||
|     padding: 20rpx 30rpx; | ||||
|     background-color: rgba(255, 255, 255, 0.9); | ||||
|     margin: 20rpx; | ||||
|     border-radius: 16rpx; | ||||
|     box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.05); | ||||
|   } | ||||
| 
 | ||||
|   .form-item { | ||||
|     margin-bottom: 40rpx; | ||||
|   } | ||||
| 
 | ||||
|   .form-label { | ||||
|     display: block; | ||||
|     font-size: 28rpx; | ||||
|     color: #333; | ||||
|     font-weight: 500; | ||||
|     margin-bottom: 20rpx; | ||||
|   } | ||||
| 
 | ||||
|   .picker { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: space-between; | ||||
|     padding: 24rpx; | ||||
|     background-color: #f7f7f7; | ||||
|     border-radius: 12rpx; | ||||
|     font-size: 28rpx; | ||||
|     color: #333; | ||||
|   } | ||||
| 
 | ||||
|   .upload-area { | ||||
|     display: flex; | ||||
|     flex-wrap: wrap; | ||||
|     gap: 20rpx; | ||||
|   } | ||||
| 
 | ||||
|   .upload-btn { | ||||
|     width: 160rpx; | ||||
|     height: 160rpx; | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|     background-color: #f7f7f7; | ||||
|     border-radius: 12rpx; | ||||
|     border: 1rpx dashed #ddd; | ||||
|   } | ||||
| 
 | ||||
|   .upload-text { | ||||
|     font-size: 24rpx; | ||||
|     color: #999; | ||||
|     margin-top: 10rpx; | ||||
|   } | ||||
| 
 | ||||
|   .image-wrapper { | ||||
|     width: 160rpx; | ||||
|     height: 160rpx; | ||||
|     position: relative; | ||||
|     border-radius: 12rpx; | ||||
|     overflow: hidden; | ||||
|   } | ||||
| 
 | ||||
|   .uploaded-image { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|   } | ||||
| 
 | ||||
|   .delete-icon { | ||||
|     position: absolute; | ||||
|     top: 8rpx; | ||||
|     right: 8rpx; | ||||
|     background-color: rgba(0, 0, 0, 0.5); | ||||
|     border-radius: 50%; | ||||
|     padding: 4rpx; | ||||
|   } | ||||
| 
 | ||||
|   .content-textarea { | ||||
|     width: 100%; | ||||
|     height: 200rpx; | ||||
|     padding: 20rpx; | ||||
|     background-color: #f7f7f7; | ||||
|     border-radius: 12rpx; | ||||
|     font-size: 28rpx; | ||||
|     color: #333; | ||||
|   } | ||||
| 
 | ||||
|   .notice-box { | ||||
|     padding: 20rpx; | ||||
|     background-color: #f0f7ff; | ||||
|     border-radius: 12rpx; | ||||
|     margin: 40rpx 0; | ||||
|   } | ||||
| 
 | ||||
|   .notice-title { | ||||
|     font-size: 28rpx; | ||||
|     color: #452aa1; | ||||
|     font-weight: bold; | ||||
|     display: block; | ||||
|     margin-bottom: 10rpx; | ||||
|   } | ||||
| 
 | ||||
|   .notice-header { | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     align-items: center; | ||||
|     padding: 20rpx 0; | ||||
|   } | ||||
| 
 | ||||
|   .toggle-btn { | ||||
|     color: #452aa1; | ||||
|     font-size: 24rpx; | ||||
|   } | ||||
| 
 | ||||
|   .notice-content { | ||||
|     font-size: 24rpx; | ||||
|     color: #666; | ||||
|     line-height: 1.6; | ||||
|     padding-bottom: 20rpx; | ||||
|   } | ||||
| 
 | ||||
|   .submit-btn { | ||||
|     margin-top: 40rpx; | ||||
|     background-color: #452aa1; | ||||
|     color: white; | ||||
|     border-radius: 8rpx; | ||||
|     height: 90rpx; | ||||
|     line-height: 90rpx; | ||||
|     font-size: 32rpx; | ||||
| 
 | ||||
|     &[disabled] { | ||||
|       background-color: #f0f0f0; | ||||
|     } | ||||
|   } | ||||
| </style> | ||||
| @ -8,8 +8,12 @@ | ||||
|         <div class="group-avatar flex items-center justify-center"> | ||||
|           <div class="avatar-placeholder" v-if="groupActiveIndex === -1"></div> | ||||
|           <div v-else> | ||||
|             <avatarModule :mode="2" :avatar="avatarImg" :groupType="groupType" | ||||
|               :customStyle="{ width: '192rpx', height: '192rpx' }"></avatarModule> | ||||
|             <avatarModule | ||||
|               :mode="2" | ||||
|               :avatar="avatarImg" | ||||
|               :groupType="groupType" | ||||
|               :customStyle="{ width: '192rpx', height: '192rpx' }" | ||||
|             ></avatarModule> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="input-group flex items-center justify-between"> | ||||
| @ -17,12 +21,25 @@ | ||||
|             群名称 | ||||
|           </div> | ||||
|           <div class="input-box"> | ||||
|             <tm-input v-model="groupName" :followTheme="false" fontColor="#747474" placeholderStyle="color: #B4B4B4" | ||||
|               focusColor="#FFF" :fontSize="28" :maxlength="20" :height="40" :transprent="true" | ||||
|               placeholder="请输入群名称(1~20个字)" :padding="[0, 0]" align="right"></tm-input> | ||||
|             <tm-input | ||||
|               v-model="groupName" | ||||
|               :followTheme="false" | ||||
|               fontColor="#747474" | ||||
|               placeholderStyle="color: #B4B4B4" | ||||
|               focusColor="#FFF" | ||||
|               :fontSize="28" | ||||
|               :maxlength="20" | ||||
|               :height="40" | ||||
|               :transprent="true" | ||||
|               placeholder="请输入群名称(1~20个字)" | ||||
|               :padding="[0, 0]" | ||||
|               align="right" | ||||
|             ></tm-input> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]"> | ||||
|         <div | ||||
|           class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]" | ||||
|         > | ||||
|           <div class="flex items-center justify-between"> | ||||
|             <div class="input-item"> | ||||
|               群类型 | ||||
| @ -35,310 +52,359 @@ | ||||
|                 <span v-else-if="groupActiveIndex === 2">项目群</span> | ||||
|               </div> | ||||
|               <div class="ml-[32rpx]"> | ||||
|                 <tm-icon :font-size="22" color="#747474" name="tmicon-angle-right"></tm-icon> | ||||
|                 <tm-icon | ||||
|                   :font-size="22" | ||||
|                   color="#747474" | ||||
|                   name="tmicon-angle-right" | ||||
|                 ></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div v-if="depCheckedKeys.length && groupActiveIndex === 1" class="mt-[32rpx]"> | ||||
|             <div v-for="(item, index) in depCheckedKeys" class="text-[#747474] text-[28rpx] leading-[40rpx] font-bold" | ||||
|           <div | ||||
|             v-if="depCheckedKeys.length && groupActiveIndex === 1" | ||||
|             class="mt-[32rpx]" | ||||
|           > | ||||
|             <div | ||||
|               v-for="(item, index) in depCheckedKeys" | ||||
|               class="text-[#747474] text-[28rpx] leading-[40rpx] font-bold" | ||||
|               :class="[ | ||||
|                 index !== 0 ? 'mt-[10rpx]' : '', | ||||
|                 depsNoExpanded_1 && index > 4 ? 'hidden' : '', | ||||
|               ]"> | ||||
|                 depsNoExpanded && index > 4 ? 'hidden' : '', | ||||
|               ]" | ||||
|             > | ||||
|               {{ item.name }} | ||||
|             </div> | ||||
|             <div class="text-[#46299D] text-[28rpx] mt-[20rpx] font-bold flex justify-center"> | ||||
|               <div v-if="depCheckedKeys.length > 5" @click="depsNoExpanded_1 = !depsNoExpanded_1" class="w-[100rpx]"> | ||||
|                 {{ depsNoExpanded_1 ? '展开' : '收起' }} | ||||
|             <div | ||||
|               class="text-[#46299D] text-[28rpx] mt-[20rpx] font-bold flex justify-center" | ||||
|             > | ||||
|               <div | ||||
|                 v-if="depCheckedKeys.length > 5" | ||||
|                 @click="depsNoExpanded = !depsNoExpanded" | ||||
|                 class="w-[100rpx]" | ||||
|               > | ||||
|                 {{ depsNoExpanded ? '展开' : '收起' }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
| 
 | ||||
|         <div v-if="groupActiveIndex === 0 || groupActiveIndex === 2" | ||||
|           class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]"> | ||||
|         <div | ||||
|           v-if="groupActiveIndex === 0 || groupActiveIndex === 2" | ||||
|           class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]" | ||||
|         > | ||||
|           <div class="flex items-center justify-between"> | ||||
|             <div class="input-item"> | ||||
|               群成员 | ||||
|             </div> | ||||
|             <div @click="chooseMembers" class="left-box"> | ||||
|               <div class="ml-[32rpx] flex items-center"> | ||||
|                 <div class="text-[#B4B4B4] text-[28rpx] font-bold mr-[32rpx]"> | ||||
|                 <div | ||||
|                   class="text-[#B4B4B4] text-[28rpx] font-bold mr-[32rpx]" | ||||
|                 > | ||||
|                   全部({{ allChooseMembers?.length || 0 }}) | ||||
|                 </div> | ||||
|                 <tm-icon :font-size="22" color="#747474" name="tmicon-angle-right"></tm-icon> | ||||
|                 <tm-icon | ||||
|                   :font-size="22" | ||||
|                   color="#747474" | ||||
|                   name="tmicon-angle-right" | ||||
|                 ></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <groupMemberList :groupType="3" :is_manager="true" :memberList="allChooseMembers" :memberListsLimit="15" | ||||
|             :hideAddRemoveBtns="true"></groupMemberList> | ||||
|           <groupMemberList | ||||
|             :groupType="3" | ||||
|             :is_manager="true" | ||||
|             :memberList="allChooseMembers" | ||||
|             :memberListsLimit="15" | ||||
|             :hideAddRemoveBtns="true" | ||||
|           ></groupMemberList> | ||||
|         </div> | ||||
| 
 | ||||
|         <div v-if="groupActiveIndex === 1" class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]"> | ||||
|         <div | ||||
|           v-if="groupActiveIndex === 1" | ||||
|           class="input-group w-full flex flex-col mt-[20rpx] leading-[40rpx]" | ||||
|         > | ||||
|           <div class="flex items-center justify-between"> | ||||
|             <div class="input-item"> | ||||
|               群管理员 | ||||
|             </div> | ||||
|             <div @click="chooseGroupAdmin" class="left-box"> | ||||
|               <div class="ml-[32rpx] flex items-center"> | ||||
|                 <div v-if="!groupAdmins.length" class="text-[#B4B4B4] text-[28rpx] font-bold mr-[32rpx]"> | ||||
|                 <div | ||||
|                   v-if="!groupAdmins.length" | ||||
|                   class="text-[#B4B4B4] text-[28rpx] font-bold mr-[32rpx]" | ||||
|                 > | ||||
|                   请选择群管理员 | ||||
|                 </div> | ||||
|                 <tm-icon :font-size="22" color="#747474" name="tmicon-angle-right"></tm-icon> | ||||
|                 <tm-icon | ||||
|                   :font-size="22" | ||||
|                   color="#747474" | ||||
|                   name="tmicon-angle-right" | ||||
|                 ></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div v-if="groupAdmins.length" class="mt-[32rpx]"> | ||||
|             <div v-for="(item, index) in groupAdmins" class="text-[#747474] text-[28rpx] leading-[40rpx] font-bold" | ||||
|             <div | ||||
|               v-for="(item, index) in groupAdmins" | ||||
|               class="text-[#747474] text-[28rpx] leading-[40rpx] font-bold" | ||||
|               :class="[ | ||||
|                 index !== 0 ? 'mt-[10rpx]' : '', | ||||
|                 depsNoExpanded_2 && index > 4 ? 'hidden' : '', | ||||
|               ]"> | ||||
|                 depsNoExpanded && index > 4 ? 'hidden' : '', | ||||
|               ]" | ||||
|             > | ||||
|               {{ item.name }} | ||||
|             </div> | ||||
|             <div class="text-[#46299D] text-[28rpx] mt-[20rpx] font-bold flex justify-center"> | ||||
|               <div v-if="groupAdmins.length > 5" @click="depsNoExpanded_2 = !depsNoExpanded_2" class="w-[100rpx]"> | ||||
|                 {{ depsNoExpanded_2 ? '展开' : '收起' }} | ||||
|             <div | ||||
|               class="text-[#46299D] text-[28rpx] mt-[20rpx] font-bold flex justify-center" | ||||
|             > | ||||
|               <div | ||||
|                 v-if="groupAdmins.length > 5" | ||||
|                 @click="depsNoExpanded = !depsNoExpanded" | ||||
|                 class="w-[100rpx]" | ||||
|               > | ||||
|                 {{ depsNoExpanded ? '展开' : '收起' }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <template #bottom> | ||||
|         <customBtn :isBottom="true" :btnText="$t('pageTitle.create.group')" @click="handleConfirm" | ||||
|           :isLoading="isLoading" :disabled="confirmBtnStatus || isLoading"></customBtn> | ||||
|         <customBtn | ||||
|           :isBottom="true" | ||||
|           :btnText="$t('pageTitle.create.group')" | ||||
|           @click="handleConfirm" | ||||
|           :disabled="confirmBtnStatus" | ||||
|         ></customBtn> | ||||
|       </template> | ||||
|     </zPaging> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
|   import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
|   import customBtn from '@/components/custom-btn/custom-btn.vue' | ||||
|   import groupMemberList from '../chatSettings/components/groupMembersList.vue' | ||||
|   import avatarModule from '@/components/avatar-module/index.vue' | ||||
| import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import customBtn from '@/components/custom-btn/custom-btn.vue' | ||||
| import groupMemberList from '../chatSettings/components/groupMembersList.vue' | ||||
| import avatarModule from '@/components/avatar-module/index.vue' | ||||
| 
 | ||||
|   import { | ||||
|     ref, | ||||
|     watch, | ||||
|     computed, | ||||
|     onMounted | ||||
|   } from 'vue' | ||||
|   import { | ||||
|     onShow, | ||||
|     onLoad, | ||||
|     onUnload | ||||
|   } from '@dcloudio/uni-app' | ||||
|   import { | ||||
|     useChatList | ||||
|   } from '@/store/chatList/index.js' | ||||
|   import { | ||||
|     useAuth | ||||
|   } from '@/store/auth' | ||||
|   import { | ||||
|     useTalkStore, | ||||
|     useUserStore, | ||||
|     useGroupStore | ||||
|   } from '@/store' | ||||
|   import addCircle from '@/static/image/chatList/addCircle.png' | ||||
|   import cahtPopover from '@/static/image/chatList/cahtPopover.png' | ||||
|   import { | ||||
|     ServeCreateGroup | ||||
|   } from '@/api/group/index' | ||||
|   import { | ||||
|     useGroupTypeStore | ||||
|   } from '@/store/groupType' | ||||
|   import { | ||||
|     handleSetWebviewStyle | ||||
|   } from '@/utils/common' | ||||
| import { ref, watch, computed, onMounted } from 'vue' | ||||
| import { onShow, onLoad, onUnload } from '@dcloudio/uni-app' | ||||
| import { useChatList } from '@/store/chatList/index.js' | ||||
| import { useAuth } from '@/store/auth' | ||||
| import { useTalkStore, useUserStore, useGroupStore } from '@/store' | ||||
| import addCircle from '@/static/image/chatList/addCircle.png' | ||||
| import cahtPopover from '@/static/image/chatList/cahtPopover.png' | ||||
| import { ServeCreateGroup } from '@/api/group/index' | ||||
| import { useGroupTypeStore } from '@/store/groupType' | ||||
| import { handleSetWebviewStyle } from '@/utils/common' | ||||
| 
 | ||||
|   const { | ||||
|     groupName, | ||||
|     groupActiveIndex, | ||||
|     depCheckedKeys, | ||||
|     groupAdmins, | ||||
|     createDepGroup, | ||||
|     resetGroupInfo, | ||||
|     allChooseMembers, | ||||
|   } = useGroupTypeStore() | ||||
|   const talkStore = useTalkStore() | ||||
|   const userStore = useUserStore() | ||||
|   const groupStore = useGroupStore() | ||||
|   const { | ||||
|     userInfo | ||||
|   } = useAuth() | ||||
| const { | ||||
|   groupName, | ||||
|   groupActiveIndex, | ||||
|   depCheckedKeys, | ||||
|   groupAdmins, | ||||
|   createDepGroup, | ||||
|   resetGroupInfo, | ||||
|   allChooseMembers, | ||||
| } = useGroupTypeStore() | ||||
| const talkStore = useTalkStore() | ||||
| const userStore = useUserStore() | ||||
| const groupStore = useGroupStore() | ||||
| const { userInfo } = useAuth() | ||||
| 
 | ||||
|   const groupChatType = ref('') | ||||
|   const depsNoExpanded_1 = ref(true) | ||||
|   const depsNoExpanded_2 = ref(true) | ||||
| const groupChatType = ref('') | ||||
| const depsNoExpanded = ref(true) | ||||
| 
 | ||||
| 
 | ||||
|   onLoad(() => { | ||||
|     groupStore.$reset() | ||||
|   }) | ||||
|   onUnload(() => { | ||||
|     resetGroupInfo(); | ||||
|   }) | ||||
|   onMounted(() => { | ||||
|     handleSetWebviewStyle() | ||||
| onLoad(()=> { | ||||
|   groupStore.$reset() | ||||
| }) | ||||
| onUnload(()=> { | ||||
|   groupName.value = ''; | ||||
|   groupActiveIndex.value = -1; | ||||
| }) | ||||
| onMounted(() => { | ||||
|   handleSetWebviewStyle() | ||||
| }) | ||||
| 
 | ||||
| //群类型 | ||||
| const groupType = computed(() => { | ||||
|   let group_type = '' | ||||
|   switch (groupActiveIndex.value) { | ||||
|     case 0: | ||||
|       group_type = 1 | ||||
|       break | ||||
|     case 1: | ||||
|       group_type = 2 | ||||
|       break | ||||
|     case 2: | ||||
|       group_type = 3 | ||||
|       break | ||||
|     default: | ||||
|       group_type = '' | ||||
|   } | ||||
|   return group_type | ||||
| }) | ||||
| 
 | ||||
| //点击跳转到选择群类型页面 | ||||
| const chooseGroupType = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/chooseGroupType/index', | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
|   //群类型 | ||||
|   const groupType = computed(() => { | ||||
|     let group_type = '' | ||||
|     switch (groupActiveIndex.value) { | ||||
|       case 0: | ||||
|         group_type = 1 | ||||
|         break | ||||
|       case 1: | ||||
|         group_type = 2 | ||||
|         break | ||||
|       case 2: | ||||
|         group_type = 3 | ||||
|         break | ||||
|       default: | ||||
|         group_type = '' | ||||
| const chooseGroupAdmin = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: | ||||
|       '/pages/chatSettings/groupManage/selectMembers?manageType=admin&isCreateDepGroup=1', | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| const chooseMembers = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/chooseByDeps/index?chooseMode=2', | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //点击发起群聊 | ||||
| const handleConfirm = async () => { | ||||
|   // console.log(allChooseMembers.value) | ||||
|   let erp_ids = '' | ||||
|   if (allChooseMembers?.value?.length > 0) { | ||||
|     allChooseMembers?.value?.forEach((ele) => { | ||||
|       if (!erp_ids) { | ||||
|         erp_ids = String(ele.ID) | ||||
|       } else { | ||||
|         erp_ids += ',' + ele.ID | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
|   if (groupActiveIndex.value === 0) { | ||||
|     //普通群 | ||||
|     let params = { | ||||
|       avatar: '', | ||||
|       name: groupName.value, | ||||
|       erp_ids: erp_ids, | ||||
|       type: 1, | ||||
|       profile: '', | ||||
|     } | ||||
|     return group_type | ||||
|   }) | ||||
| 
 | ||||
|   //点击跳转到选择群类型页面 | ||||
|   const chooseGroupType = () => { | ||||
|     uni.navigateTo({ | ||||
|       url: '/pages/chooseGroupType/index', | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   const chooseGroupAdmin = () => { | ||||
|     uni.navigateTo({ | ||||
|       url: '/pages/chatSettings/groupManage/selectMembers?manageType=admin&isCreateDepGroup=1', | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   const chooseMembers = () => { | ||||
|     uni.navigateTo({ | ||||
|       url: '/pages/chooseByDeps/index?chooseMode=2', | ||||
|     }) | ||||
|   } | ||||
|   const isLoading = ref(false) | ||||
| 
 | ||||
|   // 点击发起群聊 | ||||
|   const handleConfirm = async () => { | ||||
|     if (isLoading.value) return | ||||
|     isLoading.value = true | ||||
| 
 | ||||
|     try { | ||||
|       let erp_ids = '' | ||||
|       if (allChooseMembers?.value?.length > 0) { | ||||
|         allChooseMembers.value.forEach((ele) => { | ||||
|           if (!erp_ids) { | ||||
|             erp_ids = String(ele.ID) | ||||
|           } else { | ||||
|             erp_ids += ',' + ele.ID | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
| 
 | ||||
|       let res = null | ||||
| 
 | ||||
|       if (groupActiveIndex.value === 0) { | ||||
|         // 普通群 | ||||
|         const params = { | ||||
|           avatar: '', | ||||
|           name: groupName.value, | ||||
|           erp_ids: erp_ids, | ||||
|           type: 1, | ||||
|           profile: '', | ||||
|         } | ||||
|         console.log('普通群参数:', params) | ||||
|         res = await ServeCreateGroup(params) | ||||
|       } else if (groupActiveIndex.value === 1) { | ||||
|         // 部门群 | ||||
|         res = await createDepGroup() | ||||
|       } else if (groupActiveIndex.value === 2) { | ||||
|         // 项目群 | ||||
|         const params = { | ||||
|           avatar: '', | ||||
|           name: groupName.value, | ||||
|           erp_ids: erp_ids, | ||||
|           type: 3, | ||||
|           profile: '', | ||||
|         } | ||||
|         console.log('项目群参数:', params) | ||||
|         res = await ServeCreateGroup(params) | ||||
|       } | ||||
| 
 | ||||
|       if (res?.code === 200) { | ||||
|         resetGroupInfo() | ||||
|         uni.navigateBack() | ||||
|       } | ||||
|     } catch (err) { | ||||
|       console.error(err) | ||||
|     } finally { | ||||
|       isLoading.value = false | ||||
|     console.log(params) | ||||
|     const res = await ServeCreateGroup(params) | ||||
|     if (res.code === 200) { | ||||
|       resetGroupInfo() | ||||
|       uni.navigateBack() | ||||
|     } | ||||
|   } else if (groupActiveIndex.value === 1) { | ||||
|     //部门群 | ||||
|     const res = await createDepGroup() | ||||
|     if (res.code === 200) { | ||||
|       resetGroupInfo() | ||||
|       uni.navigateBack() | ||||
|     } | ||||
|   } else if (groupActiveIndex.value === 2) { | ||||
|     //项目群 | ||||
|     let params = { | ||||
|       avatar: '', | ||||
|       name: groupName.value, | ||||
|       erp_ids: erp_ids, | ||||
|       type: 3, | ||||
|       profile: '', | ||||
|     } | ||||
|     console.log(params) | ||||
|     const res = await ServeCreateGroup(params) | ||||
|     if (res.code === 200) { | ||||
|       resetGroupInfo() | ||||
|       uni.navigateBack() | ||||
|     } | ||||
|   } else { | ||||
|   } | ||||
|   //发起群聊按钮可点击状态 | ||||
|   const confirmBtnStatus = computed(() => { | ||||
|     return groupActiveIndex.value === -1; | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
|   onShow(() => { | ||||
|     depsNoExpanded_1.value = true; | ||||
|     depsNoExpanded_2.value = true; | ||||
|   }) | ||||
| //发起群聊按钮可点击状态 | ||||
| const confirmBtnStatus = computed(() => { | ||||
|   let disabledT = false | ||||
|   console.log(groupActiveIndex.value !== -1) | ||||
|   if ( | ||||
|     groupName.value === '' || | ||||
|     (groupActiveIndex.value && groupActiveIndex.value === -1) || | ||||
|     (!groupActiveIndex.value && groupActiveIndex.value !== 0) | ||||
|   ) { | ||||
|     return true | ||||
|   } | ||||
|   switch (groupActiveIndex.value) { | ||||
|     case 0: | ||||
|       if(allChooseMembers?.value?.length < 3) { | ||||
|         disabledT = true | ||||
|       } | ||||
|       break | ||||
|     case 1: | ||||
|       if (!depCheckedKeys.value.length) { | ||||
|         disabledT = true | ||||
|       } | ||||
|       if (!groupAdmins.value.length) { | ||||
|         disabledT = true | ||||
|       } | ||||
|       break | ||||
|     case 2: | ||||
|       if(allChooseMembers?.value?.length < 3) { | ||||
|         disabledT = true | ||||
|       } | ||||
|       break | ||||
|     default: | ||||
|       break | ||||
|   } | ||||
|   return disabledT | ||||
| }) | ||||
| 
 | ||||
| onShow(() => { | ||||
|   depsNoExpanded.value = true | ||||
| }) | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
|   ::v-deep .zp-paging-container-content { | ||||
|     height: 100%; | ||||
|     display: flex; | ||||
|   } | ||||
| ::v-deep .zp-paging-container-content { | ||||
|   height: 100%; | ||||
|   display: flex; | ||||
| } | ||||
| 
 | ||||
|   .create-group-chat { | ||||
|     background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|     background-size: cover; | ||||
|     background-position: center bottom; | ||||
|     width: 100%; | ||||
|     padding: 0 32rpx 20rpx; | ||||
| 
 | ||||
|     .group-avatar { | ||||
|       padding: 60rpx 0; | ||||
| 
 | ||||
|       .avatar-placeholder { | ||||
|         width: 192rpx; | ||||
|         height: 192rpx; | ||||
|         background-color: #e0e0e0; | ||||
|         border-radius: 50%; | ||||
|       } | ||||
| .create-group-chat { | ||||
|   background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|   background-size: cover; | ||||
|   background-position: center bottom; | ||||
|   width: 100%; | ||||
|   padding: 0 32rpx 20rpx; | ||||
|   .group-avatar { | ||||
|     padding: 60rpx 0; | ||||
|     .avatar-placeholder { | ||||
|       width: 192rpx; | ||||
|       height: 192rpx; | ||||
|       background-color: #e0e0e0; | ||||
|       border-radius: 50%; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| .divider { | ||||
|   height: 1rpx; | ||||
|   background-color: #7c7c7c; | ||||
|   opacity: 0.6; | ||||
| } | ||||
| 
 | ||||
|   .divider { | ||||
|     height: 1rpx; | ||||
|     background-color: #7c7c7c; | ||||
|     opacity: 0.6; | ||||
|   } | ||||
| 
 | ||||
|   .input-group { | ||||
|     background-color: #fff; | ||||
|     padding: 38rpx 40rpx 32rpx 32rpx; | ||||
|   } | ||||
| 
 | ||||
|   .input-item { | ||||
|     line-height: 40rpx; | ||||
|     font-size: 28rpx; | ||||
|     color: #000; | ||||
|     font-weight: bold; | ||||
|   } | ||||
| 
 | ||||
|   .input-box { | ||||
|     margin-left: 84rpx; | ||||
|     line-height: 40rpx; | ||||
|     width: 404rpx; | ||||
|     font-weight: bold; | ||||
|   } | ||||
| 
 | ||||
|   .left-box { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|   } | ||||
| .input-group { | ||||
|   background-color: #fff; | ||||
|   padding: 38rpx 40rpx 32rpx 32rpx; | ||||
| } | ||||
| .input-item { | ||||
|   line-height: 40rpx; | ||||
|   font-size: 28rpx; | ||||
|   color: #000; | ||||
|   font-weight: bold; | ||||
| } | ||||
| .input-box { | ||||
|   margin-left: 84rpx; | ||||
|   line-height: 40rpx; | ||||
|   width: 404rpx; | ||||
|   font-weight: bold; | ||||
| } | ||||
| .left-box { | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -1,34 +1,25 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <zPaging :fixed="false" :height="'210px'" :show-scrollbar="false"> | ||||
|       <div class="emojiRoot"> | ||||
|         <div | ||||
|           v-for="(img, key) in emojis" | ||||
|           v-html="img" | ||||
|           :key="key" | ||||
|           @click="onSendEmoticon(1, key, img)" | ||||
|           class="option pointer flex-center" | ||||
|         /> | ||||
|       </div> | ||||
|     </zPaging> | ||||
|   <div class="emojiRoot"> | ||||
|     <div v-for="(img, key) in emojis" v-html="img" :key="key" @click="onSendEmoticon(1, key, img)" | ||||
|       class="option pointer flex-center" /> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import { ref, reactive, defineProps, defineEmits } from 'vue' | ||||
| import dayjs from 'dayjs' | ||||
| import { ref, reactive, defineProps,defineEmits } from "vue" | ||||
| import dayjs from "dayjs"; | ||||
| import { beautifyTime } from '@/utils/datetime' | ||||
| import { useTalkStore } from '@/store' | ||||
| import { useSessionMenu } from '@/hooks' | ||||
| import { emojis } from '@/utils/emojis' | ||||
| 
 | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   data: { | ||||
|     type: Object, | ||||
|     default: {}, | ||||
|     required: false, | ||||
|   }, | ||||
| }) | ||||
| }); | ||||
| 
 | ||||
| const emit = defineEmits(['on-select']) | ||||
| const onSendEmoticon = (type, value, img = '') => { | ||||
| @ -42,15 +33,21 @@ const onSendEmoticon = (type, value, img = '') => { | ||||
|     emit('on-select', { type, value, img }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| </script> | ||||
| <style lang="scss" scoped> | ||||
| .emojiRoot { | ||||
|   width: 100%; | ||||
|   height: 420rpx; | ||||
|   padding: 5rpx 32rpx; | ||||
|   display: flex; | ||||
|   justify-content: flex-start; | ||||
|   align-items: center; | ||||
|   flex-wrap: wrap; | ||||
|   overflow: auto; | ||||
| } | ||||
| 
 | ||||
| .option { | ||||
|  | ||||
| @ -1,39 +1,25 @@ | ||||
| <template> | ||||
|   <div class="emojiRoot"> | ||||
|     <div | ||||
|       @click="() => photoActionsSelect(0)" | ||||
|       class="flex flex-col items-center" | ||||
|     > | ||||
|       <div | ||||
|         class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center" | ||||
|       > | ||||
|     <div @click="() => photoActionsSelect(0)" class="flex flex-col items-center"> | ||||
|       <div class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center"> | ||||
|         <tm-image :width="53" :height="44" :src="photoAlbum"></tm-image> | ||||
|       </div> | ||||
|       <div class="mt-[6rpx] text-[#666666] text-[24rpx]">照片</div> | ||||
|     </div> | ||||
|     <div | ||||
|       @click="() => photoActionsSelect(1)" | ||||
|       class="flex flex-col items-center" | ||||
|     > | ||||
|       <div | ||||
|         class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center" | ||||
|       > | ||||
|     <div @click="() => photoActionsSelect(1)" class="flex flex-col items-center"> | ||||
|       <div class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center"> | ||||
|         <tm-image :width="53" :height="44" :src="videoImg"></tm-image> | ||||
|       </div> | ||||
|       <div class="mt-[6rpx] text-[#666666] text-[24rpx]">视频</div> | ||||
|     </div> | ||||
|     <div @click="takePhoto" class="flex flex-col items-center"> | ||||
|       <div | ||||
|         class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center" | ||||
|       > | ||||
|       <div class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center"> | ||||
|         <tm-image :width="53" :height="44" :src="photoGraph"></tm-image> | ||||
|       </div> | ||||
|       <div class="mt-[6rpx] text-[#666666] text-[24rpx]">拍摄</div> | ||||
|     </div> | ||||
|     <div @click="chooseFile" class="flex flex-col items-center"> | ||||
|       <div | ||||
|         class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center" | ||||
|       > | ||||
|       <div class="w-[106rpx] h-[106rpx] bg-[#F9F9F9] rounded-[60rpx] flex-center"> | ||||
|         <tm-image :width="53" :height="44" :src="folder"></tm-image> | ||||
|       </div> | ||||
|       <div class="mt-[6rpx] text-[#666666] text-[24rpx]">文件</div> | ||||
| @ -57,7 +43,6 @@ import videoImg from '@/static/image/chatList/video@2x.png' | ||||
| import folder from '@/static/image/chatList/folder.png' | ||||
| import { uploadImg } from '@/api/chat' | ||||
| import { uniqueId } from '@/utils' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   sendUserInfo: { | ||||
| @ -93,7 +78,7 @@ const onProgressFn = (progress, id) => { | ||||
|   updateUploadProgress(id, (progress.loaded / progress.total) * 100) | ||||
| } | ||||
| 
 | ||||
| const photoActionsSelect = (index) => { | ||||
| const photoActionsSelect = async (index) => { | ||||
|   if (index === 0) { | ||||
|     if (typeof plus === 'undefined') { | ||||
|       uni.chooseImage({ | ||||
| @ -102,15 +87,8 @@ const photoActionsSelect = (index) => { | ||||
|         success: async (res) => { | ||||
|           console.log(res, 'res') | ||||
|           res.tempFiles.forEach(async (file) => { | ||||
|             const fileSizeInMB = (file.size / (1024 * 1024)).toFixed(2) | ||||
|             if (fileSizeInMB > 100) { | ||||
|               plus.nativeUI.toast('图片大小不能超过100MB') | ||||
|               return | ||||
|             } | ||||
|             const result = await onUploadImageVideo(file, 'image') | ||||
|             if (result) { | ||||
|               emit('selectImg', result, result.file_num) | ||||
|             } | ||||
|             let data = await onUploadImageVideo(file, 'image') | ||||
|             emit('selectImg', data, data.file_num) | ||||
|           }) | ||||
|         }, | ||||
|       }) | ||||
| @ -129,17 +107,8 @@ const photoActionsSelect = (index) => { | ||||
|                   fileReader.onloadend = async (e) => { | ||||
|                     const base64Url = e.target.result | ||||
|                     const fileObj = base64ToFile(base64Url) | ||||
|                     const fileSizeInMB = (fileObj.size / (1024 * 1024)).toFixed( | ||||
|                       2, | ||||
|                     ) | ||||
|                     if (fileSizeInMB > 100) { | ||||
|                       plus.nativeUI.toast('图片大小不能超过100MB') | ||||
|                       return | ||||
|                     } | ||||
|                     let data = await onUploadImageVideo(fileObj, 'image') | ||||
|                     if (data) { | ||||
|                       emit('selectImg', data, data.file_num) | ||||
|                     } | ||||
|                     emit('selectImg', data, data.file_num) | ||||
|                   } | ||||
|                 }) | ||||
|               }, | ||||
| @ -163,44 +132,183 @@ const photoActionsSelect = (index) => { | ||||
|       ) | ||||
|     } | ||||
|   } else { | ||||
|     // handleFindWebview(`getPlusVideoPicker()`) | ||||
|     // return | ||||
|     uni.chooseVideo({ | ||||
|       sourceType: ['album'], | ||||
|       compressed: true, | ||||
|       maxDuration: 60, | ||||
|       success: async (res) => { | ||||
|         console.log(res, 'res') | ||||
|         const fileSizeInMB = (res.tempFile.size / (1024 * 1024)).toFixed(2) | ||||
|         if (fileSizeInMB > 100) { | ||||
|           plus.nativeUI.toast('视频大小不能超过100MB') | ||||
|           return | ||||
|         } | ||||
|         let data = await onUploadImageVideo( | ||||
|           res.tempFile, | ||||
|           'video', | ||||
|           res.tempFilePath, | ||||
|         ) | ||||
|         if (data) { | ||||
|           emit('selectImg', data, data.file_num) | ||||
|         } | ||||
|     // uni.chooseVideo({ | ||||
|     //   sourceType: ['album'], | ||||
|     //   compressed: true, | ||||
|     //   maxDuration: 60, | ||||
|     //   success: async (res) => { | ||||
|     //     console.log( 'chooseVideo res.tempFilePath',res.tempFilePath) | ||||
|     //     console.log( 'chooseVideo res.tempFile',res.tempFile) | ||||
|     //     return | ||||
|     //     let data = await onUploadImageVideo( | ||||
|     //       res.tempFile, | ||||
|     //       'video', | ||||
|     //       res.tempFilePath, | ||||
|     //     ) | ||||
|     //     emit('selectImg', data, data.file_num) | ||||
|     //   }, | ||||
|     // }) | ||||
| 
 | ||||
|     plus?.gallery.pick( | ||||
|       (res) => { | ||||
|         res.files.reverse() | ||||
|         res.files.forEach(async (filePath) => { | ||||
|           plus?.io?.resolveLocalFileSystemURL( | ||||
|             filePath, | ||||
|             async (entry) => { | ||||
|               entry.file(async (file) => { | ||||
|                  | ||||
|                 try { | ||||
|                   console.log('filePath', filePath) | ||||
|                   // 获取文件内容并创建 Blob URL | ||||
|                  playVideoWithNativeBridge(filePath) | ||||
|                   const videoInfo = await getVideoInfo(filePath); | ||||
|                   console.log('视频信息:', videoInfo); | ||||
| 
 | ||||
|                   console.log('文件详细信息:', { | ||||
|                     name: file.name, | ||||
|                     size: file.size, | ||||
|                     type: file.type, | ||||
|                   }); | ||||
|                   const fileInfo = await getFilePath(file); | ||||
|                   console.log('fileInfo', fileInfo) | ||||
| 
 | ||||
|                   // 继续处理上传逻辑 | ||||
|                   let data = await onUploadImageVideo(file, 'video', videoInfo); | ||||
|                   emit('selectImg', data, data.file_num); | ||||
|                 } catch (error) { | ||||
|                   console.log('文件处理失败:', error); | ||||
|                 } | ||||
|               }) | ||||
|             }, | ||||
|             (err) => { | ||||
|               console.log(err) | ||||
|             }, | ||||
|           ) | ||||
|         }) | ||||
|       }, | ||||
|     }) | ||||
|       (err) => { | ||||
|         console.log(err) | ||||
|       }, | ||||
|       { | ||||
|         filter: 'video', | ||||
|         maximum: 2, | ||||
|         multiple: true, | ||||
|         onmaxed: () => { | ||||
|           plus.nativeUI.toast('最多只能选择2个视频') | ||||
|         }, | ||||
|       }, | ||||
|     ) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const onUploadImageVideo = async (file, type = 'image', fileUrl) => { | ||||
|   console.log('开始上传文件:', file.name) | ||||
|   uploadsStore.updateUploadStatus(true) | ||||
| const playVideoWithNativeBridge = (filePath) => { | ||||
|   return new Promise((resolve, reject) => { | ||||
|     if (window.webkit && window.webkit.messageHandlers && window.webkit.messageHandlers.playVideo) { | ||||
|       // iOS WebView 桥接 | ||||
|       window.webkit.messageHandlers.playVideo.postMessage({ | ||||
|         filePath: filePath | ||||
|       }); | ||||
|       resolve(); | ||||
|     } else if (window.android && window.android.playVideo) { | ||||
|       // Android WebView 桥接 | ||||
|       window.android.playVideo(filePath); | ||||
|       resolve(); | ||||
|     } else if (window.plus) { | ||||
|       // 使用 plus API 创建本地 HTTP 服务 | ||||
|       plus.bridge.exec('VideoPlayerModule', 'playVideoFromPath', [filePath],  | ||||
|         (httpUrl) => { | ||||
|           console.log('获取到HTTP URL:', httpUrl); | ||||
|           resolve(httpUrl); | ||||
|         },  | ||||
|         (error) => { | ||||
|           console.error('原生方法调用失败:', error); | ||||
|           reject(error); | ||||
|         } | ||||
|       ); | ||||
|     } else { | ||||
|       reject(new Error('没有可用的原生桥接方法')); | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| const getFilePath = async (file) => { | ||||
|   try { | ||||
|     // #ifdef H5 | ||||
|     // 将 plus 文件对象转换为 Blob | ||||
|     return new Promise((resolve, reject) => { | ||||
|       plus.io.requestFileSystem(plus.io.PRIVATE_DOC, (fs) => { | ||||
|         const reader = new plus.io.FileReader(); | ||||
| 
 | ||||
|         reader.onload = function (e) { | ||||
|           // 将文件内容转换为 Blob | ||||
|           const blob = new Blob([e.target.result], { type: file.type || 'video/mp4' }); | ||||
|           // 创建新的 File 对象 | ||||
|           const newFile = new File([blob], file.name || 'video.mp4', { | ||||
|             type: file.type || 'video/mp4' | ||||
|           }); | ||||
| 
 | ||||
|           // 创建新的 FormData | ||||
|           // const formData = new FormData(); | ||||
|           // formData.append('file', newFile); | ||||
|           // 添加其他需要的字段 | ||||
|           // formData.append('other_field', 'value'); | ||||
| 
 | ||||
|           resolve(newFile); | ||||
|         }; | ||||
| 
 | ||||
|         reader.onerror = function (error) { | ||||
|           console.error('读取文件失败:', error); | ||||
|           reject(error); | ||||
|         }; | ||||
| 
 | ||||
|         reader.readAsArrayBuffer(file); | ||||
|       }); | ||||
|     }); | ||||
|     // #endif | ||||
| 
 | ||||
|     // #ifndef H5 | ||||
|     const formData = new FormData(); | ||||
|     formData.append('file', file); | ||||
|     return formData; | ||||
|     // #endif | ||||
|   } catch (error) { | ||||
|     console.error('处理文件失败:', error); | ||||
|     throw error; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| const onUploadImageVideo = async (file, type = 'image', resp) => { | ||||
|   console.log(file, 'file') | ||||
|   return new Promise(async (resolve) => { | ||||
|     if (type === 'image') { | ||||
|       let image = new Image() | ||||
|       image.src = URL.createObjectURL(file) | ||||
|       image.onload = async () => { | ||||
|       image.onload = () => { | ||||
|         const form = new FormData() | ||||
|         form.append('file', file) | ||||
|         form.append('source', 'fonchain-chat') | ||||
|         form.append('urlParam', `width=${image.width}&height=${image.height}`) | ||||
| 
 | ||||
|         let randomId = uniqueId() | ||||
|         let newItem = { | ||||
|           avatar: userStore.avatar, | ||||
| @ -230,160 +338,131 @@ const onUploadImageVideo = async (file, type = 'image', fileUrl) => { | ||||
|         } | ||||
| 
 | ||||
|         virtualList.value.unshift(newItem) | ||||
| 
 | ||||
|         try { | ||||
|           const result = await uploadImg(form, (e) => onProgressFn(e, randomId)) | ||||
|           console.log('上传完成,结果:', result) | ||||
| 
 | ||||
|           if (result.status === 0) { | ||||
|             // 更新上传状态为成功 | ||||
|             const index = virtualList.value.findIndex( | ||||
|               (item) => item.file_num === randomId, | ||||
|             ) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 2 | ||||
|               virtualList.value[index].uploadCurrent = 100 | ||||
|             } | ||||
| 
 | ||||
|             // 确保返回数据 | ||||
|             resolve({ | ||||
|               type: 'image', | ||||
|               url: result.data.ori_url, | ||||
|               size: file.size, | ||||
|               width: image.width, | ||||
|               height: image.height, | ||||
|               file_num: randomId, | ||||
|             }) | ||||
|           } else { | ||||
|             uploadsStore.updateUploadStatus(false) | ||||
|             // 更新上传状态为失败 | ||||
|             const index = virtualList.value.findIndex( | ||||
|               (item) => item.file_num === randomId, | ||||
|             ) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 3 | ||||
|             } | ||||
|             message.error(result.msg) | ||||
|             resolve('') | ||||
|           } | ||||
|         } catch (error) { | ||||
|           console.error('上传出错:', error) | ||||
|           uploadsStore.updateUploadStatus(false) | ||||
|           // 更新上传状态为失败 | ||||
|           const index = virtualList.value.findIndex( | ||||
|             (item) => item.file_num === randomId, | ||||
|           ) | ||||
|           if (index !== -1) { | ||||
|             virtualList.value[index].uploadStatus = 3 | ||||
|           } | ||||
|           message.error('上传失败') | ||||
|           resolve('') | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       uni.getVideoInfo({ | ||||
|         src: fileUrl, | ||||
|         success: async (resp) => { | ||||
|           console.log('视频信息:', resp) | ||||
|           const form = new FormData() | ||||
|           form.append('file', file) | ||||
|           form.append('source', 'fonchain-chat') | ||||
|           form.append('type', 'video') | ||||
|           form.append('urlParam', `width=${resp.width}&height=${resp.height}`) | ||||
|           let randomId = uniqueId() | ||||
|           let newItem = { | ||||
|             avatar: userStore.avatar, | ||||
|             created_at: dayjs().format('YYYY-MM-DD HH:mm:ss'), | ||||
|             extra: { | ||||
|               duration: parseInt(resp.duration), | ||||
|               height: resp.height, | ||||
|               name: '', | ||||
|               url: fileUrl, | ||||
|               width: resp.width, | ||||
|             }, | ||||
|             float: 'right', | ||||
|             isCheck: false, | ||||
|             is_mark: 0, | ||||
|             is_read: 0, | ||||
|             is_revoke: 0, | ||||
|             msg_id: randomId, | ||||
|             file_num: randomId, | ||||
|             msg_type: 5, | ||||
|             nickname: userStore.nickname, | ||||
|             receiver_id: dialogueStore.talk.receiver_id, | ||||
|             sequence: -1, | ||||
|             talk_type: dialogueStore.talk.talk_type, | ||||
|             user_id: userStore.uid, | ||||
|             uploadCurrent: 0, | ||||
|             uploadStatus: 1, | ||||
|           } | ||||
|           virtualList.value.unshift(newItem) | ||||
| 
 | ||||
|           try { | ||||
|             const result = await uploadImg(form, (e) => | ||||
|               onProgressFn(e, randomId), | ||||
|             ) | ||||
|             console.log('视频上传完成,结果:', result) | ||||
| 
 | ||||
|             if (result.status === 0) { | ||||
|         uploadImg(form, (e) => onProgressFn(e, randomId)).then( | ||||
|           ({ status, data, msg }) => { | ||||
|             if (status == 0) { | ||||
|               // 更新上传状态为成功 | ||||
|               const index = virtualList.value.findIndex( | ||||
|                 (item) => item.file_num === randomId, | ||||
|               ) | ||||
|               const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|               if (index !== -1) { | ||||
|                 virtualList.value[index].uploadStatus = 2 | ||||
|                 virtualList.value[index].uploadCurrent = 100 | ||||
|               } | ||||
| 
 | ||||
|               resolve({ | ||||
|                 type: 'video', | ||||
|                 url: result.data.ori_url, | ||||
|                 cover: result.data.cover_url, | ||||
|                 duration: parseInt(resp.duration), | ||||
|                 type: 'image', | ||||
|                 url: data.ori_url, | ||||
|                 size: file.size, | ||||
|                 width: image.width, | ||||
|                 height: image.height, | ||||
|                 file_num: randomId, | ||||
|               }) | ||||
|             } else { | ||||
|               uploadsStore.updateUploadStatus(false) | ||||
|               // 更新上传状态为失败 | ||||
|               const index = virtualList.value.findIndex( | ||||
|                 (item) => item.file_num === randomId, | ||||
|               ) | ||||
|               const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|               if (index !== -1) { | ||||
|                 virtualList.value[index].uploadStatus = 3 | ||||
|               } | ||||
|               message.error(result.msg) | ||||
|               resolve('') | ||||
|               message.error(msg) | ||||
|             } | ||||
|           } catch (error) { | ||||
|             console.error('视频上传出错:', error) | ||||
|             uploadsStore.updateUploadStatus(false) | ||||
|           }, | ||||
|         ) | ||||
|       } | ||||
|     } else { | ||||
|       const form = new FormData() | ||||
|       form.append('file', file) | ||||
|       form.append('source', 'fonchain-chat') | ||||
|       form.append('type', 'video') | ||||
|       form.append('urlParam', `width=${resp.width}&height=${resp.height}`) | ||||
|       // 打印 FormData 内容 | ||||
|       for (let pair of form.entries()) { | ||||
|         console.log('video form', `${pair[0]}: ${pair[1]}`); | ||||
|       } | ||||
|       let randomId = uniqueId() | ||||
|       let newItem = { | ||||
|         avatar: userStore.avatar, | ||||
|         created_at: dayjs().format('YYYY-MM-DD HH:mm:ss'), | ||||
|         extra: { | ||||
|           duration: parseInt(resp.duration), | ||||
|           height: resp.height, | ||||
|           name: '', | ||||
|           url: resp.fileUrl, | ||||
|           width: resp.width, | ||||
|         }, | ||||
|         float: 'right', | ||||
|         isCheck: false, | ||||
|         is_mark: 0, | ||||
|         is_read: 0, | ||||
|         is_revoke: 0, | ||||
|         msg_id: randomId, | ||||
|         file_num: randomId, | ||||
|         msg_type: 5, | ||||
|         nickname: userStore.nickname, | ||||
|         receiver_id: dialogueStore.talk.receiver_id, | ||||
|         sequence: -1, | ||||
|         talk_type: dialogueStore.talk.talk_type, | ||||
|         user_id: userStore.uid, | ||||
|         uploadCurrent: 0, | ||||
|         uploadStatus: 1, // 1 上传中 2 上传成功 3 上传失败 | ||||
|       } | ||||
|       virtualList.value.unshift(newItem) | ||||
|       console.log('uploadImg params', form) | ||||
|       uploadImg(form, (e) => onProgressFn(e, randomId)).then( | ||||
|         ({ status, data, msg }) => { | ||||
|           if (status == 0) { | ||||
|             // 更新上传状态为成功 | ||||
|             const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 2 | ||||
|               virtualList.value[index].uploadCurrent = 100 | ||||
|             } | ||||
|             console.log(data) | ||||
|             resolve({ | ||||
|               type: 'video', | ||||
|               url: data.ori_url, | ||||
|               cover: data.cover_url, | ||||
|               duration: parseInt(resp.duration), | ||||
|               size: file.size, | ||||
|               file_num: randomId, | ||||
|             }) | ||||
|           } else { | ||||
|             // 更新上传状态为失败 | ||||
|             const index = virtualList.value.findIndex( | ||||
|               (item) => item.file_num === randomId, | ||||
|             ) | ||||
|             const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 3 | ||||
|             } | ||||
|             message.error('上传失败') | ||||
|             resolve('') | ||||
|             message.error(msg) | ||||
|           } | ||||
|         }, | ||||
|         fail: (error) => { | ||||
|           console.error('获取视频信息失败:', error) | ||||
|           uploadsStore.updateUploadStatus(false) | ||||
|           message.error('获取视频信息失败') | ||||
|           resolve('') | ||||
|         }, | ||||
|       }) | ||||
|       ) | ||||
| 
 | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| const getVideoInfo = (filePath) => { | ||||
|   return new Promise((resolve, reject) => { | ||||
|     plus.io.getVideoInfo({ | ||||
|       filePath: filePath, | ||||
|       success: (res) => { | ||||
|         resolve({ | ||||
|           width: res.width, | ||||
|           height: res.height, | ||||
|           duration: res.duration, | ||||
|           fileUrl: filePath, | ||||
|           size: res.size | ||||
|         }); | ||||
|       }, | ||||
|       fail: (err) => { | ||||
|         console.log('获取视频信息失败:', err); | ||||
|         reject(err); | ||||
|       } | ||||
|     }); | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| const base64ToFile = (base64) => { | ||||
|   if (!base64) { | ||||
|     message.warning('您的系统暂不支持发送原图哦') | ||||
|   } | ||||
|   // base64转file | ||||
|   const [header, base64String] = base64.split(';base64,') | ||||
|   const imageType = header.split(':')[1] | ||||
| @ -493,7 +572,7 @@ const getCamera = () => { | ||||
|         }, | ||||
|       ) | ||||
|     }, | ||||
|     () => {}, | ||||
|     () => { }, | ||||
|     { index: '2' }, | ||||
|   ) | ||||
| } | ||||
| @ -529,11 +608,6 @@ const chooseFile = () => { | ||||
|     count: 1, | ||||
|     extension: [''], | ||||
|     success: (res) => { | ||||
|       const fileSizeInMB = (res.tempFiles[0].size / (1024 * 1024)).toFixed(2) | ||||
|       if (fileSizeInMB > 100) { | ||||
|         plus.nativeUI.toast('文件大小不能超过100MB') | ||||
|         return | ||||
|       } | ||||
|       let randomId = uniqueId() | ||||
|       let newItem = { | ||||
|         avatar: userStore.avatar, | ||||
| @ -561,34 +635,23 @@ const chooseFile = () => { | ||||
|         uploadStatus: 1, // 1 上传中 2 上传成功 3 上传失败 | ||||
|       } | ||||
|       virtualList.value.unshift(newItem) | ||||
|       uploadsStore.updateUploadStatus(true) | ||||
|       uploadsStore.initUploadFile( | ||||
|         res.tempFiles[0], | ||||
|         props.talkParams, | ||||
|         randomId, | ||||
|         (status, data, msg) => { | ||||
|           if (status === 0) { | ||||
|             // 更新上传状态为成功 | ||||
|             const index = virtualList.value.findIndex( | ||||
|               (item) => item.file_num === randomId, | ||||
|             ) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 2 | ||||
|               virtualList.value[index].uploadCurrent = 100 | ||||
|             } | ||||
|           } else { | ||||
|             uploadsStore.updateUploadStatus(false) | ||||
|             // 更新上传状态为失败 | ||||
|             const index = virtualList.value.findIndex( | ||||
|               (item) => item.file_num === randomId, | ||||
|             ) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 3 | ||||
|             } | ||||
|             message.error(msg) | ||||
|       uploadsStore.initUploadFile(res.tempFiles[0], props.talkParams, randomId, (status, data, msg) => { | ||||
|         if (status === 0) { | ||||
|           // 更新上传状态为成功 | ||||
|           const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|           if (index !== -1) { | ||||
|             virtualList.value[index].uploadStatus = 2 | ||||
|             virtualList.value[index].uploadCurrent = 100 | ||||
|           } | ||||
|         }, | ||||
|       ) | ||||
|         } else { | ||||
|           // 更新上传状态为失败 | ||||
|           const index = virtualList.value.findIndex(item => item.file_num === randomId) | ||||
|           if (index !== -1) { | ||||
|             virtualList.value[index].uploadStatus = 3 | ||||
|           } | ||||
|           message.error(msg) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
|  | ||||
| @ -37,7 +37,7 @@ | ||||
|           </div> | ||||
|           <div class="user-info-main user-info-card"> | ||||
|             <div class="user-info-main-title"> | ||||
|               <img src="@/static/image/mine/ming001@3x.png" /> | ||||
|               <img src="/src/static/image/mine/ming001@3x.png" /> | ||||
|               <span class="text-[28rpx] font-medium"> | ||||
|                 {{ $t('index.mine.basic') }} | ||||
|               </span> | ||||
| @ -61,17 +61,13 @@ | ||||
|         <template #bottom> | ||||
|           <customBtn | ||||
|             :isBottom="true" | ||||
|             :btnText=" | ||||
|               state.canSendMsg | ||||
|                 ? $t('user.detail.sendMsg') | ||||
|                 : $t('addressBook.btns.addFriend') | ||||
|             " | ||||
|             :btnText="$t('user.detail.sendMsg')" | ||||
|             :subBtnText=" | ||||
|               !state.canSendMsg || state.userInfo.sys_id === state.uid | ||||
|               state.userInfo.sys_id === state.uid | ||||
|                 ? '' | ||||
|                 : $t('user.detail.ringBell') | ||||
|             " | ||||
|             @clickBtn="checkSendPermission" | ||||
|             @clickBtn="toTalkUser" | ||||
|             @clickSubBtn="handleCall" | ||||
|           ></customBtn> | ||||
|         </template> | ||||
| @ -88,7 +84,7 @@ | ||||
|         <div class="do-phone-call-header"> | ||||
|           <span>{{ $t('popup.title.phone') }}</span> | ||||
|           <img | ||||
|             src="@/static/image/login/check-circle-filled@3x.png" | ||||
|             src="/src/static/image/login/check-circle-filled@3x.png" | ||||
|             @click="hidePhoneCallPopup" | ||||
|           /> | ||||
|         </div> | ||||
| @ -116,7 +112,6 @@ const talkStore = useTalkStore() | ||||
| const userStore = useUserStore() | ||||
| 
 | ||||
| import { getUserInfoByClickAvatar } from '@/api/user/index' | ||||
| import { ServeCheckFriend, ServeAddFriend } from '@/api/addressBook/index' | ||||
| 
 | ||||
| import { useI18n } from 'vue-i18n' | ||||
| const { t } = useI18n() | ||||
| @ -128,7 +123,6 @@ const state = reactive({ | ||||
|   isShowPhoneCall: false, //是否显示电话拨号弹框 | ||||
|   phoneNumber: '', //手机号 | ||||
|   uid: computed(() => userStore.uid), //当前用户id | ||||
|   canSendMsg: false, //是否可以发送消息(是好友或同公司别),如果不可以就需要先加好友 | ||||
| }) | ||||
| 
 | ||||
| onLoad((options) => { | ||||
| @ -151,7 +145,6 @@ const getUserInfo = () => { | ||||
|     console.log(data) | ||||
|     if (code == 200) { | ||||
|       state.userInfo = data | ||||
|       checkNeedAddFriend() | ||||
|       let department = '' | ||||
|       let post = '' | ||||
|       if (data?.erp_dept_position?.length > 0) { | ||||
| @ -244,45 +237,6 @@ const doPhoneCall = () => { | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //检查是否可以发送消息,如果不可以要先添加好友 | ||||
| const checkSendPermission = () => { | ||||
|   if (state.canSendMsg) { | ||||
|     toTalkUser() | ||||
|   } else { | ||||
|     doAddFriend() | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| //校验是否需要加好友 | ||||
| const checkNeedAddFriend = () => { | ||||
|   state.canSendMsg = true | ||||
|   // let params = { | ||||
|   //   receiver_id: state.userInfo.sys_id, //聊天的用户id | ||||
|   //   talk_type: 1, | ||||
|   // } | ||||
|   // ServeCheckFriend(params).then((res) => { | ||||
|   //   console.log(res) | ||||
|   //   if (res?.code === 200) { | ||||
|   //     state.canSendMsg = res.data?.is_friend || false | ||||
|   //   } | ||||
|   // }) | ||||
| } | ||||
| 
 | ||||
| //主动加好友(单向好友) | ||||
| const doAddFriend = () => { | ||||
|   let params = { | ||||
|     receiver_id: state.userInfo.sys_id, //聊天的用户id | ||||
|     talk_type: 1, | ||||
|   } | ||||
|   ServeAddFriend(params).then((res) => { | ||||
|     console.log(res) | ||||
|     if (res?.code === 200) { | ||||
|       message.success(t('addressBook.message.addSuccess') + ' !') | ||||
|       state.canSendMsg = true | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .outer-layer { | ||||
| @ -325,7 +279,6 @@ const doAddFriend = () => { | ||||
|         flex-direction: row; | ||||
|         align-items: center; | ||||
|         justify-content: center; | ||||
|         flex-shrink: 0; | ||||
|         img { | ||||
|           width: 100%; | ||||
|           height: 100%; | ||||
|  | ||||
| @ -1,17 +1,14 @@ | ||||
| <script lang="ts" setup> | ||||
| import zPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import { ref, onMounted } from 'vue' | ||||
| import { ServeGetForwardRecords } from '@/api/chat' | ||||
| import { MessageComponents } from '@/constant/message' | ||||
| import { ITalkRecord } from '@/types/chat' | ||||
| import WdLoading from "@/uni_modules/wot-design-uni/components/wd-loading/wd-loading.vue"; | ||||
| import { useInject } from '@/hooks' | ||||
| import { parseTime } from '@/utils/datetime' | ||||
| 
 | ||||
| const emit = defineEmits(['close']) | ||||
| 
 | ||||
| const msgId = ref(null) | ||||
| const createdAt = ref(null) | ||||
| const { showUserInfoModal } = useInject() | ||||
| const isShow = ref(true) | ||||
| const items = ref<ITalkRecord[]>([]) | ||||
| @ -23,8 +20,7 @@ const onMaskClick = () => { | ||||
| 
 | ||||
| const onLoadData = () => { | ||||
|   ServeGetForwardRecords({ | ||||
|     msg_id: msgId.value, | ||||
|     biz_date: createdAt.value | ||||
|     msg_id: msgId.value | ||||
|   }).then((res) => { | ||||
|     if (res.code == 200) { | ||||
|       items.value = res.data.items || [] | ||||
| @ -44,7 +40,6 @@ onMounted(() => { | ||||
|   const page = pages[pages.length - 1] | ||||
|   const options = page.$page.options | ||||
|   msgId.value = options.msgId | ||||
|   createdAt.value = parseTime(new Date((options.created_at as any)), '{y}{m}') | ||||
|   console.log(msgId.value,'msgId.value'); | ||||
| 
 | ||||
|   onLoadData() | ||||
| @ -52,12 +47,12 @@ onMounted(() => { | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="forward-record-page"> | ||||
|     <zPaging ref="zPaging" :show-scrollbar="false"> | ||||
|       <template #top> | ||||
|         <customNavbar :title="`${title}的会话记录`"></customNavbar> | ||||
|       </template> | ||||
|       <div class="main-box"> | ||||
|   <div class="outer-layer"> | ||||
|     <div> | ||||
|       <tm-navbar :hideBack="false" hideHome :title="`${title}的会话记录`" > | ||||
|       </tm-navbar> | ||||
|     </div> | ||||
|     <div class="main-box"> | ||||
|       <div v-if="items.length === 0" class="flex justify-center items-center w-full mt-[200rpx]"> | ||||
|         <wd-loading  /> | ||||
|       </div> | ||||
| @ -81,15 +76,16 @@ onMounted(() => { | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     </zPaging> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .forward-record-page { | ||||
| .outer-layer { | ||||
|   overflow-y: auto; | ||||
|   flex: 1; | ||||
|   background-image: url("@/static/image/clockIn/z3280@3x.png"); | ||||
|   background-size: cover; | ||||
|   padding: 0 66rpx 20rpx 50rpx; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
| } | ||||
| @ -100,7 +96,7 @@ onMounted(() => { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   overflow: auto; | ||||
|   padding: 28rpx 66rpx 20rpx 50rpx; | ||||
|   padding-top: 28rpx; | ||||
| } | ||||
| 
 | ||||
| .message-item { | ||||
|  | ||||
| @ -7,9 +7,9 @@ | ||||
|       > | ||||
|         <div class="avatarImg"> | ||||
|           <tm-badge | ||||
|             :count="props.data.is_disturb === 0 ? props.data.unread_num : ''" | ||||
|             :count="props.data.is_disturb === 1 ? props.data.unread_num : ''" | ||||
|             :dot=" | ||||
|               props.data.is_disturb === 1 && props.data.unread_num | ||||
|               props.data.is_disturb === 0 && props.data.unread_num | ||||
|                 ? true | ||||
|                 : false | ||||
|             " | ||||
| @ -65,9 +65,6 @@ | ||||
|           </div> | ||||
|           <div class="chatInfo_2 w-full mr-[6rpx]"> | ||||
|             <div class="w-full chatInfo_2_1 textEllipsis"> | ||||
|               <span v-if="props.data.atsign_num" style="color: red;"> | ||||
|                 [有人@你] | ||||
|               </span> | ||||
|               {{ props.data.msg_text }} | ||||
|             </div> | ||||
|           </div> | ||||
| @ -110,9 +107,6 @@ import { useSessionMenu } from '@/hooks' | ||||
| const talkStore = useTalkStore() | ||||
| const { onToTopTalk, onRemoveTalk } = useSessionMenu() | ||||
| const dialogueStore = useDialogueStore() | ||||
| const dialogueParams = reactive({ | ||||
|   unReadNum: computed(() => dialogueStore.unreadNum), | ||||
| }) | ||||
| const props = defineProps({ | ||||
|   data: { | ||||
|     type: Object, | ||||
| @ -138,18 +132,14 @@ const cellClick = () => { | ||||
| 
 | ||||
|   // 清空消息未读数 | ||||
|   if (props.data.unread_num > 0) { | ||||
|     ServeClearTalkUnreadNum( | ||||
|       { | ||||
|         talk_type: props.data.talk_type, | ||||
|         receiver_id: props.data.receiver_id, | ||||
|       }, | ||||
|       dialogueParams.unReadNum, | ||||
|     ).then(() => { | ||||
|     ServeClearTalkUnreadNum({ | ||||
|       talk_type: props.data.talk_type, | ||||
|       receiver_id: props.data.receiver_id, | ||||
|     }).then(() => { | ||||
|       talkStore.updateItem({ | ||||
|         index_name: props.data.index_name, | ||||
|         unread_num: 0, | ||||
|       }) | ||||
|       dialogueStore.clearUnreadNum() | ||||
|     }) | ||||
|   } | ||||
|   uni.navigateTo({ | ||||
| @ -265,7 +255,6 @@ const handleDelete = () => { | ||||
|   display: -webkit-box; | ||||
|   -webkit-line-clamp: 1; | ||||
|   -webkit-box-orient: vertical; | ||||
|   word-break: break-all; | ||||
| } | ||||
| 
 | ||||
| .divider { | ||||
|  | ||||
| @ -31,25 +31,12 @@ | ||||
|           > | ||||
|             <template v-slot:left> | ||||
|               <div class="flex items-center ml-[48rpx]"> | ||||
|                 <!-- <image | ||||
|                 <image | ||||
|                   class="w-[72rpx] h-[72rpx]" | ||||
|                   style="border-radius: 50%;" | ||||
|                   style="border-radius: 50%" | ||||
|                   :src="userStore.avatar" | ||||
|                   mode="scaleToFill" | ||||
|                 /> --> | ||||
|                 <avatarModule | ||||
|                   :mode="1" | ||||
|                   :avatar="userStore.avatar" | ||||
|                   :groupType="0" | ||||
|                   :userName="userStore.nickname" | ||||
|                   :customStyle="{ width: '72rpx', height: '72rpx' }" | ||||
|                   :customTextStyle="{ | ||||
|                     fontSize: '32rpx', | ||||
|                     fontWeight: 'bold', | ||||
|                     color: '#fff', | ||||
|                     lineHeight: '44rpx', | ||||
|                   }" | ||||
|                 ></avatarModule> | ||||
|                 /> | ||||
|                 <div class="ml-[24rpx] text-[36rpx] font-bold"> | ||||
|                   {{ userStore.nickname }} | ||||
|                 </div> | ||||
| @ -60,17 +47,17 @@ | ||||
|                 <tm-popover position="br" color="#333333" :width="260"> | ||||
|                   <image | ||||
|                     class="w-[48rpx] h-[48rpx]" | ||||
|                     style="border-radius: 50%;" | ||||
|                     style="border-radius: 50%" | ||||
|                     :src="addCircle" | ||||
|                     mode="scaleToFill" | ||||
|                   /> | ||||
|                   <template v-slot:label> | ||||
|                     <div | ||||
|                       class="w-full  px-[14rpx]" | ||||
|                       class="w-full h-[208rpx] pt-[22rpx] pb-[32rpx] pl-[14rpx] pr-[12rpx]" | ||||
|                     > | ||||
|                       <div | ||||
|                         @click="creatGroupChat" | ||||
|                         class="flex items-center pl-[22rpx] py-[32rpx]" | ||||
|                         class="flex items-center pl-[22rpx] mb-[32rpx]" | ||||
|                       > | ||||
|                         <div class="mr-[26rpx] flex items-center"> | ||||
|                           <tm-image | ||||
| @ -86,27 +73,9 @@ | ||||
|                         </div> | ||||
|                       </div> | ||||
|                       <div class="divider"></div> | ||||
|                       <!-- <div | ||||
|                         @click="toAddFriendPage" | ||||
|                         class="flex items-center pl-[22rpx] py-[32rpx]" | ||||
|                       > | ||||
|                         <div class="mr-[26rpx] flex items-center"> | ||||
|                           <tm-image | ||||
|                             :width="40" | ||||
|                             :height="44" | ||||
|                             :src="addFriend" | ||||
|                           ></tm-image> | ||||
|                         </div> | ||||
|                         <div | ||||
|                           class="leading-[54rpx] text-[32rpx] text-[#FFFFFF] font-bold" | ||||
|                         > | ||||
|                           添加好友 | ||||
|                         </div> | ||||
|                       </div> | ||||
|                       <div class="divider"></div> --> | ||||
|                       <div | ||||
|                         @click="toAddressBookPage" | ||||
|                         class="flex items-center pl-[22rpx] py-[32rpx]" | ||||
|                         class="flex items-center pl-[22rpx] mt-[32rpx]" | ||||
|                       > | ||||
|                         <div class="mr-[26rpx] flex items-center"> | ||||
|                           <tm-image | ||||
| @ -151,36 +120,32 @@ | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import customInput from '@/components/custom-input/custom-input.vue' | ||||
| import { ref, watch, computed } from 'vue' | ||||
| import { onShow, onLoad } from '@dcloudio/uni-app' | ||||
| import { useChatList } from '@/store/chatList/index.js' | ||||
| import { useAuth } from '@/store/auth' | ||||
| import { ServeClearTalkUnreadNum } from '@/api/chat' | ||||
| import { useTalkStore, useUserStore, useDialogueStore } from '@/store' | ||||
| import chatItem from './components/chatItem.vue' | ||||
| import addCircle from '@/static/image/chatList/addCircle.png' | ||||
| import cahtPopover from '@/static/image/chatList/cahtPopover.png' | ||||
| import zu3289 from '@/static/image/chatList/zu3289@2x.png' | ||||
| import addFriend from '@/static/image/chatList/addFriend.png' | ||||
| import ZPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import { handleSetWebviewStyle } from '@/utils/common' | ||||
| const paging = ref() | ||||
| const isEmptyViewShow = ref(false) | ||||
| const talkStore = useTalkStore() | ||||
| const userStore = useUserStore() | ||||
| const dialogueStore = useDialogueStore() | ||||
| const dialogueParams = reactive({ | ||||
|   unReadNum: computed(() => dialogueStore.unreadNum), | ||||
| }) | ||||
| const { userInfo } = useAuth() | ||||
| import customInput from "@/components/custom-input/custom-input.vue"; | ||||
| import { ref, watch, computed } from "vue"; | ||||
| import { onShow, onLoad } from "@dcloudio/uni-app"; | ||||
| import { useChatList } from "@/store/chatList/index.js"; | ||||
| import { useAuth } from "@/store/auth"; | ||||
| import { ServeClearTalkUnreadNum } from "@/api/chat"; | ||||
| import { useTalkStore, useUserStore, useDialogueStore } from "@/store"; | ||||
| import chatItem from "./components/chatItem.vue"; | ||||
| import addCircle from "@/static/image/chatList/addCircle.png"; | ||||
| import cahtPopover from "@/static/image/chatList/cahtPopover.png"; | ||||
| import zu3289 from "@/static/image/chatList/zu3289@2x.png"; | ||||
| import ZPaging from "@/uni_modules/z-paging/components/z-paging/z-paging.vue"; | ||||
| import { handleSetWebviewStyle } from "@/utils/common"; | ||||
| const paging = ref(); | ||||
| const isEmptyViewShow = ref(false); | ||||
| const talkStore = useTalkStore(); | ||||
| const userStore = useUserStore(); | ||||
| const dialogueStore = useDialogueStore(); | ||||
| const { userInfo } = useAuth(); | ||||
| 
 | ||||
| const topItems = computed(() => talkStore.topItems) | ||||
| const items = computed(() => { | ||||
| const items = ref([]); | ||||
| const calcitems = computed(() => { | ||||
|   // if (searchKeyword.value.length === 0) { | ||||
|   console.log(talkStore.talkItems) | ||||
| 
 | ||||
|   return talkStore.talkItems | ||||
|   console.log(talkStore.talkItems); | ||||
|   items.value = JSON.parse(JSON.stringify(talkStore.talkItems)); | ||||
|   return talkStore.talkItems; | ||||
|   // } | ||||
| 
 | ||||
|   // return talkStore.talkItems.filter((item) => { | ||||
| @ -188,72 +153,59 @@ const items = computed(() => { | ||||
| 
 | ||||
|   //   return keyword.toLowerCase().indexOf(searchKeyword.value.toLowerCase()) != -1 | ||||
|   // }) | ||||
| }) | ||||
| }); | ||||
| 
 | ||||
| const queryList = (pageNo, pageSize) => { | ||||
|   // paging.value.complete(res.data.list); | ||||
|   console.log(talkStore) | ||||
|   console.log(talkStore); | ||||
|   talkStore | ||||
|     .loadTalkList() | ||||
|     .then(() => { | ||||
|       isEmptyViewShow.value = talkStore.talkItems.length === 0 | ||||
|       isEmptyViewShow.value = talkStore.talkItems.length === 0; | ||||
|       // 数据加载成功 | ||||
|       paging.value.complete(talkStore.talkItems) | ||||
|       paging.value.complete(talkStore.talkItems); | ||||
|     }) | ||||
|     .catch((error) => { | ||||
|       // 数据加载失败 | ||||
|       console.error('加载失败', error) | ||||
|       paging.value.complete(false) | ||||
|     }) | ||||
| } | ||||
|       console.error("加载失败", error); | ||||
|       paging.value.complete(false); | ||||
|     }); | ||||
| }; | ||||
| 
 | ||||
| // 添加下拉刷新处理函数 | ||||
| const onRefresh = () => { | ||||
|   console.log('触发下拉刷新') | ||||
|   console.log("触发下拉刷新"); | ||||
|   talkStore | ||||
|     .loadTalkList() | ||||
|     .then(() => { | ||||
|       // 数据加载成功 | ||||
|       paging.value.complete(talkStore.talkItems) | ||||
|       paging.value.complete(talkStore.talkItems); | ||||
|     }) | ||||
|     .catch((error) => { | ||||
|       // 数据加载失败 | ||||
|       console.error('加载失败', error) | ||||
|       paging.value.complete(false) | ||||
|     }) | ||||
| } | ||||
|       console.error("加载失败", error); | ||||
|       paging.value.complete(false); | ||||
|     }); | ||||
| }; | ||||
| 
 | ||||
| const creatGroupChat = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/creatGroupChat/index', | ||||
|   }) | ||||
| } | ||||
|     url: "/pages/creatGroupChat/index", | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| const toSearchPage = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/search/index', | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //点击跳转到添加好友页面 | ||||
| const toAddFriendPage = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/addressBook/addFriend/index', | ||||
|   }) | ||||
| } | ||||
|     url: "/pages/search/index", | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| //点击跳转到通讯录页面 | ||||
| const toAddressBookPage = () => { | ||||
|   // 旧版本-按组织架构树的通讯录 | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/chooseByDeps/index?chooseMode=3', | ||||
|   }) | ||||
| 
 | ||||
|   // 新版本-按公司别、好友、群组的通讯录 | ||||
|   // uni.navigateTo({ | ||||
|   //   url: '/pages/addressBook/index', | ||||
|   // }) | ||||
| } | ||||
|     url: "/pages/chooseByDeps/index?chooseMode=3", | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| /* watch( | ||||
| 	() => talkStore, | ||||
| @ -264,60 +216,56 @@ const toAddressBookPage = () => { | ||||
| ); */ | ||||
| 
 | ||||
| onShow(() => { | ||||
|   handleSetWebviewStyle(true) | ||||
|   handleSetWebviewStyle(true); | ||||
|   // 页面显示时重新加载数据 | ||||
|   talkStore | ||||
|     .loadTalkList() | ||||
|     .then(() => { | ||||
|       // 如果paging已经初始化,则刷新列表 | ||||
|       if (paging.value) { | ||||
|         paging.value.reload() | ||||
|         paging.value.reload(); | ||||
|       } | ||||
|     }) | ||||
|     .catch((error) => { | ||||
|       console.error('页面显示时数据加载失败', error) | ||||
|     }) | ||||
| }) | ||||
|       console.error("页面显示时数据加载失败", error); | ||||
|     }); | ||||
| }); | ||||
| 
 | ||||
| onLoad((options) => { | ||||
|   console.log(options) | ||||
|   console.log(options); | ||||
| 
 | ||||
|   if (options?.openSessionIndexName) { | ||||
|     if (items?.value?.length > 0) { | ||||
|       items.value.forEach((openSession) => { | ||||
|         if (openSession.index_name === options?.openSessionIndexName) { | ||||
|           setTimeout(() => { | ||||
|             dialogueStore.setDialogue(openSession) | ||||
|             dialogueStore.setDialogue(openSession); | ||||
|             if (openSession.unread_num > 0) { | ||||
|               ServeClearTalkUnreadNum( | ||||
|                 { | ||||
|                   talk_type: openSession.talk_type, | ||||
|                   receiver_id: openSession.receiver_id, | ||||
|                 }, | ||||
|                 dialogueParams.unReadNum, | ||||
|               ).then(() => { | ||||
|               ServeClearTalkUnreadNum({ | ||||
|                 talk_type: openSession.talk_type, | ||||
|                 receiver_id: openSession.receiver_id, | ||||
|               }).then(() => { | ||||
|                 talkStore.updateItem({ | ||||
|                   index_name: openSession.index_name, | ||||
|                   unread_num: 0, | ||||
|                 }) | ||||
|                 dialogueStore.clearUnreadNum() | ||||
|               }) | ||||
|                 }); | ||||
|               }); | ||||
|             } | ||||
|             uni.navigateTo({ | ||||
|               url: `/pages/dialog/index?sessionId=${openSession.id}`, | ||||
|             }) | ||||
|           }, 500) | ||||
|             }); | ||||
|           }, 500); | ||||
|         } | ||||
|       }) | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
| }) | ||||
| }); | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| uni-page-body, | ||||
| page { | ||||
|   height: 100vh; | ||||
|   background-image: url('@/static/image/clockIn/z3280@3x.png'); | ||||
|   background-image: url("@/static/image/clockIn/z3280@3x.png"); | ||||
|   background-size: cover; | ||||
|   flex-direction: column; | ||||
| } | ||||
|  | ||||
| @ -9,7 +9,7 @@ | ||||
|       :default-page-size="props.searchResultPageSize" | ||||
|       :loading-more-default-as-loading="true" | ||||
|       :inside-more="true" | ||||
|       :empty-view-img="searchNoData" | ||||
|       :empty-view-img="'/src/static//image/search/search-no-data.png'" | ||||
|       :empty-view-text="$t('search.hint')" | ||||
|       :empty-view-img-style="{ width: '476rpx', height: '261rpx' }" | ||||
|       :empty-view-title-style="{ | ||||
| @ -113,7 +113,6 @@ | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import searchNoData from '@/static/image/search/search-no-data.png' | ||||
| import customInput from '@/components/custom-input/custom-input.vue' | ||||
| import ZPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import useZPaging from '@/uni_modules/z-paging/components/z-paging/js/hooks/useZPaging.js' | ||||
| @ -191,14 +190,13 @@ onMounted(() => { | ||||
| 
 | ||||
| //输入搜索文本 | ||||
| const inputSearchText = (e) => { | ||||
|   // console.log(e) | ||||
|   if (e.trim() != state.searchText.trim()) { | ||||
|     state.pageNum = 1 | ||||
|     state.searchResult = null // 清空搜索结果 | ||||
|     emits('lastIdChange', 0, 0, 0) | ||||
|   } | ||||
|   state.searchText = e.trim() | ||||
|   if (!e.trim()) { | ||||
|     state.searchResult = null // 清空搜索结果 | ||||
|     emits('lastIdChange', 0, 0, 0) | ||||
|   } | ||||
|   zPaging.value?.reload() | ||||
| @ -263,7 +261,6 @@ const queryAllSearch = (pageNum, searchResultPageSize) => { | ||||
|           item.group_type = 0 | ||||
|         }) | ||||
|       } | ||||
| 
 | ||||
|       let tempGeneral_infos = Array.isArray(data.general_infos) | ||||
|         ? [...data.general_infos] | ||||
|         : data.general_infos | ||||
| @ -272,8 +269,6 @@ const queryAllSearch = (pageNum, searchResultPageSize) => { | ||||
|         data.group_member_infos || [], | ||||
|       ) | ||||
|       data.general_infos = tempGeneral_infos | ||||
| 
 | ||||
|       // 检查数据是否为空 | ||||
|       let isEmpty = true | ||||
|       let dataKeys = Object.keys(data) | ||||
|       let paginationKey = '' | ||||
| @ -283,46 +278,32 @@ const queryAllSearch = (pageNum, searchResultPageSize) => { | ||||
|           isEmpty = false | ||||
|         } | ||||
|       }) | ||||
| 
 | ||||
|       if (isEmpty) { | ||||
|         if (pageNum === 1) { | ||||
|           // 第一页请求且为空,清空结果 | ||||
|           state.searchResult = null | ||||
|         if (pageNum == 1) { | ||||
|           zPaging.value?.complete([]) | ||||
|         } else { | ||||
|           // 加载更多且为空,保持原列表不变 | ||||
|           zPaging.value?.complete( | ||||
|             state.searchResult ? [state.searchResult] : [], | ||||
|           ) | ||||
|           data = state.searchResult | ||||
|           zPaging.value?.complete([data]) | ||||
|         } | ||||
|       } else { | ||||
|         if (props.isPagination) { | ||||
|           if (pageNum === 1) { | ||||
|             // 第一页请求,直接设置新数据 | ||||
|             state.searchResult = data | ||||
|           } else { | ||||
|             // 加载更多,合并数据 | ||||
|             if ( | ||||
|               paginationKey && | ||||
|               Array.isArray( | ||||
|                 (state?.searchResult && state?.searchResult[paginationKey]) || | ||||
|                   [], | ||||
|               ) | ||||
|             ) { | ||||
|               data[paginationKey] = state.searchResult[paginationKey].concat( | ||||
|                 data[paginationKey], | ||||
|               ) | ||||
|             } | ||||
|             state.searchResult = data | ||||
|           if ( | ||||
|             paginationKey && | ||||
|             Array.isArray( | ||||
|               (state?.searchResult && state?.searchResult[paginationKey]) || [], | ||||
|             ) && | ||||
|             ((state?.searchResult && state?.searchResult[paginationKey]) || []) | ||||
|               .length > 0 | ||||
|           ) { | ||||
|             data[paginationKey] = state.searchResult[paginationKey].concat( | ||||
|               data[paginationKey], | ||||
|             ) | ||||
|           } | ||||
| 
 | ||||
|           emits( | ||||
|             'lastIdChange', | ||||
|             data.last_id, | ||||
|             data.last_group_id, | ||||
|             data.last_member_id, | ||||
|             data.last_receiver_user_name, | ||||
|             data.last_receiver_group_name, | ||||
|           ) | ||||
|           let total = data.count | ||||
|           if (props.searchRecordDetail) { | ||||
| @ -331,99 +312,28 @@ const queryAllSearch = (pageNum, searchResultPageSize) => { | ||||
|             } else if (state?.first_talk_record_infos?.talk_type === 2) { | ||||
|               total = data.group_record_count | ||||
|             } | ||||
|             let noMoreSearchResultRecord = true | ||||
|             if ( | ||||
|               Object.keys(data).includes('talk_record_infos') && | ||||
|               state.searchResult['talk_record_infos']?.length > 0 | ||||
|             ) { | ||||
|               //搜聊天记录详情 | ||||
|               if (state.searchResult['talk_record_infos']?.length < total) { | ||||
|                 noMoreSearchResultRecord = false | ||||
|               } | ||||
|             } | ||||
|             zPaging.value?.completeByNoMore([data], noMoreSearchResultRecord) | ||||
|           } else { | ||||
|             let noMoreSearchResultUser = true | ||||
|             let noMoreSearchResultGroup = true | ||||
|             let noMoreSearchResultGeneral = true | ||||
|             if ( | ||||
|               Object.keys(data).includes('user_infos') && | ||||
|               state.searchResult['user_infos']?.length > 0 | ||||
|             ) { | ||||
|               //搜人 | ||||
|               if (state.searchResult['user_infos']?.length < total) { | ||||
|                 noMoreSearchResultUser = false | ||||
|               } | ||||
|             } | ||||
|             if ( | ||||
|               Object.keys(data).includes( | ||||
|                 'group_member_infos' || 'group_infos', | ||||
|               ) && | ||||
|               state.searchResult['combinedGroup']?.length > 0 | ||||
|             ) { | ||||
|               //搜群 | ||||
|               if (state.searchResult['combinedGroup']?.length < total) { | ||||
|                 noMoreSearchResultGroup = false | ||||
|               } | ||||
|             } | ||||
|             if ( | ||||
|               Object.keys(data).includes('general_infos') && | ||||
|               state.searchResult['general_infos']?.length > 0 | ||||
|             ) { | ||||
|               //搜聊天记录 | ||||
|               if (state.searchResult['general_infos']?.length < total) { | ||||
|                 noMoreSearchResultGeneral = false | ||||
|               } | ||||
|             } | ||||
|             // zPaging.value?.completeByTotal([data], total) | ||||
|             zPaging.value?.completeByNoMore( | ||||
|               [data], | ||||
|               noMoreSearchResultUser && | ||||
|                 noMoreSearchResultGroup && | ||||
|                 noMoreSearchResultGeneral, | ||||
|             ) | ||||
|           } | ||||
|           zPaging.value?.completeByTotal([data], total) | ||||
|         } else { | ||||
|           state.searchResult = data | ||||
|           zPaging.value?.complete([data]) | ||||
|         } | ||||
|       } | ||||
|       state.searchResult = data | ||||
|     } else { | ||||
|       if (pageNum === 1) { | ||||
|         // 第一页请求失败,清空结果 | ||||
|         state.searchResult = null | ||||
|         zPaging.value?.complete([]) | ||||
|       } else { | ||||
|         // 加载更多失败,保持原列表不变 | ||||
|         zPaging.value?.complete(state.searchResult ? [state.searchResult] : []) | ||||
|       } | ||||
|       zPaging.value?.complete([]) | ||||
|     } | ||||
|   }) | ||||
| 
 | ||||
|   resp.catch(() => { | ||||
|     if (pageNum === 1) { | ||||
|       // 第一页请求异常,清空结果 | ||||
|       state.searchResult = null | ||||
|       zPaging.value?.complete([]) | ||||
|     } else { | ||||
|       // 加载更多异常,保持原列表不变 | ||||
|       zPaging.value?.complete(state.searchResult ? [state.searchResult] : []) | ||||
|     } | ||||
|     zPaging.value?.complete([]) | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //点击取消搜索 | ||||
| const cancelSearch = () => { | ||||
|   const pages = getCurrentPages() | ||||
|   if (pages.length > 1) { | ||||
|     uni.navigateBack({ | ||||
|       delta: 1, | ||||
|     }) | ||||
|   } else { | ||||
|     uni.reLaunch({ | ||||
|       url: '/pages/index/index', | ||||
|     }) | ||||
|   } | ||||
|   uni.navigateBack({ | ||||
|     delta: 1, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //获取key对应值 | ||||
| @ -493,7 +403,7 @@ const getHasMoreResult = (searchResultKey) => { | ||||
|     case 'general_infos': | ||||
|       if ( | ||||
|         state.searchResult['record_count'] && | ||||
|         state.searchResult['record_count'] >= 3 | ||||
|         state.searchResult['record_count'] > 3 | ||||
|       ) { | ||||
|         has_more_result = t('has_more') + t('chat.type.record') | ||||
|       } | ||||
|  | ||||
| @ -17,9 +17,7 @@ import { ServeSeachQueryAll, ServeGetSessionId } from '@/api/search/index' | ||||
| import { onMounted } from 'vue' | ||||
| import { handleSetWebviewStyle } from '@/utils/common' | ||||
| 
 | ||||
| import { useDialogueStore, useTalkStore } from '@/store' | ||||
| import { ServeCreateTalkList } from '@/api/chat/index.js' | ||||
| import { formatTalkItem } from '@/utils/talk' | ||||
| import { useDialogueStore } from '@/store' | ||||
| 
 | ||||
| const dialogueStore = useDialogueStore() | ||||
| 
 | ||||
| @ -52,18 +50,6 @@ const clickSearchItem = async ( | ||||
|   console.log(talk_type, receiver_id) | ||||
|   const sessionId = await getSessionId(talk_type, receiver_id) | ||||
|   if (searchResultKey === 'user_infos') { | ||||
|     if (useTalkStore().findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { | ||||
|       ServeCreateTalkList({ | ||||
|         talk_type, | ||||
|         receiver_id, | ||||
|         erp_user_id: result.erp_user_id, | ||||
|       }).then(async ({ code, data }) => { | ||||
|         if (code == 200) { | ||||
|           let item = formatTalkItem(data) | ||||
|           useTalkStore().addItem(item) | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|     dialogueStore.setDialogue({ | ||||
|       name: result.nickname, | ||||
|       talk_type: 1, | ||||
| @ -79,7 +65,7 @@ const clickSearchItem = async ( | ||||
|       receiver_id: result.group_id || result.id, | ||||
|     }) | ||||
|     uni.navigateTo({ | ||||
|       url: '/pages/dialog/index?sessionId=' + sessionId, | ||||
|       url: '/pages/dialog/index?sessionId=' + sessionId | ||||
|     }) | ||||
|   } else if (searchResultKey === 'general_infos') { | ||||
|     uni.navigateTo({ | ||||
|  | ||||
| @ -25,9 +25,7 @@ import { | ||||
| } from '@/api/search/index' | ||||
| import { reactive } from 'vue' | ||||
| 
 | ||||
| import { useDialogueStore, useTalkStore } from '@/store' | ||||
| import { ServeCreateTalkList } from '@/api/chat/index.js' | ||||
| import { formatTalkItem } from '@/utils/talk' | ||||
| import { useDialogueStore } from '@/store' | ||||
| 
 | ||||
| const dialogueStore = useDialogueStore() | ||||
| 
 | ||||
| @ -64,8 +62,6 @@ onLoad((options) => { | ||||
|           receiver_id: 0, //查详情的时候需传入 | ||||
|           last_group_id: 0, //最后一条群id | ||||
|           last_member_id: 0, //最后一条用户id | ||||
|           last_receiver_user_name: '', //最后一条用户名 | ||||
|           last_receiver_group_name: '', //最后一条群名 | ||||
|         }), | ||||
|       ) | ||||
|       state.apiRequest = ServeTalkRecord | ||||
| @ -77,19 +73,11 @@ onLoad((options) => { | ||||
| }) | ||||
| 
 | ||||
| //分页查询时,最后一条id变化 | ||||
| const lastIdChange = ( | ||||
|   last_id, | ||||
|   last_group_id, | ||||
|   last_member_id, | ||||
|   last_receiver_user_name, | ||||
|   last_receiver_group_name, | ||||
| ) => { | ||||
| const lastIdChange = (last_id, last_group_id, last_member_id) => { | ||||
|   let idChanges = { | ||||
|     last_id, | ||||
|     last_group_id, | ||||
|     last_member_id, | ||||
|     last_receiver_user_name, | ||||
|     last_receiver_group_name, | ||||
|   } | ||||
|   state.apiParams = encodeURIComponent( | ||||
|     JSON.stringify( | ||||
| @ -116,18 +104,6 @@ const clickSearchItem = async ( | ||||
|   console.log(talk_type, receiver_id) | ||||
|   const sessionId = await getSessionId(talk_type, receiver_id) | ||||
|   if (state.searchResultKey === 'user_infos') { | ||||
|     if (useTalkStore().findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { | ||||
|       ServeCreateTalkList({ | ||||
|         talk_type, | ||||
|         receiver_id, | ||||
|         erp_user_id: result.erp_user_id, | ||||
|       }).then(async ({ code, data }) => { | ||||
|         if (code == 200) { | ||||
|           let item = formatTalkItem(data) | ||||
|           useTalkStore().addItem(item) | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|     dialogueStore.setDialogue({ | ||||
|       name: result.nickname, | ||||
|       talk_type: 1, | ||||
|  | ||||
| @ -42,7 +42,7 @@ | ||||
|               <span class="text-[28rpx] font-regular"> | ||||
|                 {{ state.selectedMonth }} | ||||
|               </span> | ||||
|               <img src="@/static/image/search/down-pointer.png" /> | ||||
|               <img src="/src/static/image/search/down-pointer.png" /> | ||||
|             </div> | ||||
|           </tm-time-picker> | ||||
|           <tm-calendar-view | ||||
| @ -123,10 +123,7 @@ | ||||
|                         state.condition === 'imgAndVideo' ? '0 0 10rpx' : '', | ||||
|                     }" | ||||
|                   > | ||||
|                     <div | ||||
|                       class="condition-result-member" | ||||
|                       v-if="state.condition === 'member'" | ||||
|                     > | ||||
|                     <div class="condition-result-member"> | ||||
|                       <searchItem | ||||
|                         @click="toDialogueByMember(item)" | ||||
|                         :searchResultKey="'search_by_member_condition'" | ||||
| @ -160,27 +157,17 @@ | ||||
|                         class="condition-result-imgAndVideo-area" | ||||
|                         v-if="item?.extra?.url" | ||||
|                       > | ||||
|                         <template v-if="item?.msg_type === 3"> | ||||
|                           <tm-image | ||||
|                             preview | ||||
|                             :src="item?.extra?.url" | ||||
|                             model="aspectFill" | ||||
|                           /> | ||||
|                         </template> | ||||
|                         <template v-else-if="item?.msg_type === 5"> | ||||
|                           <div | ||||
|                             class="video-preview" | ||||
|                             @click="onPlay(item?.extra?.url)" | ||||
|                           > | ||||
|                             <tm-image | ||||
|                               :src="item?.extra?.cover" | ||||
|                               model="aspectFill" | ||||
|                             /> | ||||
|                             <div class="play-icon"> | ||||
|                               <img :src="playCircle" /> | ||||
|                             </div> | ||||
|                           </div> | ||||
|                         </template> | ||||
|                         <tm-image | ||||
|                           preview | ||||
|                           :src=" | ||||
|                             item?.msg_type === 3 | ||||
|                               ? item?.extra?.url | ||||
|                               : item?.msg_type === 5 | ||||
|                               ? item?.extra?.cover | ||||
|                               : '' | ||||
|                           " | ||||
|                           model="aspectFill" | ||||
|                         /> | ||||
|                       </div> | ||||
|                     </div> | ||||
|                     <div | ||||
| @ -212,7 +199,7 @@ | ||||
|                           v-if="state.condition === 'file'" | ||||
|                         /> | ||||
|                         <img | ||||
|                           src="@/static/image/search/result-link-icon.png" | ||||
|                           src="/src/static/image/search/result-link-icon.png" | ||||
|                           v-if="state.condition === 'link'" | ||||
|                         /> | ||||
|                       </div> | ||||
| @ -267,20 +254,6 @@ | ||||
|           </div> | ||||
|         </div> | ||||
|       </ZPaging> | ||||
|       <teleport to="body"> | ||||
|         <div v-show="open" class="video-container"> | ||||
|           <video | ||||
|             :src="currentVideoUrl" | ||||
|             controls | ||||
|             @fullscreenchange="fullscreenchange" | ||||
|             :id="currentVideoUrl" | ||||
|             playsinline | ||||
|             webkit-playsinline | ||||
|             x5-playsinline | ||||
|             class="fullscreen-video" | ||||
|           ></video> | ||||
|         </div> | ||||
|       </teleport> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| @ -290,14 +263,13 @@ import fileType_EXCEL from '@/static/image/search/fileType_EXCEL.png' | ||||
| import fileType_WORD from '@/static/image/search/fileType_WORD.png' | ||||
| import fileType_PDF from '@/static/image/search/fileType_PDF.png' | ||||
| import fileType_Files from '@/static/image/search/fileType_Files.png' | ||||
| import playCircle from '@/static/image/chatList/playCircle@2x.png' | ||||
| import { fileFormatSize, fileSuffix } from '@/utils/strings' | ||||
| import searchItem from '../components/searchItem.vue' | ||||
| import customInput from '@/components/custom-input/custom-input.vue' | ||||
| import ZPaging from '@/uni_modules/z-paging/components/z-paging/z-paging.vue' | ||||
| import useZPaging from '@/uni_modules/z-paging/components/z-paging/js/hooks/useZPaging.js' | ||||
| import { parseTime } from '@/utils/datetime' | ||||
| import { onMounted, reactive, computed, ref, nextTick } from 'vue' | ||||
| import { onMounted, reactive, computed, ref } from 'vue' | ||||
| import { onLoad } from '@dcloudio/uni-app' | ||||
| import { ServeTalkDate, ServeGetSessionId } from '@/api/search/index' | ||||
| import { ServeFindTalkRecords } from '@/api/chat/index' | ||||
| @ -336,39 +308,6 @@ const state = reactive({ | ||||
|   flatList: [], // 用于存储扁平化的数据 | ||||
| }) | ||||
| 
 | ||||
| const videoContext = ref() | ||||
| const open = ref(false) | ||||
| const currentVideoUrl = ref('') | ||||
| 
 | ||||
| const fullscreenchange = (e) => { | ||||
|   if (!e.detail.fullScreen) { | ||||
|     videoContext.value.stop() | ||||
|     videoContext.value.seek(0) | ||||
|     open.value = false | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| async function onPlay(url) { | ||||
|   currentVideoUrl.value = url | ||||
|   open.value = true | ||||
| 
 | ||||
|   // 等待 DOM 更新 | ||||
|   await nextTick() | ||||
| 
 | ||||
|   // 创建新的视频上下文 | ||||
|   videoContext.value = uni.createVideoContext(url, getCurrentInstance()) | ||||
| 
 | ||||
|   setTimeout(() => { | ||||
|     // 先请求全屏 | ||||
|     videoContext.value.requestFullScreen({ direction: 2 }) | ||||
| 
 | ||||
|     // 延迟一下再播放,确保全屏已经完成 | ||||
|     setTimeout(() => { | ||||
|       videoContext.value.play() | ||||
|     }, 100) | ||||
|   }, 200) | ||||
| } | ||||
| 
 | ||||
| onLoad((options) => { | ||||
|   console.log(options) | ||||
|   if (options.condition) { | ||||
| @ -561,16 +500,9 @@ const inputSearchText = (e) => { | ||||
| 
 | ||||
| //点击取消搜索 | ||||
| const cancelSearch = () => { | ||||
|   const pages = getCurrentPages() | ||||
|   if (pages.length > 1) { | ||||
|     uni.navigateBack({ | ||||
|       delta: 1, | ||||
|     }) | ||||
|   } else { | ||||
|     uni.reLaunch({ | ||||
|       url: '/pages/index/index', | ||||
|     }) | ||||
|   } | ||||
|   uni.navigateBack({ | ||||
|     delta: 1, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| //查询数据 | ||||
| @ -897,26 +829,6 @@ body::v-deep .round-3 { | ||||
|                   width: 164rpx !important; | ||||
|                   height: 164rpx !important; | ||||
|                 } | ||||
|                 .video-preview { | ||||
|                   position: relative; | ||||
|                   width: 164rpx; | ||||
|                   height: 164rpx; | ||||
|                   cursor: pointer; | ||||
|                   .play-icon { | ||||
|                     position: absolute; | ||||
|                     top: 50%; | ||||
|                     left: 50%; | ||||
|                     transform: translate(-50%, -50%); | ||||
|                     display: flex; | ||||
|                     align-items: center; | ||||
|                     justify-content: center; | ||||
| 
 | ||||
|                     img { | ||||
|                       width: 80rpx !important; | ||||
|                       height: 80rpx !important; | ||||
|                     } | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|             } | ||||
|           } | ||||
| @ -925,23 +837,4 @@ body::v-deep .round-3 { | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .video-container { | ||||
|   position: fixed; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|   width: 100vw; | ||||
|   height: 100vh; | ||||
|   background: #000; | ||||
|   z-index: 9999; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   justify-content: center; | ||||
| } | ||||
| 
 | ||||
| .fullscreen-video { | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   object-fit: contain; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -23,7 +23,7 @@ class WsSocket { | ||||
|       lockReconnect: false, | ||||
|       setTimeout: null, // 计时器对象
 | ||||
|       time: 3000, // 重连间隔时间
 | ||||
|       number: 20 // 重连次数
 | ||||
|       number: 10000000 // 重连次数
 | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -1,21 +1,5 @@ | ||||
| import Request from '@/service/request/index.js' | ||||
| import {useAuth} from "@/store/auth"; | ||||
| import { createApp } from 'vue'; | ||||
| import XMessage from '@/components/x-message/index.vue'; | ||||
| 
 | ||||
| // 创建消息提示实例
 | ||||
| const messageInstance = (() => { | ||||
|   const messageNode = document.createElement('div'); | ||||
|   document.body.appendChild(messageNode); | ||||
|   const app = createApp(XMessage); | ||||
|   const instance = app.mount(messageNode); | ||||
|   return { | ||||
|     warning: (msg) => instance.showMessage({ type: 'warning', message: msg }), | ||||
|     error: (msg) => instance.showMessage({ type: 'error', message: msg }), | ||||
|     success: (msg) => instance.showMessage({ type: 'success', message: msg }), | ||||
|     info: (msg) => instance.showMessage({ type: 'info', message: msg }) | ||||
|   }; | ||||
| })(); | ||||
| const { token ,refreshToken,userInfo}=useAuth() | ||||
| let isRefreshing = false; | ||||
| let refreshSubscribers = []; | ||||
| @ -40,25 +24,19 @@ const request = new Request({ | ||||
|     }, | ||||
|     responseInterceptors: async (res) => { | ||||
|       if(res.data.status===1){ | ||||
|         // message.warning(res.data.msg)
 | ||||
|         messageInstance.warning(res.data.msg) | ||||
|         message.warning(res.data.msg) | ||||
|       } | ||||
|       if (res.data.status === 401) { | ||||
|         return | ||||
|         // return getRefreshToken(res);
 | ||||
|         return getRefreshToken(res); | ||||
|         // uni.navigateTo({
 | ||||
|         //   url:'/pages/login/index'
 | ||||
|         // })
 | ||||
|       } | ||||
|       if ([200, 201, 204].includes(res.status)) { | ||||
|         if(res.data.code !== 200 && res.data.code!== 0) { | ||||
|           messageInstance.error(res.data.message || res.data.msg || 'An error occurred.'); | ||||
|         } | ||||
|         return res.config.responseType === 'blob' ? res : res; | ||||
|       } else { | ||||
|       /*  message.error(res.data.msg || 'An error occurred.');*/ | ||||
|         messageInstance.error(res.data.message || res.data.msg || 'An error occurred.'); | ||||
|         return Promise.reject(new Error(res.data.message || res.data.msg || 'An error occurred.')); | ||||
|         return Promise.reject(new Error(res.data.msg || 'An error occurred.')); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| @ -81,14 +59,13 @@ async function getRefreshToken(response) { | ||||
|           }) | ||||
|           return request.request(response.config); | ||||
|         } else { | ||||
| 
 | ||||
|           messageInstance.error(res.message || res.msg); | ||||
|           message.error(res.message || res.msg); | ||||
|           throw new Error(res.message || res.msg); | ||||
|         } | ||||
|       } catch (error) { | ||||
|         // uni.navigateTo({
 | ||||
|         //   url:'/pages/login/index'
 | ||||
|         // })
 | ||||
|         uni.navigateTo({ | ||||
|           url:'/pages/login/index' | ||||
|         }) | ||||
|         throw error | ||||
|       } finally { | ||||
|         isRefreshing = false; | ||||
| @ -96,9 +73,9 @@ async function getRefreshToken(response) { | ||||
|         refreshSubscribers = []; | ||||
|       } | ||||
|     } else { | ||||
|       // uni.navigateTo({
 | ||||
|       //   url:'/pages/login/index'
 | ||||
|       // })
 | ||||
|       uni.navigateTo({ | ||||
|         url:'/pages/login/index' | ||||
|       }) | ||||
|       throw new Error('No refresh token available.'); | ||||
|     } | ||||
|   } else { | ||||
|  | ||||
| Before Width: | Height: | Size: 3.3 KiB | 
| Before Width: | Height: | Size: 3.1 KiB | 
| Before Width: | Height: | Size: 3.0 KiB | 
| Before Width: | Height: | Size: 3.1 KiB | 
| Before Width: | Height: | Size: 3.0 KiB | 
| Before Width: | Height: | Size: 654 B | 
| Before Width: | Height: | Size: 2.2 KiB | 
| Before Width: | Height: | Size: 2.2 KiB | 
| Before Width: | Height: | Size: 3.2 KiB | 
| Before Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 1.4 KiB | 
| Before Width: | Height: | Size: 1.5 KiB | 
| Before Width: | Height: | Size: 1.5 KiB | 
| Before Width: | Height: | Size: 1.7 KiB | 
| @ -4,8 +4,8 @@ import { userInfoApi } from "@/api/user"; | ||||
| 
 | ||||
| import {ref} from 'vue' | ||||
| export const useAuth = createGlobalState(() => { | ||||
|   const token = useStorage('token', '', uniStorage) | ||||
|   // const token = ref("2046c3941ed4959f6d988d3d4a0fe40d4b52f33f3f5fc1001406064554641d9406bb13cacb92939b0ca223c17e2c2f2fe70212ef017dbae8965d5cf86bad48ce4316e605ca187bd9ffd4aa6b56865be4ad4e422701d330b52d60cfe649cd48cf3a21a2a6e9a9cabafff364ee9c311ec634b0afc09db0d3215bedce561e9d50e5a8da6092062e2ebe35f747d77d72a68ad492a4ab218c07887c9cd4867f2c2d28e4ae1fd671144cc20ef0632f9ce067289004d67f6adf41b20d6ef5cdbfb74aadc2d2736ececf07254f1a76552bde4f1161a0fca7bfe32a29685ce1e76366116b81ae2195b3713dbb04285e5ddfd36184fe671c5524d20b4fe74a555db755f8d939b0bc46fb0cb998323d54c9925729d7ca835b7925999a677faa0cbe1cbc67b5203d85317653883aec81d3e71d865b326376bea726cc66d9f7f5a160d43f671c")
 | ||||
|   // const token = useStorage('token', '', uniStorage)
 | ||||
|   const token = ref("b66054d4f8a80fd93f603224e16999b7ff9db376a89af5919c99bc45a9c00e760163364c3ef3cd2e1370a90dcac5a68d1af25895841dbe71069a9f4f90445b494b35958bb588441f74cf15932a73ef5f379871fc884982e36a72d3cdc83ad96e085288eecb0df88982aa30cb76469d404f210abe0283e52f2e1b602dfb88a0bb23acaf249ea2aef4b58f4ccaa2ca73dd62be30396431982303995e77d9a0fff14a1c3b27407c19c890687b9a0721ac3d1405981d69f3da64edb0923f73be7ed239757c031c02f8df2c3daa7ffff11bf8e0de37dd627730a14a919bd8c57fca4e8f2d5c6f6f416390fafb030830f2d94ada42d5b04d6724c56cb2dd9a78f91ea8c0e13b19e8e72cb60f84cb1808c810a3e5fc37e8a78b9f9f8ca59ef9f7195acf59e68f4de08db97e79ed13583dd7ea70b401a68bd4aa7185768b1cc455937080") | ||||
|   const refreshToken = useStorage('refreshToken', '', uniStorage) | ||||
|   const userInfo = useStorage('userInfo', {}, uniStorage) | ||||
|   const leaderList = useStorage('leaderList', [], uniStorage) | ||||
|  | ||||
| @ -13,7 +13,6 @@ import { | ||||
|   departmentV2TreeAll2, | ||||
|   userHasPermission, | ||||
|   userV2List, | ||||
|   userV2List2, | ||||
|   v2TreePositionByDepartment, | ||||
| } from '@/api/deps/index.js' | ||||
| import { useAuth } from '@/store/auth' | ||||
| @ -38,36 +37,34 @@ export const useGroupTypeStore = createGlobalState(() => { | ||||
|       depTreeMyList.value = res.data.nodes | ||||
|     } | ||||
|   } | ||||
|   // userInfo?.value?.ID
 | ||||
|   const getDepsTreeMy2 = async (chooseMode) => { | ||||
|     let params = { nowUserId: 0 } | ||||
|     if (chooseMode === 1) { | ||||
|       const isHasRes = await userHasPermission({ | ||||
|         erpUserId: userInfo?.value?.ID, | ||||
|         ruleUrl: [ | ||||
|           'auth_chat_app_create_all_dept', | ||||
|           'auth_chat_app_create_limit_dept', | ||||
|         ], | ||||
|       }) | ||||
|       if (isHasRes.code === 200) { | ||||
|         if (isHasRes.data.auth_chat_app_create_all_dept) { | ||||
|           params = { | ||||
|             nowUserId: 0, | ||||
| // userInfo?.value?.ID
 | ||||
|   const getDepsTreeMy2 = async () => { | ||||
|     const isHasRes = await userHasPermission({ | ||||
|       erpUserId: userInfo?.value?.ID, | ||||
|       ruleUrl: [ | ||||
|         "auth_chat_app_create_all_dept", | ||||
|         "auth_chat_app_create_limit_dept" | ||||
|         ] | ||||
|     }) | ||||
|     let pramas = { nowUserId : 0 } | ||||
|     if (isHasRes.code === 200) { | ||||
|       if (isHasRes.data.auth_chat_app_create_all_dept) { | ||||
|         pramas = { | ||||
|           nowUserId : 0 | ||||
|         } | ||||
|       } else{ | ||||
|         if (isHasRes.data.auth_chat_app_create_limit_dept) { | ||||
|           pramas = { | ||||
|             nowUserId : userInfo?.value?.ID | ||||
|           } | ||||
|         } else { | ||||
|           if (isHasRes.data.auth_chat_app_create_limit_dept) { | ||||
|             params = { | ||||
|               nowUserId: userInfo?.value?.ID, | ||||
|             } | ||||
|           } else { | ||||
|             params = { | ||||
|               nowUserId: 0, | ||||
|             } | ||||
|           pramas = { | ||||
|             nowUserId : 0 | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     const res = await departmentV2TreeAll2(params) | ||||
|     const res = await departmentV2TreeAll2(pramas) | ||||
|     if (res.status === 0) { | ||||
|       depTreeMyList.value = res.data.nodes | ||||
|     } | ||||
| @ -89,7 +86,7 @@ export const useGroupTypeStore = createGlobalState(() => { | ||||
|   } | ||||
| 
 | ||||
|   const getDepMembers = async (param) => { | ||||
|     const res = await userV2List2(param) | ||||
|     const res = await userV2List(param) | ||||
|     return res | ||||
|   } | ||||
| 
 | ||||
|  | ||||
| @ -15,10 +15,6 @@ import { useAuth } from '../auth/index' | ||||
| // let keyboardTimeout = null
 | ||||
| 
 | ||||
| export const useDialogueStore = defineStore('dialogue', { | ||||
|   // 添加持久化配置
 | ||||
|   persist: { | ||||
|     paths: ['talk.talk_type', 'talk.receiver_id'] | ||||
|   }, | ||||
|   state: () => { | ||||
|     return { | ||||
|       // 对话索引(聊天对话的唯一索引)
 | ||||
| @ -55,12 +51,6 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|       //是否已被解散
 | ||||
|       isDismiss: false, | ||||
| 
 | ||||
|       //是否退群/移出群
 | ||||
|       isQuit: false, | ||||
| 
 | ||||
|       //未读消息数量
 | ||||
|       unreadNum:0, | ||||
| 
 | ||||
|       // 群成员列表
 | ||||
|       members: [], | ||||
| 
 | ||||
| @ -97,21 +87,6 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|       this.online = status | ||||
|     }, | ||||
| 
 | ||||
|     // 更新未读消息数量-清空未读
 | ||||
|     clearUnreadNum() { | ||||
|       this.unreadNum = 0 | ||||
|     }, | ||||
| 
 | ||||
|     // 更新群解散状态
 | ||||
|     updateDismiss() { | ||||
|       this.isDismiss = true | ||||
|     }, | ||||
| 
 | ||||
|     // 更新群成员退出状态
 | ||||
|     updateQuit() { | ||||
|       this.isQuit = true | ||||
|     }, | ||||
| 
 | ||||
|     // 更新对话信息
 | ||||
|     setDialogue(data = {}) { | ||||
|       this.online = data.is_online == 1 | ||||
| @ -127,12 +102,9 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|       this.isShowEditor = data?.is_robot === 0 | ||||
| 
 | ||||
|       this.isDismiss = data?.is_dismiss === 1 ? true : false | ||||
|       this.isQuit = data?.is_quit === 1 ? true : false | ||||
| 
 | ||||
|       this.unreadNum = data?.unread_num || 0 | ||||
| 
 | ||||
|       this.members = [] | ||||
|       if (data.talk_type == 2 && !this.isDismiss && !this.isQuit) { | ||||
|       if (data.talk_type == 2) { | ||||
|         this.updateGroupMembers() | ||||
|       } | ||||
|     }, | ||||
| @ -242,6 +214,8 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|         if (res.code == 200) { | ||||
|           this.batchDelDialogueRecord(msgIds) | ||||
|           batchDelDialogueRecord(msgIds) | ||||
|         } else { | ||||
|           message.warning(res.message) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
| @ -265,6 +239,8 @@ export const useDialogueStore = defineStore('dialogue', { | ||||
|       ServeRevokeRecords({ msg_id }).then((res) => { | ||||
|         if (res.code == 200) { | ||||
|           this.updateDialogueRecord({ msg_id, is_revoke: 1 }) | ||||
|         } else { | ||||
|           message.warning(res.message) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
|  | ||||
| @ -5,111 +5,10 @@ import lodash from 'lodash' | ||||
| import { ref } from 'vue' | ||||
| import { createGlobalState, useStorage } from '@vueuse/core' | ||||
| import { uniStorage } from '@/utils/uniStorage.js' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| 
 | ||||
| export const useDialogueListStore = createGlobalState(() => { | ||||
|   const testDatabase = async () => { | ||||
|     // 初始化数据库
 | ||||
|     let chatDatabase = { | ||||
|       eventType: 'openDatabase', | ||||
|       eventParams: { | ||||
|         name: 'chat', | ||||
|         path: '_doc/chat.db', | ||||
|       }, | ||||
|     } | ||||
|     let chatDBexecuteSql = { | ||||
|       eventType: 'executeSql', | ||||
|       eventParams: { | ||||
|         name: 'chat', | ||||
|         sql: `CREATE TABLE IF NOT EXISTS talk_records (
 | ||||
|           id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||
|           msg_id TEXT NOT NULL, | ||||
|           sequence INTEGER NOT NULL, | ||||
|           talk_type INTEGER NOT NULL DEFAULT 1, | ||||
|           msg_type INTEGER NOT NULL DEFAULT 1, | ||||
|           user_id INTEGER NOT NULL DEFAULT 0, | ||||
|           receiver_id INTEGER NOT NULL DEFAULT 0, | ||||
|           is_revoke INTEGER NOT NULL DEFAULT 0, | ||||
|           is_mark INTEGER NOT NULL DEFAULT 0, | ||||
|           quote_id TEXT NOT NULL, | ||||
|           extra TEXT NOT NULL, | ||||
|           created_at TEXT NOT NULL, | ||||
|           updated_at TEXT NOT NULL, | ||||
|           biz_date TEXT | ||||
|         )`,
 | ||||
|       }, | ||||
|     } | ||||
| 
 | ||||
|     const content = { | ||||
|       content: '我试试传送文件和图片是不是一个接口', | ||||
|       name: '测试excel1.xlsx', | ||||
|       path: | ||||
|         'https://cdn-test.szjixun.cn/fonchain-chat/chat/file/multipart/20250307/727a2371-ffc4-46da-b953-a7d449ff82ff-测试excel1.xlsx', | ||||
|       size: 9909, | ||||
|       drive: 3, | ||||
|     } | ||||
|     const extra = JSON.stringify(content) | ||||
| 
 | ||||
|     let chatDBexecuteSql2 = { | ||||
|       eventType: 'executeSql', | ||||
|       eventParams: { | ||||
|         name: 'chat', | ||||
|         sql: | ||||
|           'INSERT INTO talk_records (msg_id, sequence, talk_type, msg_type, user_id, receiver_id, is_revoke, is_mark, quote_id, extra, created_at, updated_at, biz_date) VALUES ("' + | ||||
|           '77b715fb30f54f739a255a915ef72445' + | ||||
|           '", 166, 2, 1, 1774, 888890, 0, 0, "' + | ||||
|           '' + | ||||
|           '", "' + | ||||
|           extra + | ||||
|           '", "' + | ||||
|           '2025-03-06T15:57:07.000Z' + | ||||
|           '", "' + | ||||
|           '2025-03-06T15:57:07.000Z' + | ||||
|           '", "' + | ||||
|           '20250306' + | ||||
|           '")', | ||||
|       }, | ||||
|     } | ||||
|     let chatDBSelectSql = { | ||||
|       eventType: 'selectSql', | ||||
|       eventParams: { | ||||
|         name: 'chat', | ||||
|         sql: `SELECT * FROM talk_records ORDER BY sequence DESC LIMIT 20`, | ||||
|       }, | ||||
|     } | ||||
|     let chatDBIsOpenDatabase = { | ||||
|       eventType: 'isOpenDatabase', | ||||
|       eventParams: { | ||||
|         name: 'chat', | ||||
|         path: '_doc/chat.db', | ||||
|       }, | ||||
|     } | ||||
|     // handleFindWebview(
 | ||||
|     //   `operateSQLite('${encodeURIComponent(JSON.stringify(chatDatabase))}')`,
 | ||||
|     // )
 | ||||
|     // handleFindWebview(
 | ||||
|     //   `operateSQLite('${encodeURIComponent(
 | ||||
|     //     JSON.stringify(chatDBexecuteSql),
 | ||||
|     //   )}')`,
 | ||||
|     // )
 | ||||
|     // handleFindWebview(
 | ||||
|     //   `operateSQLite('${encodeURIComponent(
 | ||||
|     //     JSON.stringify(chatDBexecuteSql2),
 | ||||
|     //   )}')`,
 | ||||
|     // )
 | ||||
|     // handleFindWebview(
 | ||||
|     //   `operateSQLite('${encodeURIComponent(JSON.stringify(chatDBSelectSql))}')`,
 | ||||
|     // )
 | ||||
|     // handleFindWebview(
 | ||||
|     //   `operateSQLite('${encodeURIComponent(
 | ||||
|     //     JSON.stringify(chatDBIsOpenDatabase),
 | ||||
|     //   )}')`,
 | ||||
|     // )
 | ||||
|   } | ||||
|   // testDatabase()
 | ||||
| 
 | ||||
|   const dialogueList = useStorage('dialogueList', [], uniStorage) | ||||
|   // const dialogueList = ref([])
 | ||||
|   // const dialogueList = useStorage('dialogueList', [], uniStorage)
 | ||||
|   const dialogueList = ref([]) | ||||
|   const zpagingRef = ref() | ||||
|   const virtualList = ref([]) | ||||
| 
 | ||||
| @ -118,51 +17,24 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|   } | ||||
| 
 | ||||
|   const addDialogueRecord = (newRecords, type = 'add') => { | ||||
|     console.log(newRecords) | ||||
|     console.log('newRecords',newRecords) | ||||
| 
 | ||||
|     const dialogue = useDialogueStore() | ||||
|     console.log('cloneDeep', new Date().getTime()) | ||||
|     if (!dialogue || typeof dialogue !== 'object') return | ||||
|     console.log('dialogue', new Date().getTime()) | ||||
|     // 检查是否已存在相同 index_name 的对话
 | ||||
|     const existingIndex = dialogueList.value.findIndex( | ||||
|       (item) => item.index_name === dialogue.index_name, | ||||
|     ) | ||||
|     console.log('findIndex', new Date().getTime()) | ||||
|     if (existingIndex === -1) { | ||||
|       // 如果不存在,创建新对话,只保存需要的属性
 | ||||
|       const newDialogue = { | ||||
|         index_name: dialogue.index_name, | ||||
|         talk: { | ||||
|           username: dialogue.talk.username, | ||||
|           talk_type: dialogue.talk.talk_type, | ||||
|           receiver_id: dialogue.talk.receiver_id, | ||||
|         }, | ||||
|         online: dialogue.online, | ||||
|         records: dialogue.records || [], | ||||
|         unreadBubble: dialogue.unreadBubble, | ||||
|         isOpenMultiSelect: dialogue.isOpenMultiSelect, | ||||
|         isShowEditor: dialogue.isShowEditor, | ||||
|         isShowSessionList: dialogue.isShowSessionList, | ||||
|         isDismiss: dialogue.isDismiss, | ||||
|         isQuit: dialogue.isQuit, | ||||
|         unreadNum: dialogue.unreadNum, | ||||
|         members: dialogue.members.map((member) => ({ | ||||
|           id: member.id, | ||||
|           nickname: member.nickname, | ||||
|           avatar: member.avatar, | ||||
|           gender: member.gender, | ||||
|           leader: member.leader, | ||||
|           remark: member.remark, | ||||
|           online: member.online, | ||||
|           value: member.value, | ||||
|           key: member.key, | ||||
|           erp_user_id: member.erp_user_id, | ||||
|           is_mute: member.is_mute, | ||||
|           is_mine: member.is_mine, | ||||
|         })), | ||||
|         forwardType: dialogue.forwardType, | ||||
|       } | ||||
|       dialogueList.value.push(newDialogue) | ||||
|       // 如果不存在,直接添加
 | ||||
|       dialogueList.value.push(dialogue) | ||||
|       console.log('existingIndex', new Date().getTime()) | ||||
|     } else { | ||||
|       // 如果对话存在,处理 records 数组
 | ||||
|       const { records = [] } = dialogue | ||||
|       newRecords.forEach((newRecord) => { | ||||
|         const recordIndex = dialogueList.value[existingIndex].records.findIndex( | ||||
|           (record) => record.msg_id === newRecord.msg_id, | ||||
| @ -171,17 +43,44 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|         if (recordIndex === -1) { | ||||
|           // 如果记录不存在,添加到 records 数组
 | ||||
|           if (type === 'add') { | ||||
|             console.log('add', new Date().getTime()) | ||||
|             dialogueList.value[existingIndex].records.push(newRecord) | ||||
|           } else { | ||||
|             console.log('unshift', new Date().getTime()) | ||||
|             dialogueList.value[existingIndex].records.unshift(newRecord) | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       console.log('existingIndex else', new Date().getTime()) | ||||
|        | ||||
|       // 限制records数组长度为40条
 | ||||
|       const maxRecords = 40 | ||||
|       if (dialogueList.value[existingIndex].records.length > maxRecords) { | ||||
|         console.log('maxRecords') | ||||
|         const excessCount = dialogueList.value[existingIndex].records.length - maxRecords | ||||
|         // 如果是push添加,则从首部删除多余条目
 | ||||
|         if (type === 'add') { | ||||
|           dialogueList.value[existingIndex].records.splice(0, excessCount) | ||||
|         } else { | ||||
|           // 如果是unshift添加,则从尾部删除多余条目
 | ||||
|           dialogueList.value[existingIndex].records.splice(-excessCount) | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       console.log('records',dialogueList.value[existingIndex].records) | ||||
| 
 | ||||
|       // 更新除 records 和 index_name 外的其他属性
 | ||||
|       const { index_name, records: _, ...updateProps } = dialogue | ||||
|       dialogueList.value[existingIndex] = { | ||||
|         ...dialogueList.value[existingIndex], | ||||
|         ...updateProps, | ||||
|       } | ||||
|       console.log('updateProps', new Date().getTime()) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const updateDialogueRecord = (record) => { | ||||
|     const dialogue = useDialogueStore() | ||||
|     const dialogue = lodash.cloneDeep(useDialogueStore()) | ||||
|     const item = getDialogueList(dialogue.index_name) | ||||
|     const recordIndex = item.records.findIndex( | ||||
|       (item) => item.msg_id === record.msg_id, | ||||
| @ -204,7 +103,7 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|   } | ||||
| 
 | ||||
|   const deleteDialogueRecord = (record) => { | ||||
|     const dialogue = useDialogueStore() | ||||
|     const dialogue = lodash.cloneDeep(useDialogueStore()) | ||||
|     const item = getDialogueList(dialogue.index_name) | ||||
|     const recordIndex = item.records.findIndex( | ||||
|       (item) => item.msg_id === record.msg_id, | ||||
| @ -230,7 +129,7 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|   } | ||||
| 
 | ||||
|   const addChatRecord = (indexName, item) => { | ||||
|     const dialogue = useDialogueStore() | ||||
|     const dialogue = lodash.cloneDeep(useDialogueStore()) | ||||
|     if (dialogue?.index_name === indexName) { | ||||
|       if (item?.file_num) { | ||||
|         const index = virtualList.value.findIndex( | ||||
| @ -242,8 +141,6 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|             ...virtualList.value[index], // 保留原有不需要修改的字段
 | ||||
|             ...item, // 覆盖需要更新的字段
 | ||||
|           }) | ||||
|         } else { | ||||
|           zpagingRef.value?.addChatRecordData(item, false, false) | ||||
|         } | ||||
|       } else { | ||||
|         zpagingRef.value?.addChatRecordData(item, false, false) | ||||
| @ -252,7 +149,7 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
|   } | ||||
| 
 | ||||
|   const batchDelDialogueRecord = (msgIds) => { | ||||
|     const dialogue = useDialogueStore() | ||||
|     const dialogue = lodash.cloneDeep(useDialogueStore()) | ||||
|     const item = getDialogueList(dialogue.index_name) | ||||
|     item.records = item.records.filter((item) => !msgIds.includes(item.msg_id)) | ||||
|   } | ||||
| @ -270,7 +167,7 @@ export const useDialogueListStore = createGlobalState(() => { | ||||
| 
 | ||||
|   //清空聊天记录时,同时清空本地保存的聊天记录
 | ||||
|   const clearDialogueRecord = () => { | ||||
|     const dialogue = useDialogueStore() | ||||
|     const dialogue = lodash.cloneDeep(useDialogueStore()) | ||||
|     const item = getDialogueList(dialogue.index_name) | ||||
|     item.records = [] | ||||
|     virtualList.value = [] | ||||
|  | ||||
| @ -41,6 +41,8 @@ export const useEditorStore = defineStore('editor', { | ||||
|       }).then((res) => { | ||||
|         if (res.code == 200) { | ||||
|           this.loadUserEmoticon() | ||||
|         } else { | ||||
|           message.warning(res.message) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
| @ -53,6 +55,8 @@ export const useEditorStore = defineStore('editor', { | ||||
|       ServeUploadEmoticon(data).then((res) => { | ||||
|         if (res.code == 200) { | ||||
|           this.emoticon.items[1].children.unshift(res.data) | ||||
|         } else { | ||||
|           message.warning(res.message) | ||||
|         } | ||||
|       }) | ||||
|     }, | ||||
| @ -65,7 +69,9 @@ export const useEditorStore = defineStore('editor', { | ||||
|         if (res.code == 200) { | ||||
|           this.emoticon.items[1].children.splice(resoure.index, 1) | ||||
|           message.success('删除成功') | ||||
|         }  | ||||
|         } else { | ||||
|           message.warning(res.message) | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @ -2,7 +2,6 @@ import { defineStore } from 'pinia' | ||||
| import { ServeGetTalkList, ServeCreateTalkList } from '@/api/chat/index' | ||||
| import { formatTalkItem, ttime, KEY_INDEX_NAME } from '@/utils/talk' | ||||
| import { useEditorDraftStore } from './editor-draft' | ||||
| import { handleFindWebview } from '@/utils/common' | ||||
| // import { ISession } from '@/types/chat'
 | ||||
| 
 | ||||
| export const useTalkStore = defineStore('talk', { | ||||
| @ -103,8 +102,6 @@ export const useTalkStore = defineStore('talk', { | ||||
|       // 返回 Promise 对象,使调用方可以使用 then/catch
 | ||||
|       return resp.then(({ code, data }) => { | ||||
|         if (code == 200) { | ||||
|           //向OA的webview通信,改变未读消息数量
 | ||||
|           handleFindWebview(`doUpdateUnreadNum('${data.unread_num}')`) | ||||
|           this.items = data.items.map((item) => { | ||||
|             const value = formatTalkItem(item) | ||||
| 
 | ||||
|  | ||||
| @ -33,9 +33,7 @@ export const useUploadsStore = defineStore('uploads', { | ||||
|   state: () => { | ||||
|     return { | ||||
|       isShow: false, | ||||
|       items: [], | ||||
|       isUploading: false,//当前是否正在上传
 | ||||
|       uploadingNum: 0//当前正在上传数量
 | ||||
|       items: [] | ||||
|     } | ||||
|   }, | ||||
|   getters: { | ||||
| @ -77,10 +75,8 @@ export const useUploadsStore = defineStore('uploads', { | ||||
|           this.triggerUpload(upload_id,msgId) | ||||
|           this.isShow = true | ||||
|         } else { | ||||
|           this.updateUploadStatus(false) | ||||
|           message.error(res.message) | ||||
|         } | ||||
|       }).catch(()=> { | ||||
|         this.updateUploadStatus(false) | ||||
|       }) | ||||
|     }, | ||||
| 
 | ||||
| @ -97,19 +93,7 @@ export const useUploadsStore = defineStore('uploads', { | ||||
| 
 | ||||
|       item.status = 1 | ||||
| 
 | ||||
|       // 开始上传时就更新进度
 | ||||
|       const currentPercentage = (item.uploadIndex / item.files.length) * 100 | ||||
|       item.percentage = currentPercentage.toFixed(1) | ||||
|       updateUploadProgress(msgId, currentPercentage) | ||||
| 
 | ||||
|       ServeFileSubareaUpload(form, (progressEvent) => { | ||||
|         // 计算当前分片的进度
 | ||||
|         const currentChunkProgress = (progressEvent.loaded / progressEvent.total) * 100 | ||||
|         // 计算总体进度:已完成分片 + 当前分片的进度
 | ||||
|         const totalProgress = ((item.uploadIndex + currentChunkProgress / 100) / item.files.length) * 100 | ||||
|         item.percentage = totalProgress.toFixed(1) | ||||
|         updateUploadProgress(msgId, totalProgress) | ||||
|       }) | ||||
|       ServeFileSubareaUpload(form) | ||||
|         .then((res) => { | ||||
|           if (res.code == 200) { | ||||
|             item.uploadIndex++ | ||||
| @ -121,38 +105,21 @@ export const useUploadsStore = defineStore('uploads', { | ||||
| 
 | ||||
|               updateUploadProgress(msgId,100) | ||||
|               this.sendUploadMessage(item, msgId) | ||||
| 
 | ||||
|               // 更新虚拟列表中的状态
 | ||||
|               const { virtualList } = useDialogueListStore() | ||||
|               const index = virtualList.value.findIndex(item => item.file_num === msgId) | ||||
|               if (index !== -1) { | ||||
|                 virtualList.value[index].uploadStatus = 2 | ||||
|                 virtualList.value[index].uploadCurrent = 100 | ||||
|               } | ||||
|             } else { | ||||
|               // 继续上传下一个分片
 | ||||
|               this.triggerUpload(uploadId, msgId) | ||||
|               const percentage = (item.uploadIndex / item.files.length) * 100 | ||||
|               item.percentage = percentage.toFixed(1) | ||||
|               console.log(msgId,'msgId'); | ||||
|               console.log(percentage,'percentage'); | ||||
| 
 | ||||
|               updateUploadProgress(msgId,percentage) | ||||
|               this.triggerUpload(uploadId,msgId) | ||||
|             } | ||||
|           } else { | ||||
|             this.updateUploadStatus(false) | ||||
|             item.status = 3 | ||||
|             // 更新虚拟列表中的状态为失败
 | ||||
|             const { virtualList } = useDialogueListStore() | ||||
|             const index = virtualList.value.findIndex(item => item.file_num === msgId) | ||||
|             if (index !== -1) { | ||||
|               virtualList.value[index].uploadStatus = 3 | ||||
|             } | ||||
|           } | ||||
|         }) | ||||
|         .catch(() => { | ||||
|           this.updateUploadStatus(false) | ||||
|           item.status = 3 | ||||
|           // 更新虚拟列表中的状态为失败
 | ||||
|           const { virtualList } = useDialogueListStore() | ||||
|           const index = virtualList.value.findIndex(item => item.file_num === msgId) | ||||
|           if (index !== -1) { | ||||
|             virtualList.value[index].uploadStatus = 3 | ||||
|           } | ||||
|         }) | ||||
|     }, | ||||
| 
 | ||||
| @ -163,38 +130,7 @@ export const useUploadsStore = defineStore('uploads', { | ||||
|         receiver_id: item.receiver_id, | ||||
|         talk_type: item.talk_type, | ||||
|         file_num: file_num | ||||
|       }).then((res) => { | ||||
|         console.log(res, 'res') | ||||
|         if(res.code == 200){ | ||||
|           this.updateUploadStatus(false) | ||||
|         }else{ | ||||
|           this.updateUploadStatus(false) | ||||
|         } | ||||
|       }).catch(() => { | ||||
|         this.updateUploadStatus(false) | ||||
|       }) | ||||
|     }, | ||||
| 
 | ||||
|     //更新资源上传状态
 | ||||
|     updateUploadStatus(isUploading: boolean){ | ||||
|       if(isUploading){ | ||||
|         this.uploadingNum++ | ||||
|         this.isUploading = true | ||||
|       }else{ | ||||
|         this.uploadingNum-- | ||||
|         if(this.uploadingNum < 0){ | ||||
|           this.uploadingNum = 0 | ||||
|         } | ||||
|         if(this.uploadingNum === 0){ | ||||
|           this.isUploading = false | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
| 
 | ||||
|     // 清除上传
 | ||||
|     clearUpload(){ | ||||
|       this.isUploading = false | ||||
|       this.uploadingNum = 0 | ||||
|     } | ||||
|   } | ||||
| }) | ||||
|  | ||||
| @ -1,19 +0,0 @@ | ||||
| ## 1.0.250331(2025-03-31) | ||||
| 增加RecordApp.UniNativeUtsPlugin_OnJsCall接口,App端搭配原生插件使用时,可绑定接收配套原生录音插件事件:原生插件新增PcmPlayer播放器,支持流式播放、完整播放,App端边录音边播放更流畅 | ||||
| ## 1.0.250111(2025-01-11) | ||||
| 修复vue3 Fragments(multi-root 多个根节点)的兼容性问题;修复uniapp Android自带的XXPermissions库在后台无法请求权限的问题(仅限搭配原生录音插件可用) | ||||
| ## 1.0.241020(2024-10-20) | ||||
| 适配HBuilder4.28 vue3 setup编译环境下$root.$scope无法读取的bug,HBuilder4.29已修复此编译bug,但似乎还是有不能使用的问题。如果setup内不能使用,可尝试新建个vue组件,然后使用选项式api来调用录音功能,页面的setup内使用此vue组件 | ||||
| ## 1.0.240910(2024-09-10) | ||||
| - 新增RecordApp.UniMainCallBack_Register接口,允许App renderjs层多次回调数据给逻辑层 | ||||
| - iOS App请求权限时,会预先检查NSMicrophoneUsageDescription是否声明,避免无声明时调用录音会崩溃 | ||||
| - 新增appNativePlugin_sampleRate原生插件录音选项 | ||||
| - Android App已提供后台录音保活功能,启用后App在后台或锁屏后可继续正常录音 | ||||
| ## 1.0.240625(2024-06-25) | ||||
| 调整UniWebViewCallAsync调用失败时返回更详细信息。android_audioSource默认值由1改成0,新增ios_categoryOptions原生插件录音选项 | ||||
| ## 1.0.240409(2024-04-09) | ||||
| 增加功能调用,完善demo项目 | ||||
| ## 1.0.231208(2023-12-08) | ||||
| 完善文档,增加asr语音识别示例 | ||||
| ## 1.0.231201(2023-12-04) | ||||
| 第一次发布 | ||||
| @ -1,6 +0,0 @@ | ||||
| <template> | ||||
| <view> | ||||
| 	<view style="font-weight: bold;">Recorder-UniCore Vue Component</view> | ||||
| 	<view style="font-size:14px; color:#f60">无需手动显示本UI组件,只需在script中正常引入 RecordApp + app-uni-support.js 即可实现 H5、iOS Android App、微信小程序 多端录音</view> | ||||
| </view> | ||||
| </template> | ||||
| @ -1,495 +0,0 @@ | ||||
| /* | ||||
| Recorder ../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/i18n/Template.js | ||||
| https://github.com/xiangyuecn/Recorder
 | ||||
| 
 | ||||
| Usage: Recorder.i18n.lang="Your-Language-Name" or "your-language" | ||||
| 
 | ||||
| Desc: This file is a language translation template file. After copying and renaming, translate the text into the corresponding language. 此文件为语言翻译模板文件,复制并改名后,将文本翻译成对应语言即可。 | ||||
| 
 | ||||
| 注意:请勿修改//@@打头的文本行;以下代码结构由/src/package-i18n.js自动生成,只允许在字符串中填写翻译后的文本,请勿改变代码结构;翻译的文本如果需要明确的空值,请填写"=Empty";文本中的变量用{n}表示(n代表第几个变量),所有变量必须都出现至少一次,如果不要某变量用{n!}表示
 | ||||
| 
 | ||||
| Note: Do not modify the text lines starting with //@@; The following code structure is automatically generated by /src/package-i18n.js, only the translated text is allowed to be filled in the string, please do not change the code structure; If the translated text requires an explicit empty value, please fill in "=Empty"; Variables in the text are represented by {n} (n represents the number of variables), all variables must appear at least once, if a variable is not required, it is represented by {n!}
 | ||||
| */ | ||||
| (function(factory){ | ||||
| 	var browser=typeof window=="object" && !!window.document; | ||||
| 	var win=browser?window:Object; //非浏览器环境,Recorder挂载在Object下面
 | ||||
| 	factory(win.Recorder,browser); | ||||
| }(function(Recorder,isBrowser){ | ||||
| "use strict"; | ||||
| var i18n=Recorder.i18n; | ||||
| 
 | ||||
| //@@User Code-1 Begin 手写代码放这里 Put the handwritten code here @@
 | ||||
| 
 | ||||
| //@@User Code-1 End @@
 | ||||
| 
 | ||||
| //@@Exec i18n.lang="Your-Language-Name";
 | ||||
| Recorder.CLog('Import Page[Recorder_UniCore] lang="Your-Language-Name"'); | ||||
| 
 | ||||
| //@@Exec i18n.alias["Your-Language-Name"]="your-language";
 | ||||
| 
 | ||||
| var putSet={lang:"your-language"}; | ||||
| 
 | ||||
| //@@Exec i18n.data["rtl$your-language"]=false;
 | ||||
| i18n.data["desc-page-Recorder_UniCore$your-language"]="This file is a language translation template file. After copying and renaming, translate the text into the corresponding language. 此文件为语言翻译模板文件,复制并改名后,将文本翻译成对应语言即可。"; | ||||
| //@@Exec i18n.GenerateDisplayEnglish=true;
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| //*************** Begin srcFile=../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/app-uni-support.js ***************
 | ||||
| i18n.put(putSet, | ||||
| [ //@@PutList 
 | ||||
| 
 | ||||
| //@@zh="微信小程序中需要:{1}"
 | ||||
| //@@en="WeChat miniProgram requires: {1}"
 | ||||
| //@@Put0
 | ||||
|  "RXs7:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="Recorder-UniCore目前只支持:H5、APP(Android iOS)、MP-WEIXIN,其他平台环境需要自行编写适配文件实现接入"
 | ||||
| //@@en="Recorder-UniCore currently only supports: H5, APP (Android iOS), MP-WEIXIN, other platform environments need to write their own adaptation files to achieve access"
 | ||||
| ,"4ATo:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 需要传入当前页面或组件的this对象作为参数"
 | ||||
| //@@en="RecordApp.UniWebViewActivate needs to pass in the this object of the current page or component as a parameter"
 | ||||
| ,"GwCz:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 发生不应该出现的错误(可能需要升级插件代码):"
 | ||||
| //@@en="An error occurred in RecordApp.UniWebViewActivate that should not occur (the plug-in code may need to be upgraded): "
 | ||||
| ,"ipB3:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 已切换当前页面或组件的renderjs所在的WebView"
 | ||||
| //@@en="RecordApp.UniWebViewActivate has switched the WebView where the renderjs of the current page or component is located"
 | ||||
| ,"WpKg:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 发生不应该出现的错误(可能需要升级插件代码):"
 | ||||
| //@@en="An error occurred in RecordApp.UniRenderjsRegister that should not occur (the plugin code may need to be upgraded): "
 | ||||
| ,"Uc9E:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 重复注册当前页面renderjs模块,一个组件内只允许一个renderjs模块进行注册"
 | ||||
| //@@en="RecordApp.UniRenderjsRegister repeatedly registers the renderjs module of the current page. Only one renderjs module is allowed to be registered in a component"
 | ||||
| ,"mzKj:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 已注册当前页面renderjs模块"
 | ||||
| //@@en="RecordApp.UniRenderjsRegister has registered the renderjs module of the current page"
 | ||||
| ,"7kJS:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="严重兼容性问题:无法获取页面或组件this.$root.$scope或.$page"
 | ||||
| //@@en="Serious compatibility issue: Unable to get page or component this.$root.$scope or .$page"
 | ||||
| ,"KpY6:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需要先调用RecordApp.UniWebViewActivate方法"
 | ||||
| //@@en="You need to call the RecordApp.UniWebViewActivate method first"
 | ||||
| ,"AGd7:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.RequestPermission方法"
 | ||||
| //@@en="You need to call the RecordApp.RequestPermission method first"
 | ||||
| ,"7ot0:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需重新调用RecordApp.RequestPermission方法"
 | ||||
| //@@en="The RecordApp.RequestPermission method needs to be called again"
 | ||||
| ,"VsdN:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="检测到有其他页面或组件调用了RecordApp.UniPageOnShow(WvCid={1}),但未调用过RecordApp.UniWebViewActivate(当前WvCid={2}),部分功能会继续使用之前Activate的WebView和组件,请确保这是符合你的业务逻辑,不是因为忘记了调用UniWebViewActivate"
 | ||||
| //@@en="It is detected that another page or component has called RecordApp.UniPageOnShow (WvCid={1}), but RecordApp.UniWebViewActivate (current WvCid={2}) has not been called. Some functions will continue to use the previously Activated WebView and components. Please make sure This is in line with your business logic, not because you forgot to call UniWebViewActivate"
 | ||||
| ,"SWsy:"+ //args: {1}-{2}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="{1}未正确查询到节点,将使用传入的当前页面或组件this的$el.parentNode作为组件根节点。如果template下存在多个根节点(vue3 multi-root),尽量在最外面再套一层view来避免兼容性问题"
 | ||||
| //@@en="{1} does not query the node correctly, and will use the current page or component this's $el.parentNode as the component root node. If there are multiple root nodes under the template (vue3 multi-root), try to add another layer of view on the outermost to avoid compatibility issues"
 | ||||
| ,"dX7B:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="{1}需在renderjs中调用并且传入当前模块的this"
 | ||||
| //@@en="{1} needs to be called in renderjs and pass in this of the current module"
 | ||||
| ,"dX5B:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="{1}需要传入当前页面或组件的this对象作为参数"
 | ||||
| //@@en="{1} needs to pass in the this object of the current page or component as a parameter"
 | ||||
| ,"dX6B:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前不是App逻辑层"
 | ||||
| //@@en="Currently it is not the App logic layer"
 | ||||
| ,"TfJX:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前还未调用过RecordApp.UniWebViewActivate"
 | ||||
| //@@en="RecordApp.UniWebViewActivate has not been called yet"
 | ||||
| ,"peIm:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未找到此页面renderjs所在的WebView"
 | ||||
| //@@en="The WebView where renderjs for this page is not found"
 | ||||
| ,"qDo1:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh=",不可以调用RecordApp.UniWebViewEval"
 | ||||
| //@@en=", RecordApp.UniWebViewEval cannot be called"
 | ||||
| ,"igw2:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前不是App逻辑层"
 | ||||
| //@@en="Currently it is not the App logic layer"
 | ||||
| ,"lU1W:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前还未调用过RecordApp.UniWebViewActivate"
 | ||||
| //@@en="RecordApp.UniWebViewActivate has not been called yet"
 | ||||
| ,"mSbR:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未找到此页面renderjs所在的WebView Cid"
 | ||||
| //@@en="The WebView Cid where renderjs for this page is not found"
 | ||||
| ,"6Iql:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh=",不可以调用RecordApp.UniWebViewVueCall"
 | ||||
| //@@en=", RecordApp.UniWebViewVueCall cannot be called"
 | ||||
| ,"TtoS:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="renderjs中未import导入RecordApp"
 | ||||
| //@@en="RecordApp is not imported in renderjs"
 | ||||
| ,"U1Be:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="renderjs中的mounted内需要调用RecordApp.UniRenderjsRegister"
 | ||||
| //@@en="RecordApp.UniRenderjsRegister needs to be called in mounted in renderjs"
 | ||||
| ,"Bcgi:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="没有找到组件的renderjs模块"
 | ||||
| //@@en="The renderjs module for the component was not found"
 | ||||
| ,"URyD:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="{1}连接renderjs超时"
 | ||||
| //@@en="{1} connection renderjs timeout"
 | ||||
| ,"KQhJ:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="{1}处理超时"
 | ||||
| //@@en="{1} processing timeout"
 | ||||
| ,"RDcZ:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需要在页面中提供一个renderjs,在里面import导入RecordApp、录音格式编码器、可视化插件等"
 | ||||
| //@@en="You need to provide a renderjs in the page, and import RecordApp, recording format encoder, visualization plug-in, etc."
 | ||||
| ,"TSmQ:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需在renderjs中import {1}"
 | ||||
| //@@en="Need to import {1} in renderjs"
 | ||||
| ,"AN0e:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="不应该出现的MainReceiveBind重复绑定"
 | ||||
| //@@en="MainReceiveBind duplicate binding that should not occur"
 | ||||
| ,"vEgr:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="从renderjs发回数据但UniMainCallBack回调不存在:"
 | ||||
| //@@en="Sending data back from renderjs but UniMainCallBack callback does not exist: "
 | ||||
| ,"kZx6:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="[MainReceive]从renderjs发回未知数据:"
 | ||||
| //@@en="[MainReceive] Unknown data sent back from renderjs: "
 | ||||
| ,"ZHwv:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="只允许在renderjs中调用RecordApp.UniWebViewSendBigBytesToMain"
 | ||||
| //@@en="Only allowed to call RecordApp.UniWebViewSendBigBytesToMain in renderjs"
 | ||||
| ,"MujG:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="renderjs中的mounted内需要调用RecordApp.UniRenderjsRegister才能调用RecordApp.UniWebViewSendBigBytesToMain"
 | ||||
| //@@en="RecordApp.UniRenderjsRegister needs to be called in mounted in renderjs to call RecordApp.UniWebViewSendBigBytesToMain"
 | ||||
| ,"kE91:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无效的BigBytes回传数据"
 | ||||
| //@@en="Invalid BigBytes return data"
 | ||||
| ,"CjMb:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="保存文件{1}失败:"
 | ||||
| //@@en="Failed to save file {1}: "
 | ||||
| ,"UqfI:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前环境未支持保存本地文件"
 | ||||
| //@@en="The current environment does not support saving local files"
 | ||||
| ,"kxOd:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh=" | RecordApp的uni-app支持文档和示例:{1} "
 | ||||
| //@@en=" | RecordApp’s uni-app support documentation and examples: {1}"
 | ||||
| ,"1f2V:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前录音由原生录音插件提供支持"
 | ||||
| //@@en="Current recording is powered by native recording plug-in"
 | ||||
| ,"XSYY:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前录音由uts插件提供支持"
 | ||||
| //@@en="Current recording is powered by uts plugin"
 | ||||
| ,"nnM6:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前已配置RecordApp.UniWithoutAppRenderjs,必须提供原生录音插件或uts插件才能录音,请参考RecordApp.UniNativeUtsPlugin配置"
 | ||||
| //@@en="RecordApp.UniWithoutAppRenderjs is currently configured. A native recording plug-in or uts plug-in must be provided to record. Please refer to the RecordApp.UniNativeUtsPlugin configuration"
 | ||||
| ,"fqhr:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前RecordApp运行在逻辑层中(性能会略低一些,可视化等插件不可用)"
 | ||||
| //@@en="Currently RecordApp runs in the logical layer (performance will be slightly lower, and plug-ins such as visualization are not available) "
 | ||||
| ,"xYRb:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未找到当前页面renderjs所在的WebView"
 | ||||
| //@@en="The WebView where renderjs of the current page is located is not found"
 | ||||
| ,"S3eF:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前RecordApp运行在renderjs所在的WebView中(逻辑层中只能做有限的实时处理,可视化等插件均需要在renderjs中进行调用)"
 | ||||
| //@@en="The current RecordApp runs in the WebView where renderjs is located (only limited real-time processing can be done in the logic layer, and visualization and other plug-ins need to be called in renderjs) "
 | ||||
| ,"0hyi:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh=",请检查此页面代码中是否编写了lang=renderjs的module,并且调用了RecordApp.UniRenderjsRegister;如果确实没有renderjs,比如nvue页面,请设置RecordApp.UniWithoutAppRenderjs=true并且搭配配套的原生插件在逻辑层中直接录音"
 | ||||
| //@@en=", please check whether the module with lang=renderjs is written in the code of this page and RecordApp.UniRenderjsRegister is called; if there is indeed no renderjs, such as nvue page, please set RecordApp.UniWithoutAppRenderjs=true and use the matching native plug-in to record directly in the logic layer"
 | ||||
| ,"e6Mo:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="【在App内使用{1}的授权许可】"
 | ||||
| //@@en="[License for use of {1} within the App] "
 | ||||
| ,"FabE:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已购买原生录音插件,获得授权许可"
 | ||||
| //@@en="Purchased the native recording plug-in and obtained the license"
 | ||||
| ,"w37G:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已购买uts插件,获得授权许可"
 | ||||
| //@@en="Purchased uts plug-in and obtained license"
 | ||||
| ,"e71S:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="UniAppUseLicense填写无效,如果已获取到了商用授权,请填写:{1},否则请使用空字符串"
 | ||||
| //@@en="UniAppUseLicense is invalid. If you have obtained a commercial license, please fill in: {1}, otherwise please use an empty string"
 | ||||
| ,"aPoj:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未找到Canvas:{1},请确保此DOM已挂载(可尝试用$nextTick等待DOM更新)"
 | ||||
| //@@en="Canvas not found: {1}, please make sure this DOM is mounted (try $nextTick to wait for DOM update) "
 | ||||
| ,"k7im:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="RecordApp.UniFindCanvas未适配当前环境"
 | ||||
| //@@en="RecordApp.UniFindCanvas does not adapt to the current environment"
 | ||||
| ,"yI24:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未配置RecordApp.UniNativeUtsPlugin原生录音插件"
 | ||||
| //@@en="RecordApp.UniNativeUtsPlugin native recording plug-in is not configured"
 | ||||
| ,"H753:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="renderjs中不支持设置RecordApp.UniNativeUtsPlugin"
 | ||||
| //@@en="Setting RecordApp.UniNativeUtsPlugin is not supported in renderjs"
 | ||||
| ,"l6sY:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="当前App未打包进双端原生插件[{1}],尝试加载单端[{2}]"
 | ||||
| //@@en="The current App is not packaged into the dual-end native plug-in [{1}], try to load the single-end [{2}]"
 | ||||
| ,"kSjQ:"+ //args: {1}-{2}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已加载原生录音插件[{1}]"
 | ||||
| //@@en="Native recording plugin loaded [{1}]"
 | ||||
| ,"Xh1W:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="配置了RecordApp.UniNativeUtsPlugin,但当前App未打包进原生录音插件[{1}]"
 | ||||
| //@@en="RecordApp.UniNativeUtsPlugin is configured, but the current App is not packaged with the native recording plug-in [{1}]"
 | ||||
| ,"SCW9:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="提供的RecordApp.UniNativeUtsPlugin值不是RecordApp的uts原生录音插件"
 | ||||
| //@@en="The provided RecordApp.UniNativeUtsPlugin value is not RecordApp’s uts native recording plug-in"
 | ||||
| ,"TGMm:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需在App逻辑层中调用原生插件功能"
 | ||||
| //@@en="The native plug-in function needs to be called in the App logic layer"
 | ||||
| ,"MrBx:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音,不可以调用{1}"
 | ||||
| //@@en="Recording has not started and {1} cannot be called"
 | ||||
| ,"0FGq:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.UniWebViewActivate,然后才可以调用RequestPermission"
 | ||||
| //@@en="RecordApp.UniWebViewActivate needs to be called first, and then RequestPermission can be called"
 | ||||
| ,"PkQ2:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5权限请求"
 | ||||
| //@@en="Non-H5 permission requests that should not appear"
 | ||||
| ,"Jk72:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="正在调用plus.ios@AVAudioSession请求iOS原生录音权限"
 | ||||
| //@@en="Calling plus.ios@AVAudioSession to request iOS native recording permissions"
 | ||||
| ,"Y3rC:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="项目配置中未声明iOS录音权限{1}"
 | ||||
| //@@en="iOS recording permission {1} is not declared in the project configuration"
 | ||||
| ,"9xoE:"+ //args: {1}
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已获得iOS原生录音权限"
 | ||||
| //@@en="Obtained iOS native recording permissions"
 | ||||
| ,"j15C:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="plus.ios请求录音权限,状态值: "
 | ||||
| //@@en="plus.ios requests recording permission, status value: "
 | ||||
| ,"iKhe:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="正在调用plus.android.requestPermissions请求Android原生录音权限"
 | ||||
| //@@en="Calling plus.android.requestPermissions to request Android native recording permissions"
 | ||||
| ,"7Noe:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已获得Android原生录音权限:"
 | ||||
| //@@en="Obtained Android native recording permission: "
 | ||||
| ,"Bgls:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="plus.android请求录音权限:无权限"
 | ||||
| //@@en="plus.android requests recording permission: No permission"
 | ||||
| ,"Ruxl:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="plus.android请求录音权限出错:"
 | ||||
| //@@en="plus.android error in requesting recording permission: "
 | ||||
| ,"0JQw:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="调用plus的权限请求出错:"
 | ||||
| //@@en="An error occurred in the permission request to call plus: "
 | ||||
| ,"Mvl7:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="用户拒绝了录音权限"
 | ||||
| //@@en="User denied recording permission"
 | ||||
| ,"0caE:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="正在调用原生插件请求录音权限"
 | ||||
| //@@en="Calling the native plug-in to request recording permission"
 | ||||
| ,"Lx5r:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="已获得录音权限"
 | ||||
| //@@en="Recording permission obtained"
 | ||||
| ,"Lx6r:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无录音权限"
 | ||||
| //@@en="No recording permission"
 | ||||
| ,"Lx7r:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无法调用RequestPermission:"
 | ||||
| //@@en="Unable to call RequestPermission: "
 | ||||
| ,"ksoA:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无法连接到renderjs"
 | ||||
| //@@en="Unable to connect to renderjs"
 | ||||
| ,"KnF0:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.UniWebViewActivate,然后才可以调用Start"
 | ||||
| //@@en="RecordApp.UniWebViewActivate needs to be called first, and then Start can be called"
 | ||||
| ,"XCMU:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5录音Start"
 | ||||
| //@@en="Start of non-H5 recordings that should not appear"
 | ||||
| ,"rSLO:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无法调用Start:"
 | ||||
| //@@en="Unable to call Start: "
 | ||||
| ,"Bjx9:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到renderjs回传的onRecEncodeChunk"
 | ||||
| //@@en="Recording did not start, but onRecEncodeChunk returned by renderjs was received"
 | ||||
| ,"MTdp:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到Uni Native PCM数据"
 | ||||
| //@@en="Recording did not start, but Uni Native PCM data was received"
 | ||||
| ,"BjGP:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到UniNativeUtsPlugin PCM数据"
 | ||||
| //@@en="Recording did not start, but UniNativeUtsPlugin PCM data was received"
 | ||||
| ,"byzO:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音"
 | ||||
| //@@en="Recording not started"
 | ||||
| ,"YP4V:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5录音Stop"
 | ||||
| //@@en="Stop non-H5 recordings that should not appear"
 | ||||
| ,"TPhg:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="未开始录音"
 | ||||
| //@@en="Recording not started"
 | ||||
| ,"pP4O:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="无法调用Stop:"
 | ||||
| //@@en="Unable to call Stop: "
 | ||||
| ,"H6cq:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| //@@zh="不应该出现的renderjs发回的文件数据丢失"
 | ||||
| //@@en="The file data sent back by renderjs should not be lost"
 | ||||
| ,"gomD:"+ //no args
 | ||||
|        "" /** TODO: translate to your-language **/ | ||||
| 
 | ||||
| ]); | ||||
| //*************** End srcFile=../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/app-uni-support.js ***************
 | ||||
| 
 | ||||
| //@@User Code-2 Begin 手写代码放这里 Put the handwritten code here @@
 | ||||
| 
 | ||||
| //@@User Code-2 End @@
 | ||||
| 
 | ||||
| })); | ||||
| @ -1,406 +0,0 @@ | ||||
| /* | ||||
| Recorder ../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/i18n/en-US.js | ||||
| https://github.com/xiangyuecn/Recorder
 | ||||
| 
 | ||||
| Usage: Recorder.i18n.lang="en-US" or "en" | ||||
| 
 | ||||
| Desc: English, 英语。This translation mainly comes from: google translation + Baidu translation, translated from Chinese to English. 此翻译主要来自:google翻译+百度翻译,由中文翻译成英文。 | ||||
| 
 | ||||
| 注意:请勿修改//@@打头的文本行;以下代码结构由/src/package-i18n.js自动生成,只允许在字符串中填写翻译后的文本,请勿改变代码结构;翻译的文本如果需要明确的空值,请填写"=Empty";文本中的变量用{n}表示(n代表第几个变量),所有变量必须都出现至少一次,如果不要某变量用{n!}表示
 | ||||
| 
 | ||||
| Note: Do not modify the text lines starting with //@@; The following code structure is automatically generated by /src/package-i18n.js, only the translated text is allowed to be filled in the string, please do not change the code structure; If the translated text requires an explicit empty value, please fill in "=Empty"; Variables in the text are represented by {n} (n represents the number of variables), all variables must appear at least once, if a variable is not required, it is represented by {n!}
 | ||||
| */ | ||||
| (function(factory){ | ||||
| 	var browser=typeof window=="object" && !!window.document; | ||||
| 	var win=browser?window:Object; //非浏览器环境,Recorder挂载在Object下面
 | ||||
| 	factory(win.Recorder,browser); | ||||
| }(function(Recorder,isBrowser){ | ||||
| "use strict"; | ||||
| var i18n=Recorder.i18n; | ||||
| 
 | ||||
| //@@User Code-1 Begin 手写代码放这里 Put the handwritten code here @@
 | ||||
| 
 | ||||
| //@@User Code-1 End @@
 | ||||
| 
 | ||||
| //@@Exec i18n.lang="en-US";
 | ||||
| Recorder.CLog('Import Page[Recorder_UniCore] lang="en-US"'); | ||||
| 
 | ||||
| //@@Exec i18n.alias["en-US"]="en";
 | ||||
| 
 | ||||
| var putSet={lang:"en"}; | ||||
| 
 | ||||
| //@@Exec i18n.data["rtl$en"]=false;
 | ||||
| i18n.data["desc-page-Recorder_UniCore$en"]="English, 英语。This translation mainly comes from: google translation + Baidu translation, translated from Chinese to English. 此翻译主要来自:google翻译+百度翻译,由中文翻译成英文。"; | ||||
| //@@Exec i18n.GenerateDisplayEnglish=false;
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| //*************** Begin srcFile=../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/app-uni-support.js ***************
 | ||||
| i18n.put(putSet, | ||||
| [ //@@PutList 
 | ||||
| 
 | ||||
| //@@zh="微信小程序中需要:{1}"
 | ||||
| //@@Put0
 | ||||
|  "RXs7:"+ //args: {1}
 | ||||
|        "WeChat miniProgram requires: {1}" | ||||
| 
 | ||||
| //@@zh="Recorder-UniCore目前只支持:H5、APP(Android iOS)、MP-WEIXIN,其他平台环境需要自行编写适配文件实现接入"
 | ||||
| ,"4ATo:"+ //no args
 | ||||
|        "Recorder-UniCore currently only supports: H5, APP (Android iOS), MP-WEIXIN, other platform environments need to write their own adaptation files to achieve access" | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 需要传入当前页面或组件的this对象作为参数"
 | ||||
| ,"GwCz:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate needs to pass in the this object of the current page or component as a parameter" | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 发生不应该出现的错误(可能需要升级插件代码):"
 | ||||
| ,"ipB3:"+ //no args
 | ||||
|        "An error occurred in RecordApp.UniWebViewActivate that should not occur (the plug-in code may need to be upgraded): " | ||||
| 
 | ||||
| //@@zh="RecordApp.UniWebViewActivate 已切换当前页面或组件的renderjs所在的WebView"
 | ||||
| ,"WpKg:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate has switched the WebView where the renderjs of the current page or component is located" | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 发生不应该出现的错误(可能需要升级插件代码):"
 | ||||
| ,"Uc9E:"+ //no args
 | ||||
|        "An error occurred in RecordApp.UniRenderjsRegister that should not occur (the plugin code may need to be upgraded): " | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 重复注册当前页面renderjs模块,一个组件内只允许一个renderjs模块进行注册"
 | ||||
| ,"mzKj:"+ //no args
 | ||||
|        "RecordApp.UniRenderjsRegister repeatedly registers the renderjs module of the current page. Only one renderjs module is allowed to be registered in a component" | ||||
| 
 | ||||
| //@@zh="RecordApp.UniRenderjsRegister 已注册当前页面renderjs模块"
 | ||||
| ,"7kJS:"+ //no args
 | ||||
|        "RecordApp.UniRenderjsRegister has registered the renderjs module of the current page" | ||||
| 
 | ||||
| //@@zh="严重兼容性问题:无法获取页面或组件this.$root.$scope或.$page"
 | ||||
| ,"KpY6:"+ //no args
 | ||||
|        "Serious compatibility issue: Unable to get page or component this.$root.$scope or .$page" | ||||
| 
 | ||||
| //@@zh="需要先调用RecordApp.UniWebViewActivate方法"
 | ||||
| ,"AGd7:"+ //no args
 | ||||
|        "You need to call the RecordApp.UniWebViewActivate method first" | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.RequestPermission方法"
 | ||||
| ,"7ot0:"+ //no args
 | ||||
|        "You need to call the RecordApp.RequestPermission method first" | ||||
| 
 | ||||
| //@@zh="需重新调用RecordApp.RequestPermission方法"
 | ||||
| ,"VsdN:"+ //no args
 | ||||
|        "The RecordApp.RequestPermission method needs to be called again" | ||||
| 
 | ||||
| //@@zh="检测到有其他页面或组件调用了RecordApp.UniPageOnShow(WvCid={1}),但未调用过RecordApp.UniWebViewActivate(当前WvCid={2}),部分功能会继续使用之前Activate的WebView和组件,请确保这是符合你的业务逻辑,不是因为忘记了调用UniWebViewActivate"
 | ||||
| ,"SWsy:"+ //args: {1}-{2}
 | ||||
|        "It is detected that another page or component has called RecordApp.UniPageOnShow (WvCid={1}), but RecordApp.UniWebViewActivate (current WvCid={2}) has not been called. Some functions will continue to use the previously Activated WebView and components. Please make sure This is in line with your business logic, not because you forgot to call UniWebViewActivate" | ||||
| 
 | ||||
| //@@zh="{1}未正确查询到节点,将使用传入的当前页面或组件this的$el.parentNode作为组件根节点。如果template下存在多个根节点(vue3 multi-root),尽量在最外面再套一层view来避免兼容性问题"
 | ||||
| ,"dX7B:"+ //args: {1}
 | ||||
|        "{1} does not query the node correctly, and will use the current page or component this's $el.parentNode as the component root node. If there are multiple root nodes under the template (vue3 multi-root), try to add another layer of view on the outermost to avoid compatibility issues" | ||||
| 
 | ||||
| //@@zh="{1}需在renderjs中调用并且传入当前模块的this"
 | ||||
| ,"dX5B:"+ //args: {1}
 | ||||
|        "{1} needs to be called in renderjs and pass in this of the current module" | ||||
| 
 | ||||
| //@@zh="{1}需要传入当前页面或组件的this对象作为参数"
 | ||||
| ,"dX6B:"+ //args: {1}
 | ||||
|        "{1} needs to pass in the this object of the current page or component as a parameter" | ||||
| 
 | ||||
| //@@zh="当前不是App逻辑层"
 | ||||
| ,"TfJX:"+ //no args
 | ||||
|        "Currently it is not the App logic layer" | ||||
| 
 | ||||
| //@@zh="当前还未调用过RecordApp.UniWebViewActivate"
 | ||||
| ,"peIm:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate has not been called yet" | ||||
| 
 | ||||
| //@@zh="未找到此页面renderjs所在的WebView"
 | ||||
| ,"qDo1:"+ //no args
 | ||||
|        "The WebView where renderjs for this page is not found" | ||||
| 
 | ||||
| //@@zh=",不可以调用RecordApp.UniWebViewEval"
 | ||||
| ,"igw2:"+ //no args
 | ||||
|        ", RecordApp.UniWebViewEval cannot be called" | ||||
| 
 | ||||
| //@@zh="当前不是App逻辑层"
 | ||||
| ,"lU1W:"+ //no args
 | ||||
|        "Currently it is not the App logic layer" | ||||
| 
 | ||||
| //@@zh="当前还未调用过RecordApp.UniWebViewActivate"
 | ||||
| ,"mSbR:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate has not been called yet" | ||||
| 
 | ||||
| //@@zh="未找到此页面renderjs所在的WebView Cid"
 | ||||
| ,"6Iql:"+ //no args
 | ||||
|        "The WebView Cid where renderjs for this page is not found" | ||||
| 
 | ||||
| //@@zh=",不可以调用RecordApp.UniWebViewVueCall"
 | ||||
| ,"TtoS:"+ //no args
 | ||||
|        ", RecordApp.UniWebViewVueCall cannot be called" | ||||
| 
 | ||||
| //@@zh="renderjs中未import导入RecordApp"
 | ||||
| ,"U1Be:"+ //no args
 | ||||
|        "RecordApp is not imported in renderjs" | ||||
| 
 | ||||
| //@@zh="renderjs中的mounted内需要调用RecordApp.UniRenderjsRegister"
 | ||||
| ,"Bcgi:"+ //no args
 | ||||
|        "RecordApp.UniRenderjsRegister needs to be called in mounted in renderjs" | ||||
| 
 | ||||
| //@@zh="没有找到组件的renderjs模块"
 | ||||
| ,"URyD:"+ //no args
 | ||||
|        "The renderjs module for the component was not found" | ||||
| 
 | ||||
| //@@zh="{1}连接renderjs超时"
 | ||||
| ,"KQhJ:"+ //args: {1}
 | ||||
|        "{1} connection renderjs timeout" | ||||
| 
 | ||||
| //@@zh="{1}处理超时"
 | ||||
| ,"RDcZ:"+ //args: {1}
 | ||||
|        "{1} processing timeout" | ||||
| 
 | ||||
| //@@zh="需要在页面中提供一个renderjs,在里面import导入RecordApp、录音格式编码器、可视化插件等"
 | ||||
| ,"TSmQ:"+ //no args
 | ||||
|        "You need to provide a renderjs in the page, and import RecordApp, recording format encoder, visualization plug-in, etc." | ||||
| 
 | ||||
| //@@zh="需在renderjs中import {1}"
 | ||||
| ,"AN0e:"+ //args: {1}
 | ||||
|        "Need to import {1} in renderjs" | ||||
| 
 | ||||
| //@@zh="不应该出现的MainReceiveBind重复绑定"
 | ||||
| ,"vEgr:"+ //no args
 | ||||
|        "MainReceiveBind duplicate binding that should not occur" | ||||
| 
 | ||||
| //@@zh="从renderjs发回数据但UniMainCallBack回调不存在:"
 | ||||
| ,"kZx6:"+ //no args
 | ||||
|        "Sending data back from renderjs but UniMainCallBack callback does not exist: " | ||||
| 
 | ||||
| //@@zh="[MainReceive]从renderjs发回未知数据:"
 | ||||
| ,"ZHwv:"+ //no args
 | ||||
|        "[MainReceive] Unknown data sent back from renderjs: " | ||||
| 
 | ||||
| //@@zh="只允许在renderjs中调用RecordApp.UniWebViewSendBigBytesToMain"
 | ||||
| ,"MujG:"+ //no args
 | ||||
|        "Only allowed to call RecordApp.UniWebViewSendBigBytesToMain in renderjs" | ||||
| 
 | ||||
| //@@zh="renderjs中的mounted内需要调用RecordApp.UniRenderjsRegister才能调用RecordApp.UniWebViewSendBigBytesToMain"
 | ||||
| ,"kE91:"+ //no args
 | ||||
|        "RecordApp.UniRenderjsRegister needs to be called in mounted in renderjs to call RecordApp.UniWebViewSendBigBytesToMain" | ||||
| 
 | ||||
| //@@zh="无效的BigBytes回传数据"
 | ||||
| ,"CjMb:"+ //no args
 | ||||
|        "Invalid BigBytes return data" | ||||
| 
 | ||||
| //@@zh="保存文件{1}失败:"
 | ||||
| ,"UqfI:"+ //args: {1}
 | ||||
|        "Failed to save file {1}: " | ||||
| 
 | ||||
| //@@zh="当前环境未支持保存本地文件"
 | ||||
| ,"kxOd:"+ //no args
 | ||||
|        "The current environment does not support saving local files" | ||||
| 
 | ||||
| //@@zh=" | RecordApp的uni-app支持文档和示例:{1} "
 | ||||
| ,"1f2V:"+ //args: {1}
 | ||||
|        " | RecordApp’s uni-app support documentation and examples: {1}" | ||||
| 
 | ||||
| //@@zh="当前录音由原生录音插件提供支持"
 | ||||
| ,"XSYY:"+ //no args
 | ||||
|        "Current recording is powered by native recording plug-in" | ||||
| 
 | ||||
| //@@zh="当前录音由uts插件提供支持"
 | ||||
| ,"nnM6:"+ //no args
 | ||||
|        "Current recording is powered by uts plugin" | ||||
| 
 | ||||
| //@@zh="当前已配置RecordApp.UniWithoutAppRenderjs,必须提供原生录音插件或uts插件才能录音,请参考RecordApp.UniNativeUtsPlugin配置"
 | ||||
| ,"fqhr:"+ //no args
 | ||||
|        "RecordApp.UniWithoutAppRenderjs is currently configured. A native recording plug-in or uts plug-in must be provided to record. Please refer to the RecordApp.UniNativeUtsPlugin configuration" | ||||
| 
 | ||||
| //@@zh="当前RecordApp运行在逻辑层中(性能会略低一些,可视化等插件不可用)"
 | ||||
| ,"xYRb:"+ //no args
 | ||||
|        "Currently RecordApp runs in the logical layer (performance will be slightly lower, and plug-ins such as visualization are not available) " | ||||
| 
 | ||||
| //@@zh="未找到当前页面renderjs所在的WebView"
 | ||||
| ,"S3eF:"+ //no args
 | ||||
|        "The WebView where renderjs of the current page is located is not found" | ||||
| 
 | ||||
| //@@zh="当前RecordApp运行在renderjs所在的WebView中(逻辑层中只能做有限的实时处理,可视化等插件均需要在renderjs中进行调用)"
 | ||||
| ,"0hyi:"+ //no args
 | ||||
|        "The current RecordApp runs in the WebView where renderjs is located (only limited real-time processing can be done in the logic layer, and visualization and other plug-ins need to be called in renderjs) " | ||||
| 
 | ||||
| //@@zh=",请检查此页面代码中是否编写了lang=renderjs的module,并且调用了RecordApp.UniRenderjsRegister;如果确实没有renderjs,比如nvue页面,请设置RecordApp.UniWithoutAppRenderjs=true并且搭配配套的原生插件在逻辑层中直接录音"
 | ||||
| ,"e6Mo:"+ //no args
 | ||||
|        ", please check whether the module with lang=renderjs is written in the code of this page and RecordApp.UniRenderjsRegister is called; if there is indeed no renderjs, such as nvue page, please set RecordApp.UniWithoutAppRenderjs=true and use the matching native plug-in to record directly in the logic layer" | ||||
| 
 | ||||
| //@@zh="【在App内使用{1}的授权许可】"
 | ||||
| ,"FabE:"+ //args: {1}
 | ||||
|        "[License for use of {1} within the App] " | ||||
| 
 | ||||
| //@@zh="已购买原生录音插件,获得授权许可"
 | ||||
| ,"w37G:"+ //no args
 | ||||
|        "Purchased the native recording plug-in and obtained the license" | ||||
| 
 | ||||
| //@@zh="已购买uts插件,获得授权许可"
 | ||||
| ,"e71S:"+ //no args
 | ||||
|        "Purchased uts plug-in and obtained license" | ||||
| 
 | ||||
| //@@zh="UniAppUseLicense填写无效,如果已获取到了商用授权,请填写:{1},否则请使用空字符串"
 | ||||
| ,"aPoj:"+ //args: {1}
 | ||||
|        "UniAppUseLicense is invalid. If you have obtained a commercial license, please fill in: {1}, otherwise please use an empty string" | ||||
| 
 | ||||
| //@@zh="未找到Canvas:{1},请确保此DOM已挂载(可尝试用$nextTick等待DOM更新)"
 | ||||
| ,"k7im:"+ //args: {1}
 | ||||
|        "Canvas not found: {1}, please make sure this DOM is mounted (try $nextTick to wait for DOM update) " | ||||
| 
 | ||||
| //@@zh="RecordApp.UniFindCanvas未适配当前环境"
 | ||||
| ,"yI24:"+ //no args
 | ||||
|        "RecordApp.UniFindCanvas does not adapt to the current environment" | ||||
| 
 | ||||
| //@@zh="未配置RecordApp.UniNativeUtsPlugin原生录音插件"
 | ||||
| ,"H753:"+ //no args
 | ||||
|        "RecordApp.UniNativeUtsPlugin native recording plug-in is not configured" | ||||
| 
 | ||||
| //@@zh="renderjs中不支持设置RecordApp.UniNativeUtsPlugin"
 | ||||
| ,"l6sY:"+ //no args
 | ||||
|        "Setting RecordApp.UniNativeUtsPlugin is not supported in renderjs" | ||||
| 
 | ||||
| //@@zh="当前App未打包进双端原生插件[{1}],尝试加载单端[{2}]"
 | ||||
| ,"kSjQ:"+ //args: {1}-{2}
 | ||||
|        "The current App is not packaged into the dual-end native plug-in [{1}], try to load the single-end [{2}]" | ||||
| 
 | ||||
| //@@zh="已加载原生录音插件[{1}]"
 | ||||
| ,"Xh1W:"+ //args: {1}
 | ||||
|        "Native recording plugin loaded [{1}]" | ||||
| 
 | ||||
| //@@zh="配置了RecordApp.UniNativeUtsPlugin,但当前App未打包进原生录音插件[{1}]"
 | ||||
| ,"SCW9:"+ //args: {1}
 | ||||
|        "RecordApp.UniNativeUtsPlugin is configured, but the current App is not packaged with the native recording plug-in [{1}]" | ||||
| 
 | ||||
| //@@zh="提供的RecordApp.UniNativeUtsPlugin值不是RecordApp的uts原生录音插件"
 | ||||
| ,"TGMm:"+ //no args
 | ||||
|        "The provided RecordApp.UniNativeUtsPlugin value is not RecordApp’s uts native recording plug-in" | ||||
| 
 | ||||
| //@@zh="需在App逻辑层中调用原生插件功能"
 | ||||
| ,"MrBx:"+ //no args
 | ||||
|        "The native plug-in function needs to be called in the App logic layer" | ||||
| 
 | ||||
| //@@zh="未开始录音,不可以调用{1}"
 | ||||
| ,"0FGq:"+ //args: {1}
 | ||||
|        "Recording has not started and {1} cannot be called" | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.UniWebViewActivate,然后才可以调用RequestPermission"
 | ||||
| ,"PkQ2:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate needs to be called first, and then RequestPermission can be called" | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5权限请求"
 | ||||
| ,"Jk72:"+ //no args
 | ||||
|        "Non-H5 permission requests that should not appear" | ||||
| 
 | ||||
| //@@zh="正在调用plus.ios@AVAudioSession请求iOS原生录音权限"
 | ||||
| ,"Y3rC:"+ //no args
 | ||||
|        "Calling plus.ios@AVAudioSession to request iOS native recording permissions" | ||||
| 
 | ||||
| //@@zh="项目配置中未声明iOS录音权限{1}"
 | ||||
| ,"9xoE:"+ //args: {1}
 | ||||
|        "iOS recording permission {1} is not declared in the project configuration" | ||||
| 
 | ||||
| //@@zh="已获得iOS原生录音权限"
 | ||||
| ,"j15C:"+ //no args
 | ||||
|        "Obtained iOS native recording permissions" | ||||
| 
 | ||||
| //@@zh="plus.ios请求录音权限,状态值: "
 | ||||
| ,"iKhe:"+ //no args
 | ||||
|        "plus.ios requests recording permission, status value: " | ||||
| 
 | ||||
| //@@zh="正在调用plus.android.requestPermissions请求Android原生录音权限"
 | ||||
| ,"7Noe:"+ //no args
 | ||||
|        "Calling plus.android.requestPermissions to request Android native recording permissions" | ||||
| 
 | ||||
| //@@zh="已获得Android原生录音权限:"
 | ||||
| ,"Bgls:"+ //no args
 | ||||
|        "Obtained Android native recording permission: " | ||||
| 
 | ||||
| //@@zh="plus.android请求录音权限:无权限"
 | ||||
| ,"Ruxl:"+ //no args
 | ||||
|        "plus.android requests recording permission: No permission" | ||||
| 
 | ||||
| //@@zh="plus.android请求录音权限出错:"
 | ||||
| ,"0JQw:"+ //no args
 | ||||
|        "plus.android error in requesting recording permission: " | ||||
| 
 | ||||
| //@@zh="调用plus的权限请求出错:"
 | ||||
| ,"Mvl7:"+ //no args
 | ||||
|        "An error occurred in the permission request to call plus: " | ||||
| 
 | ||||
| //@@zh="用户拒绝了录音权限"
 | ||||
| ,"0caE:"+ //no args
 | ||||
|        "User denied recording permission" | ||||
| 
 | ||||
| //@@zh="正在调用原生插件请求录音权限"
 | ||||
| ,"Lx5r:"+ //no args
 | ||||
|        "Calling the native plug-in to request recording permission" | ||||
| 
 | ||||
| //@@zh="已获得录音权限"
 | ||||
| ,"Lx6r:"+ //no args
 | ||||
|        "Recording permission obtained" | ||||
| 
 | ||||
| //@@zh="无录音权限"
 | ||||
| ,"Lx7r:"+ //no args
 | ||||
|        "No recording permission" | ||||
| 
 | ||||
| //@@zh="无法调用RequestPermission:"
 | ||||
| ,"ksoA:"+ //no args
 | ||||
|        "Unable to call RequestPermission: " | ||||
| 
 | ||||
| //@@zh="无法连接到renderjs"
 | ||||
| ,"KnF0:"+ //no args
 | ||||
|        "Unable to connect to renderjs" | ||||
| 
 | ||||
| //@@zh="需先调用RecordApp.UniWebViewActivate,然后才可以调用Start"
 | ||||
| ,"XCMU:"+ //no args
 | ||||
|        "RecordApp.UniWebViewActivate needs to be called first, and then Start can be called" | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5录音Start"
 | ||||
| ,"rSLO:"+ //no args
 | ||||
|        "Start of non-H5 recordings that should not appear" | ||||
| 
 | ||||
| //@@zh="无法调用Start:"
 | ||||
| ,"Bjx9:"+ //no args
 | ||||
|        "Unable to call Start: " | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到renderjs回传的onRecEncodeChunk"
 | ||||
| ,"MTdp:"+ //no args
 | ||||
|        "Recording did not start, but onRecEncodeChunk returned by renderjs was received" | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到Uni Native PCM数据"
 | ||||
| ,"BjGP:"+ //no args
 | ||||
|        "Recording did not start, but Uni Native PCM data was received" | ||||
| 
 | ||||
| //@@zh="未开始录音,但收到UniNativeUtsPlugin PCM数据"
 | ||||
| ,"byzO:"+ //no args
 | ||||
|        "Recording did not start, but UniNativeUtsPlugin PCM data was received" | ||||
| 
 | ||||
| //@@zh="未开始录音"
 | ||||
| ,"YP4V:"+ //no args
 | ||||
|        "Recording not started" | ||||
| 
 | ||||
| //@@zh="不应当出现的非H5录音Stop"
 | ||||
| ,"TPhg:"+ //no args
 | ||||
|        "Stop non-H5 recordings that should not appear" | ||||
| 
 | ||||
| //@@zh="未开始录音"
 | ||||
| ,"pP4O:"+ //no args
 | ||||
|        "Recording not started" | ||||
| 
 | ||||
| //@@zh="无法调用Stop:"
 | ||||
| ,"H6cq:"+ //no args
 | ||||
|        "Unable to call Stop: " | ||||
| 
 | ||||
| //@@zh="不应该出现的renderjs发回的文件数据丢失"
 | ||||
| ,"gomD:"+ //no args
 | ||||
|        "The file data sent back by renderjs should not be lost" | ||||
| 
 | ||||
| ]); | ||||
| //*************** End srcFile=../app-support-sample/demo_UniApp/uni_modules/Recorder-UniCore/app-uni-support.js ***************
 | ||||
| 
 | ||||
| //@@User Code-2 Begin 手写代码放这里 Put the handwritten code here @@
 | ||||
| 
 | ||||
| //@@User Code-2 End @@
 | ||||
| 
 | ||||
| })); | ||||
| @ -1,19 +0,0 @@ | ||||
| 《许可及服务协议》 | ||||
| 
 | ||||
| **您(以下称“用户”)下载、使用我(以下称“作者”)提供的Recorder-UniCore组件(含原生录音插件、uts插件,以下统称“本组件”),应当阅读并遵守本许可协议。请用户务必审慎阅读、充分理解各条款内容,特别是免除或者限制责任的条款,并选择接受或不接受。除非用户已阅读并接受本协议所有条款,否则用户无权下载、使用本组件及相关服务,用户的下载、使用等行为即视为用户已阅读并同意本许可协议的约束。** | ||||
| 
 | ||||
| 1. 用户应当直接从作者许可的途径,如作者的GitHub、Gitee仓库、已上架的DCloud插件市场、QQ群等途径中获取本组件;其他途径获取到的组件代码是未经过作者授权的,存在安全隐患,可能会导致你的程序、资产受到侵害,作者对因此给用户造成的损失不予负责。 | ||||
| 
 | ||||
| 2. 作者将积极并采取措施保护用户的信息和隐私;组件本身不会搜集存储任何用户信息。 | ||||
| 
 | ||||
| 3. 除法律法规有明确规定外,作者将尽最大努力确保本组件及其所涉及的技术及信息安全、有效、准确、可靠,但受限于现有技术,用户理解作者不能对此进行担保。 | ||||
| 
 | ||||
| 4. 用户理解,对于不可抗力及第三方原因导致的您的直接或间接损失,作者无法承担责任。 | ||||
| 
 | ||||
| 5. 用户因使用本组件进行生成、处理数据,由此引起或与有关的包括但不限于利润损失、资料损失、业务中断的损害赔偿或其它商业损害赔偿或损失,需由用户自行承担。 | ||||
| 
 | ||||
| 6. 如若发生赔偿、退款等行为,赔偿、退款等累计金额不得超过用户实际支付给作者的总金额。 | ||||
| 
 | ||||
| 7. 已授予的授权许可,包括免费授权,和已购买的原生录音插件、uts插件,均仅限在授权指定的uni-app的应用标识(AppID)对应的项目上使用,不可在其他项目上使用;用户不得对本组件及其中的相关信息擅自出租、出借、销售、逆向工程、破解,不得在未取得作者授权的情况下借助本组件发展与本组件有关联的衍生软件产品、服务、插件、外挂等。 | ||||
| 
 | ||||
| 8. 用户不得使用本组件从事违反法律法规政策、破坏公序良俗、损害公共利益的行为。 | ||||
| @ -1,88 +0,0 @@ | ||||
| { | ||||
|   "id": "Recorder-UniCore", | ||||
|   "displayName": "跨平台Recorder录音插件:支持多种格式、音频可视化、实时上传、语音识别", | ||||
|   "version": "1.0.250331", | ||||
|   "description": "支持H5、Android iOS App、微信小程序;mp3 wav pcm g711a g711u ogg amr 录音格式;实时帧回调处理 音频转码 波形动画显示 ASR语音转文字 无录制时长限制", | ||||
|   "keywords": [ | ||||
|     "Recorder-UniCore", | ||||
|     "recorder-core", | ||||
|     "RecordApp", | ||||
|     "record", | ||||
|     "recording" | ||||
| ], | ||||
|   "repository": "https://github.com/xiangyuecn/Recorder", | ||||
|   "engines": { | ||||
|     "HBuilderX": "^3.6.11" | ||||
|   }, | ||||
| "dcloudext": { | ||||
|     "type": "component-vue", | ||||
|     "sale": { | ||||
|       "regular": { | ||||
|         "price": "0.00" | ||||
|       }, | ||||
|       "sourcecode": { | ||||
|         "price": "0.00" | ||||
|       } | ||||
|     }, | ||||
|     "contact": { | ||||
|       "qq": "753610399" | ||||
|     }, | ||||
|     "declaration": { | ||||
|       "ads": "无", | ||||
|       "data": "插件不采集任何数据", | ||||
|       "permissions": "录音权限" | ||||
|     }, | ||||
|     "npmurl": "" | ||||
|   }, | ||||
|   "uni_modules": { | ||||
|     "dependencies": [], | ||||
|     "encrypt": [], | ||||
|     "platforms": { | ||||
|       "cloud": { | ||||
|         "tcb": "y", | ||||
|         "aliyun": "y", | ||||
|         "alipay": "n" | ||||
|       }, | ||||
|       "client": { | ||||
|         "Vue": { | ||||
|           "vue2": "y", | ||||
|           "vue3": "y" | ||||
|         }, | ||||
|         "App": { | ||||
|             "app-vue": "y", | ||||
|             "app-nvue": "y", | ||||
|             "app-uvue": "n", | ||||
|             "app-harmony": "u" | ||||
|         }, | ||||
|         "H5-mobile": { | ||||
|           "Safari": "y", | ||||
|           "Android Browser": "y", | ||||
|           "微信浏览器(Android)": "y", | ||||
|           "QQ浏览器(Android)": "y" | ||||
|         }, | ||||
|         "H5-pc": { | ||||
|           "Chrome": "y", | ||||
|           "IE": "n", | ||||
|           "Edge": "y", | ||||
|           "Firefox": "y", | ||||
|           "Safari": "y" | ||||
|         }, | ||||
|         "小程序": { | ||||
|           "微信": "y", | ||||
|           "阿里": "n", | ||||
|           "百度": "n", | ||||
|           "字节跳动": "n", | ||||
|           "QQ": "n", | ||||
|           "钉钉": "n", | ||||
|           "快手": "n", | ||||
|           "飞书": "n", | ||||
|           "京东": "n" | ||||
|         }, | ||||
|         "快应用": { | ||||
|           "华为": "u", | ||||
|           "联盟": "u" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -1,425 +0,0 @@ | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| # Recorder-UniCore组件:uni-app内使用RecordApp录音 | ||||
| 
 | ||||
| 本组件使用`Recorder`开源库来进行录音和音频数据处理,使用`RecordApp`和本组件内的`app-uni-support.js`来适配到不同平台环境下进行录音。 | ||||
| 
 | ||||
| - 支持vue2、vue3、nvue | ||||
| - 支持编译成:H5、Android App、iOS App、微信小程序 | ||||
| - 支持已有的大部分录音格式:mp3、wav、pcm、amr、ogg、g711a、g711u等 | ||||
| - 支持实时处理,包括变速变调、实时上传、ASR语音转文字 | ||||
| - 支持可视化波形显示;可配置回声消除、降噪;**注意:不支持通话时录音** | ||||
| - 支持PCM音频流式播放、完整播放,App端用原生插件边录音边播放更流畅 | ||||
| - 支持离线使用,本组件和配套原生插件均不依赖网络 | ||||
| - App端有配套的[原生录音插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin)可供搭配使用,兼容性和体验更好 | ||||
| 
 | ||||
| **详细文档(含Demo项目):** [https://github.com/xiangyuecn/Recorder/tree/master/app-support-sample/demo_UniApp](https://github.com/xiangyuecn/Recorder/tree/master/app-support-sample/demo_UniApp) | ||||
| 
 | ||||
| **Recorder开源库地址:** [https://github.com/xiangyuecn/Recorder](https://github.com/xiangyuecn/Recorder) | ||||
| 
 | ||||
| 如果github打不开,可以[点此访问Gitee仓库地址](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample/demo_UniApp) 。 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| ## 测试方法 | ||||
| **示例项目如果在HBuilder中编译失败,请删掉node_modules目录重新手动执行npm install(偶尔出现HBuilder自动创建项目依赖包不完整,导致无法编译)** | ||||
| 
 | ||||
| 1. 在本插件市场页面右侧下载或导入示例项目(或打开上面详细文档链接中的Demo源码) | ||||
| 2. 在测试项目根目录执行 `npm install --registry=https://registry.npmmirror.com/` ,完成`recorder-core`依赖的安装 | ||||
| 3. 在HBuilder中打开本测试项目文件夹 | ||||
| 4. 在HBuilder中运行到浏览器、手机、微信小程序,即可在不同环境下测试 | ||||
| 5. 测试中提供了:基础录音、播放、上传、WebSocket实时语音通话对讲、ASR语音识别等功能 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| # 集成到自己项目中 | ||||
| 
 | ||||
| 你可以直接参考上面的测试示例项目源码,里面的`main_recTest.vue`更容易入门;示例项目中已经实现了很多功能,简单使用可直接照抄Demo代码到你的项目中。 | ||||
| 
 | ||||
| 
 | ||||
| ## 一、引入js文件 | ||||
| 1. 在你的项目根目录安装`recorder-core`:`npm install recorder-core --registry=https://registry.npmmirror.com/` | ||||
| 2. 导入Recorder-UniCore组件:插件市场下载本组件,然后添加到你的项目中 `/uni_modules/Recorder-UniCore` | ||||
| 3. 项目配置好录音权限,参考下面的录音权限配置章节,**特别注意App后台录音配置、小程序权限声明** | ||||
| 4. 在需要录音的vue文件script内编写以下代码,按需引入需要的js | ||||
| 
 | ||||
| ``` html | ||||
| <template> | ||||
|     <view> | ||||
|         ... 建议template下只有一个根节点(最外面套一层view),如果不小心踩到了vue3的Fragments(multi-root 多个根节点)特性(vue2编译会报错,vue3不会),可能会出现奇奇怪怪的兼容性问题  | ||||
|     </view> | ||||
| </template> | ||||
| 
 | ||||
| <script> /**这里是逻辑层**/ | ||||
| //必须引入的Recorder核心(文件路径是 /src/recorder-core.js 下同),使用import、require都行 | ||||
| import Recorder from 'recorder-core' //注意如果未引用Recorder变量,可能编译时会被优化删除(如vue3 tree-shaking),请改成 import 'recorder-core',或随便调用一下 Recorder.a=1 保证强引用 | ||||
| 
 | ||||
| //必须引入的RecordApp核心文件(文件路径是 /src/app-support/app.js) | ||||
| import RecordApp from 'recorder-core/src/app-support/app' | ||||
| 
 | ||||
| //所有平台必须引入的uni-app支持文件(如果编译出现路径错误,请把@换成 ../../ 这种) | ||||
| import '@/uni_modules/Recorder-UniCore/app-uni-support.js' | ||||
| 
 | ||||
| /** 需要编译成微信小程序时,引入微信小程序支持文件 **/ | ||||
| // #ifdef MP-WEIXIN | ||||
|     import 'recorder-core/src/app-support/app-miniProgram-wx-support.js' | ||||
| // #endif | ||||
| 
 | ||||
| 
 | ||||
| /** H5、小程序环境中:引入需要的格式编码器、可视化插件,App环境中在renderjs中引入 **/ | ||||
| // 注意:如果App中需要在逻辑层中调用Recorder的编码/转码功能,需要去掉此条件编译,否则会报未加载编码器的错误 | ||||
| // #ifdef H5 || MP-WEIXIN | ||||
|     //按需引入你需要的录音格式支持文件,如果需要多个格式支持,把这些格式的编码引擎js文件统统引入进来即可 | ||||
|     import 'recorder-core/src/engine/mp3' | ||||
|     import 'recorder-core/src/engine/mp3-engine' //如果此格式有额外的编码引擎(*-engine.js)的话,必须要加上 | ||||
|      | ||||
|     //可选的插件支持项,把需要的插件按需引入进来即可 | ||||
|     import 'recorder-core/src/extensions/waveview' | ||||
| // #endif | ||||
| 
 | ||||
| // ... 这后面写页面代码,用选项式API风格(vue2、vue3)、setup组合式API风格(仅vue3)都可以 | ||||
| </script> | ||||
| ``` | ||||
| 
 | ||||
| 5. 编译成app时,默认需要额外提供一个renderjs模块,请照抄下面这段代码放到vue文件末尾 | ||||
| 
 | ||||
| ``` html | ||||
| <!-- #ifdef APP --> | ||||
| <script module="yourModuleName" lang="renderjs"> //此模块内部只能用选项式API风格,vue2、vue3均可用,请照抄这段代码;不可改成setup组合式API风格,否则可能不能import vue导致编译失败 | ||||
| /**需要编译成App时,你需要添加一个renderjs模块,然后一模一样的import上面那些js(微信的js除外) | ||||
|     ,因为App中默认是在renderjs(WebView)中进行录音和音频编码 | ||||
|     。如果配置了 RecordApp.UniWithoutAppRenderjs=true 且未调用依赖renderjs的功能时(如nvue、可视化、仅H5中可用的插件) | ||||
|     ,可不提供此renderjs模块,同时逻辑层中需要将相关import的条件编译去掉**/ | ||||
| import 'recorder-core' | ||||
| import RecordApp from 'recorder-core/src/app-support/app' | ||||
| import '../../uni_modules/Recorder-UniCore/app-uni-support.js' //renderjs中似乎不支持"@/"打头的路径,如果编译路径错误请改正路径即可 | ||||
| 
 | ||||
| //按需引入你需要的录音格式支持文件,和插件 | ||||
| import 'recorder-core/src/engine/mp3' | ||||
| import 'recorder-core/src/engine/mp3-engine'  | ||||
| 
 | ||||
| import 'recorder-core/src/extensions/waveview' | ||||
| 
 | ||||
| export default { | ||||
|     mounted(){ | ||||
|         //App的renderjs必须调用的函数,传入当前模块this | ||||
|         RecordApp.UniRenderjsRegister(this); | ||||
|     }, | ||||
|     methods: { | ||||
|         //这里定义的方法,在逻辑层中可通过 RecordApp.UniWebViewVueCall(this,'this.xxxFunc()') 直接调用 | ||||
|         //调用逻辑层的方法,请直接用 this.$ownerInstance.callMethod("xxxFunc",{args}) 调用,二进制数据需转成base64来传递 | ||||
|     } | ||||
| } | ||||
| </script> | ||||
| <!-- #endif --> | ||||
| ``` | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| ## 二、调用录音 | ||||
| ``` javascript | ||||
| /**在逻辑层中编写**/ | ||||
| //import ... 上面那些import代码 | ||||
| 
 | ||||
| //var vue3This=getCurrentInstance().proxy; //当用vue3 setup组合式 API (Composition API) 编写时,直接在import后面取到当前实例this,在需要this的地方传vue3This变量即可,其他的和选项式 API (Options API) 没有任何区别;import {getCurrentInstance} from 'vue';详细可以参考Demo项目中的 page_vue3____composition_api.vue | ||||
| 
 | ||||
| //RecordApp.UniNativeUtsPlugin={ nativePlugin:true };  //App中启用配套的原生录音插件支持,配置后会使用原生插件进行录音,没有原生插件时依旧使用renderjs H5录音 | ||||
| //App中提升后台录音的稳定性:配置了原生插件后,可配置 `RecordApp.UniWithoutAppRenderjs=true` 禁用renderjs层音频编码(WebWorker加速),变成逻辑层中直接编码(但会降低逻辑层性能),后台运行时可避免部分手机WebView运行受限的影响 | ||||
| //App中提升后台录音的稳定性:需要启用后台录音保活服务(iOS不需要,参考录音权限配置),Android 9开始,锁屏或进入后台一段时间后App可能会被禁止访问麦克风导致录音静音、无法录音(renderjs中H5录音也受影响),请调用配套原生插件的`androidNotifyService`接口,或使用第三方保活插件 | ||||
| 
 | ||||
| export default { | ||||
| data() { return {} } //视图没有引用到的变量无需放data里,直接this.xxx使用 | ||||
| 
 | ||||
| ,mounted() { | ||||
|     this.isMounted=true; | ||||
|     //页面onShow时【必须调用】的函数,传入当前组件this | ||||
|     RecordApp.UniPageOnShow(this); | ||||
| } | ||||
| ,onShow(){ //onShow可能比mounted先执行,页面可能还未准备好 | ||||
|     if(this.isMounted) RecordApp.UniPageOnShow(this); | ||||
| } | ||||
| 
 | ||||
| ,methods:{ | ||||
|     //请求录音权限 | ||||
|     recReq(){ | ||||
|         //编译成App时提供的授权许可(编译成H5、小程序为免费授权可不填写);如果未填写授权许可,将会在App打开后第一次调用请求录音权限时,弹出“未获得商用授权时,App上仅供测试”提示框 | ||||
|         //RecordApp.UniAppUseLicense='我已获得UniAppID=*****的商用授权'; | ||||
|          | ||||
|         //RecordApp.RequestPermission_H5OpenSet={ audioTrackSet:{ noiseSuppression:true,echoCancellation:true,autoGainControl:true } }; //这个是Start中的audioTrackSet配置,在h5(H5、App+renderjs)中必须提前配置,因为h5中RequestPermission会直接打开录音 | ||||
|          | ||||
|         RecordApp.UniWebViewActivate(this); //App环境下必须先切换成当前页面WebView | ||||
|         RecordApp.RequestPermission(()=>{ | ||||
|             console.log("已获得录音权限,可以开始录音了"); | ||||
|         },(msg,isUserNotAllow)=>{ | ||||
|             if(isUserNotAllow){//用户拒绝了录音权限 | ||||
|                 //这里你应当编写代码进行引导用户给录音权限,不同平台分别进行编写 | ||||
|             } | ||||
|             console.error("请求录音权限失败:"+msg); | ||||
|         }); | ||||
|     } | ||||
|      | ||||
|     //开始录音 | ||||
|     ,recStart(){ | ||||
|         //Android App如果要后台录音,需要启用后台录音保活服务(iOS不需要),需使用配套原生插件、或使用第三方保活插件 | ||||
|         //RecordApp.UniNativeUtsPluginCallAsync("androidNotifyService",{ title:"正在录音" ,content:"正在录音中,请勿关闭App运行" }).then(()=>{...}).catch((e)=>{...}) 注意必须RecordApp.RequestPermission得到权限后调用 | ||||
|          | ||||
|         //录音配置信息 | ||||
|         var set={ | ||||
|             type:"mp3",sampleRate:16000,bitRate:16 //mp3格式,指定采样率hz、比特率kbps,其他参数使用默认配置;注意:是数字的参数必须提供数字,不要用字符串;需要使用的type类型,需提前把格式支持文件加载进来,比如使用wav格式需要提前加载wav.js编码引擎 | ||||
|             /*,audioTrackSet:{ //可选,如果需要同时播放声音(比如语音通话),需要打开回声消除(并不一定会生效;打开后声音可能会从听筒播放,部分环境下(如小程序、App原生插件)可调用接口切换成扬声器外放) | ||||
|                 //注意:H5、App+renderjs中需要在请求录音权限前进行相同配置RecordApp.RequestPermission_H5OpenSet后此配置才会生效 | ||||
|                 echoCancellation:true,noiseSuppression:true,autoGainControl:true} */ | ||||
|             ,onProcess:(buffers,powerLevel,duration,sampleRate,newBufferIdx,asyncEnd)=>{ | ||||
|                 //全平台通用:可实时上传(发送)数据,配合Recorder.SampleData方法,将buffers中的新数据连续的转换成pcm上传,或使用mock方法将新数据连续的转码成其他格式上传,可以参考Recorder文档里面的:Demo片段列表 -> 实时转码并上传-通用版;基于本功能可以做到:实时转发数据、实时保存数据、实时语音识别(ASR)等 | ||||
|                  | ||||
|                 //注意:App里面是在renderjs中进行实际的音频格式编码操作,此处的buffers数据是renderjs实时转发过来的,修改此处的buffers数据不会改变renderjs中buffers,所以不会改变生成的音频文件,可在onProcess_renderjs中进行修改操作就没有此问题了;如需清理buffers内存,此处和onProcess_renderjs中均需要进行清理,H5、小程序中无此限制 | ||||
|                 //注意:如果你要用只支持在浏览器中使用的Recorder扩展插件,App里面请在renderjs中引入此扩展插件,然后在onProcess_renderjs中调用这个插件;H5可直接在这里进行调用,小程序不支持这类插件;如果调用插件的逻辑比较复杂,建议封装成js文件,这样逻辑层、renderjs中直接import,不需要重复编写 | ||||
|                  | ||||
|                 //H5、小程序等可视化图形绘制,直接运行在逻辑层;App里面需要在onProcess_renderjs中进行这些操作 | ||||
|                 // #ifdef H5 || MP-WEIXIN | ||||
|                 if(this.waveView) this.waveView.input(buffers[buffers.length-1],powerLevel,sampleRate); | ||||
|                 // #endif | ||||
|                  | ||||
|                 /*实时释放清理内存,用于支持长时间录音;在指定了有效的type时,编码器内部可能还会有其他缓冲,必须同时提供takeoffEncodeChunk才能清理内存,否则type需要提供unknown格式来阻止编码器内部缓冲,App的onProcess_renderjs中需要进行相同操作 | ||||
|                 if(this.clearBufferIdx>newBufferIdx){ this.clearBufferIdx=0 } //重新录音了就重置 | ||||
|                 for(var i=this.clearBufferIdx||0;i<newBufferIdx;i++) buffers[i]=null; | ||||
|                 this.clearBufferIdx=newBufferIdx; */ | ||||
|             } | ||||
|             ,onProcess_renderjs:`function(buffers,powerLevel,duration,sampleRate,newBufferIdx,asyncEnd){ | ||||
|                 //App中在这里修改buffers会改变生成的音频文件,但注意:buffers会先转发到逻辑层onProcess后才会调用本方法,因此在逻辑层的onProcess中需要重新修改一遍 | ||||
|                 //本方法可以返回true,renderjs中的onProcess将开启异步模式,处理完后调用asyncEnd结束异步,注意:这里异步修改的buffers一样的不会在逻辑层的onProcess中生效 | ||||
|                 //App中是在renderjs中进行的可视化图形绘制,因此需要写在这里,this是renderjs模块的this(也可以用This变量);如果代码比较复杂,请直接在renderjs的methods里面放个方法xxxFunc,这里直接使用this.xxxFunc(args)进行调用 | ||||
|                 if(this.waveView) this.waveView.input(buffers[buffers.length-1],powerLevel,sampleRate); | ||||
|                  | ||||
|                 /*和onProcess中一样进行释放清理内存,用于支持长时间录音 | ||||
|                 if(this.clearBufferIdx>newBufferIdx){ this.clearBufferIdx=0 } //重新录音了就重置 | ||||
|                 for(var i=this.clearBufferIdx||0;i<newBufferIdx;i++) buffers[i]=null; | ||||
|                 this.clearBufferIdx=newBufferIdx; */ | ||||
|             }` | ||||
|             ,onProcessBefore_renderjs:`function(buffers,powerLevel,duration,sampleRate,newBufferIdx){ | ||||
|                 //App中本方法会在逻辑层onProcess之前调用,因此修改的buffers会转发给逻辑层onProcess,本方法没有asyncEnd参数不支持异步处理 | ||||
|                 //一般无需提供本方法只用onProcess_renderjs就行,renderjs的onProcess内部调用过程:onProcessBefore_renderjs -> 转发给逻辑层onProcess -> onProcess_renderjs | ||||
|             }` | ||||
| 
 | ||||
|             ,takeoffEncodeChunk:true?null:(chunkBytes)=>{ | ||||
|                 //全平台通用:实时接收到编码器编码出来的音频片段数据,chunkBytes是Uint8Array二进制数据,可以实时上传(发送)出去 | ||||
|                 //App中如果未配置RecordApp.UniWithoutAppRenderjs时,建议提供此回调,因为录音结束后会将整个录音文件从renderjs传回逻辑层,由于uni-app的逻辑层和renderjs层数据交互性能实在太拉跨了,大点的文件传输会比较慢,提供此回调后可避免Stop时产生超大数据回传 | ||||
|                  | ||||
|                 //App中使用原生插件时,可方便的将数据实时保存到同一文件,第一帧时append:false新建文件,后面的append:true追加到文件 | ||||
|                 //RecordApp.UniNativeUtsPluginCallAsync("writeFile",{path:"xxx.mp3",append:回调次数!=1, dataBase64:RecordApp.UniBtoa(chunkBytes.buffer)}).then(...).catch(...) | ||||
|             } | ||||
|             ,takeoffEncodeChunk_renderjs:true?null:`function(chunkBytes){ | ||||
|                 //App中这里可以做一些仅在renderjs中才生效的事情,不提供也行,this是renderjs模块的this(也可以用This变量) | ||||
|             }` | ||||
|              | ||||
|             ,start_renderjs:`function(){ | ||||
|                 //App中可以放一个函数,在Start成功时renderjs中会先调用这里的代码,this是renderjs模块的this(也可以用This变量) | ||||
|                 //放一些仅在renderjs中才生效的事情,比如初始化,不提供也行 | ||||
|             }` | ||||
|             ,stop_renderjs:`function(arrayBuffer,duration,mime){ | ||||
|                 //App中可以放一个函数,在Stop成功时renderjs中会先调用这里的代码,this是renderjs模块的this(也可以用This变量) | ||||
|                 //放一些仅在renderjs中才生效的事情,不提供也行 | ||||
|             }` | ||||
|         }; | ||||
|          | ||||
|         RecordApp.UniWebViewActivate(this); //App环境下必须先切换成当前页面WebView | ||||
|         RecordApp.Start(set,()=>{ | ||||
|             console.log("已开始录音"); | ||||
|             //【稳如老狗WDT】可选的,监控是否在正常录音有onProcess回调,如果长时间没有回调就代表录音不正常 | ||||
|             //var wdt=this.watchDogTimer=setInterval ... 请参考示例Demo的main_recTest.vue中的watchDogTimer实现 | ||||
|              | ||||
|             //创建音频可视化图形绘制,App环境下是在renderjs中绘制,H5、小程序等是在逻辑层中绘制,因此需要提供两段相同的代码 | ||||
|             //view里面放一个canvas,canvas需要指定宽高(下面style里指定了300*100) | ||||
|             //<canvas type="2d" class="recwave-WaveView" style="width:300px;height:100px"></canvas> | ||||
|             RecordApp.UniFindCanvas(this,[".recwave-WaveView"],` | ||||
|                 this.waveView=Recorder.WaveView({compatibleCanvas:canvas1, width:300, height:100}); | ||||
|             `,(canvas1)=>{ | ||||
|                 this.waveView=Recorder.WaveView({compatibleCanvas:canvas1, width:300, height:100}); | ||||
|             }); | ||||
|         },(msg)=>{ | ||||
|             console.error("开始录音失败:"+msg); | ||||
|         }); | ||||
|     } | ||||
|      | ||||
|     //暂停录音 | ||||
|     ,recPause(){ | ||||
|         if(RecordApp.GetCurrentRecOrNull()){ | ||||
|             RecordApp.Pause(); | ||||
|             console.log("已暂停"); | ||||
|         } | ||||
|     } | ||||
|     //继续录音 | ||||
|     ,recResume(){ | ||||
|         if(RecordApp.GetCurrentRecOrNull()){ | ||||
|             RecordApp.Resume(); | ||||
|             console.log("继续录音中..."); | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     //停止录音 | ||||
|     ,recStop(){ | ||||
|         //RecordApp.UniNativeUtsPluginCallAsync("androidNotifyService",{ close:true }) //关闭Android App后台录音保活服务 | ||||
|          | ||||
|         RecordApp.Stop((arrayBuffer,duration,mime)=>{ | ||||
|             //全平台通用:arrayBuffer是音频文件二进制数据,可以保存成文件或者发送给服务器 | ||||
|             //App中如果在Start参数中提供了stop_renderjs,renderjs中的函数会比这个函数先执行 | ||||
|              | ||||
|             //注意:当Start时提供了takeoffEncodeChunk后,你需要自行实时保存录音文件数据,因此Stop时返回的arrayBuffer的长度将为0字节 | ||||
|              | ||||
|             //如果是H5环境,也可以直接构造成Blob/File文件对象,和Recorder使用一致 | ||||
|             // #ifdef H5 | ||||
|                 var blob=new Blob([arrayBuffer],{type:mime}); | ||||
|                 console.log(blob, (window.URL||webkitURL).createObjectURL(blob)); | ||||
|                 var file=new File([arrayBuffer],"recorder.mp3"); | ||||
|                 //uni.uploadFile({file:file, ...}) //参考demo中的test_upload_saveFile.vue | ||||
|             // #endif | ||||
|              | ||||
|             //如果是App、小程序环境,可以直接保存到本地文件,然后调用相关网络接口上传 | ||||
|             // #ifdef APP || MP-WEIXIN | ||||
|                 RecordApp.UniSaveLocalFile("recorder.mp3",arrayBuffer,(savePath)=>{ | ||||
|                     console.log(savePath); //app保存的文件夹为`plus.io.PUBLIC_DOWNLOADS`,小程序为 `wx.env.USER_DATA_PATH` 路径 | ||||
|                     //uni.uploadFile({filePath:savePath, ...}) //参考demo中的test_upload_saveFile.vue | ||||
|                 },(errMsg)=>{ console.error(errMsg) }); | ||||
|             // #endif | ||||
|         },(msg)=>{ | ||||
|             console.error("结束录音失败:"+msg); | ||||
|         }); | ||||
|     } | ||||
|      | ||||
| } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| # 录音权限配置、需要注意的细节 | ||||
| ## 编译成H5时录音和权限 | ||||
| 编译成H5时,录音功能由Recorder H5提供,无需额外处理录音权限。 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| ## 编译成微信小程序时录音和权限 | ||||
| 编译成微信小程序时,录音功能由小程序的`RecorderManager`提供,屏蔽了微信原有的底层细节(无录音时长限制)。 | ||||
| 
 | ||||
| 小程序录音需要用户授予录音权限,调用`RecordApp.RequestPermission`的时候会检查是否能正常录音,如果用户拒绝了录音权限,会进入错误回调,回调里面你应当编写代码检查`wx.getSetting`中的`scope.record`录音权限,然后引导用户进行授权(可调用`wx.openSetting`打开设置页面,方便用户给权限)。 | ||||
| 
 | ||||
| **注意:上架小程序需要到小程序管理后台《[用户隐私保护指引](https://developers.weixin.qq.com/miniprogram/dev/framework/user-privacy/miniprogram-intro.html)》中声明录音权限,否则正式版将无法调用录音功能(请求权限时会直接走错误回调)。** | ||||
| 
 | ||||
| 更多细节请参考 [miniProgram-wx](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample/miniProgram-wx) 测试项目文档。 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| ## 编译成App时录音和权限 | ||||
| 编译成App录音时,分两种情况: | ||||
| 1. 默认未配置`RecordApp.UniNativeUtsPlugin`(未使用原生录音插件和uts插件)时,会在renderjs中使用Recorder H5进行录音,录音数据会实时回传到逻辑层。 | ||||
| 2. 配置了`RecordApp.UniNativeUtsPlugin`使用原生录音插件或uts插件时,会直接调用原生插件进行录音;录音数据默认会传递到renderjs中进行音频编码处理(WebWorker加速),然后再实时回传到逻辑层,如果配置了`RecordApp.UniWithoutAppRenderjs=true`时,音频编码处理将会在逻辑层中直接处理。 | ||||
| 
 | ||||
| 当App是在renderjs中使用H5进行录音时(未使用原生录音插件和uts插件),iOS上只支持14.3以上版本,**且iOS上每次进入页面后第一次请求录音权限时、或长时间无操作再请求录音权限时WebView均会弹出录音权限对话框**,不同旧iOS版本(低于iOS17)下H5录音可能存在的问题在App中同样会存在;使用配套的[原生录音插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin)或uts插件时无以上问题和版本限制(uts插件开发中暂不可用),Android也无以上问题。 | ||||
| 
 | ||||
| 当音频编码是在renderjs中进行处理时,录音结束后会将整个录音文件传回逻辑层,由于uni-app的逻辑层和renderjs层大点的文件传输会比较慢,**建议Start时使用takeoffEncodeChunk实时获取音频文件数据可避免Stop时产生超大数据回传**;配置了`RecordApp.UniWithoutAppRenderjs=true`后,因为音频编码直接是在逻辑层中进行,将不存在传输性能损耗,但会影响逻辑层的性能(正常情况轻微不明显),需要配套使用原生录音插件才可以进行此项配置。 | ||||
| 
 | ||||
| 在调用`RecordApp.RequestPermission`的时候,`Recorder-UniCore`组件会自动处理好App的系统录音权限,只需要在uni-app项目的 `manifest.json` 中配置好Android和iOS的录音权限声明。 | ||||
| ``` | ||||
| //Android需要勾选的权限,第二个也必须勾选 | ||||
| <uses-permission android:name="android.permission.RECORD_AUDIO"/> | ||||
| <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/> | ||||
| 【注意】Android如果需要在后台录音,需要启用后台录音保活服务,Android 9开始,锁屏或进入后台一段时间后App可能会被禁止访问麦克风导致录音静音、无法录音(renderjs中H5录音、原生插件录音均受影响),请调用配套原生插件的`androidNotifyService`接口,或使用第三方保活插件 | ||||
| 
 | ||||
| //iOS需要声明的权限 | ||||
| NSMicrophoneUsageDescription | ||||
| 【注意】iOS需要在 `App常用其它设置`->`后台运行能力`中提供`audio`配置,不然App切到后台后立马会停止录音 | ||||
| ``` | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| ## PCM音频流式播放、语音通话、回声消除、声音外放 | ||||
| 在App、H5中,均可使用H5版的[BufferStreamPlayer](https://gitee.com/xiangyuecn/Recorder/blob/master/src/extensions/buffer_stream.player.js)来实时流式播放语音;其中App中需要在renderjs中加载BufferStreamPlayer,在逻辑层中调用`RecordApp.UniWebViewVueCall`等方法将逻辑层中接收到的实时语音数据发送到renderjs中播放;播放声音的同时进行录音,声音可能会被录进去产生回声,因此一般需要打开回声消除;调用代码参考demo中的[test_realtime_voice.vue](https://gitee.com/xiangyuecn/Recorder/blob/master/app-support-sample/demo_UniApp/pages/recTest/test_realtime_voice.vue)。 | ||||
| 
 | ||||
| App中如果搭配使用了配套的[原生录音插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin),可以调用原生实现的PcmPlayer播放器实时流式播放PCM音频,边录音边播放更流畅;同时也支持完整播放,比如AI语音合成的播放;调用代码参考demo中的[test_player_nativePlugin_pcmPlayer.vue](https://gitee.com/xiangyuecn/Recorder/blob/master/app-support-sample/demo_UniApp/pages/recTest/test_player_nativePlugin_pcmPlayer.vue)。 | ||||
| 
 | ||||
| 微信小程序请参考 [miniProgram-wx](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample/miniProgram-wx) 文档里面的同名章节,使用WebAudioContext播放。 | ||||
| 
 | ||||
| 配置audioTrackSet可尝试打开回声消除,或者切换听筒播放或外放,打开回声消除时,一般会转为听筒播放显著降低回声。 | ||||
| ``` js | ||||
| //打开回声消除 | ||||
| RecordApp.Start({ | ||||
|     ... 更多配置参数请参考RecordApp文档 | ||||
|     //此配置App、H5、小程序均可打开回声消除;注意:H5、App+renderjs中需要在请求录音权限前进行相同配置RecordApp.RequestPermission_H5OpenSet后此配置才会生效 | ||||
|     ,audioTrackSet:{echoCancellation:true,noiseSuppression:true,autoGainControl:true} | ||||
|      | ||||
|     //Android指定麦克风源(App搭配原生插件、小程序可用),0 DEFAULT 默认音频源,1 MIC 主麦克风,5 CAMCORDER 相机方向的麦,6 VOICE_RECOGNITION 语音识别,7 VOICE_COMMUNICATION 语音通信(带回声消除) | ||||
|     ,android_audioSource:7 //提供此配置时优先级比audioTrackSet更高,默认值为0 | ||||
|      | ||||
|     //iOS的AVAudioSession setCategory的withOptions参数值(App搭配原生插件可用),取值请参考配套原生插件文档中的iosSetDefault_categoryOptions | ||||
|     //,ios_categoryOptions:0x1|0x4 //默认值为5(0x1|0x4) | ||||
| }); | ||||
| 
 | ||||
| //App搭配原生插件时尝试切换听筒播放或外放 | ||||
| await RecordApp.UniNativeUtsPluginCallAsync("setSpeakerOff",{off:true或false}); | ||||
| //小程序尝试切换 | ||||
| wx.setInnerAudioOption({ speakerOn:false或true }) | ||||
| //H5不支持切换 | ||||
| ``` | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| # 详细文档、RecordApp方法、属性文档 | ||||
| 请先阅读 [demo_UniApp文档](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample/demo_UniApp),含Demo项目;更高级使用还需深入阅读 [Recorder文档](https://gitee.com/xiangyuecn/Recorder)、[RecordApp文档](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample) (均为完整的一个README.md文档),Recorder文档中包含了更丰富的示例代码:基础录音、实时处理、格式转码、音频分析、音频混音、音频生成 等等,大部分能在uniapp中直接使用。 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| # 本组件的授权许可限制 | ||||
| **本组件内的app-uni-support.js文件在uni-app中编译到App平台时仅供测试用(App平台包括:Android App、iOS App),不可用于正式发布或商用,正式发布或商用需先到DCloud插件市场购买[此带授权的插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin-Android)(费用为¥199元,赠送Android版原生插件),即可获得授权许可**;编译到其他平台时无此授权限制,比如:H5、小程序,均为免费授权。 | ||||
| 
 | ||||
| 在App中,如果未获得授权许可,将会在App打开后第一次调用`RecordApp.RequestPermission`请求录音权限时,弹出“未获得商用授权时,App上仅供测试”提示框。 | ||||
| 
 | ||||
| 在DCloud插件市场购买了[带授权的插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin-Android)获得了授权后,请在调用`RecordApp.RequestPermission`请求录音权限前,赋值`RecordApp.UniAppUseLicense="我已获得UniAppID=***的商用授权"`(星号为你项目的uni-app应用标识),就不会弹提示框了;或者直接使用配套的[原生录音插件](https://ext.dcloud.net.cn/plugin?name=Recorder-NativePlugin),设置`RecordApp.UniNativeUtsPlugin`参数后,也不会弹提示框;其他情况请联系作者咨询,更多细节请参考[本组件的GitHub文档](https://gitee.com/xiangyuecn/Recorder/tree/master/app-support-sample/demo_UniApp)。 | ||||
| 
 | ||||
| 获取授权、需要技术支持、或有不清楚的地方可以联系我们,客服联系方式:QQ 1251654593 ,或者直接联系作者QQ 753610399 (回复可能没有客服及时)。 | ||||
| 
 | ||||
| 插件开发维护不易,感谢支持~ | ||||
| 
 | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| *⠀* | ||||
| 
 | ||||
| 
 | ||||
| @ -1,2 +0,0 @@ | ||||
| ## 1.0.0(2023-09-05) | ||||
| 实现基础功能 | ||||
| @ -1,130 +0,0 @@ | ||||
| <template> | ||||
| 	<view :style="[boxStyel]" class="box"> | ||||
| 		<view class="audio-style" :style="[audioStyel]" :class="{ 'animation': isPlay }"> | ||||
| 			<view class="small" :style="{'background-color': color}"></view> | ||||
| 			<view class="middle" :style="{'border-right-color': color}"></view> | ||||
| 			<view class="large" :style="{'border-right-color': color}"></view> | ||||
| 		</view> | ||||
| 	</view> | ||||
| </template> | ||||
| <script> | ||||
| export default { | ||||
| 	emits: [], | ||||
| 	props: { | ||||
| 		isPlay: { | ||||
| 			type: [Boolean], | ||||
| 			default: false | ||||
| 		}, | ||||
| 		direction: { | ||||
| 			type: String, | ||||
| 			default: 'right' | ||||
| 		}, | ||||
| 		size: { | ||||
| 			type: Number, | ||||
| 			default: 24 | ||||
| 		}, | ||||
| 		color: { | ||||
| 			type: String, | ||||
| 			default: '#222' | ||||
| 		} | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		audioStyel() { | ||||
| 			return { | ||||
| 				transform: `scale(${this.size / 24})` | ||||
| 			}; | ||||
| 		}, | ||||
| 		boxStyel() { | ||||
| 			const directDic = { right: '0deg', bottom: '90deg', left: '180deg', top: '270deg' }; | ||||
| 			const dir = directDic[this.direction || 'left']; | ||||
| 			const style = { | ||||
| 				transform: `rotate(${dir})`, | ||||
| 				width: this.size + 'px', | ||||
| 				height: this.size + 'px' | ||||
| 			}; | ||||
| 			return style; | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: {} | ||||
| }; | ||||
| </script> | ||||
| <style lang="scss" scoped> | ||||
| view{ | ||||
| 	box-sizing: border-box; | ||||
| } | ||||
| .box { | ||||
| 	// border: 1px solid #4c4c4c; | ||||
| 	display: inline-flex; | ||||
| 	align-items: center; | ||||
| 	justify-content: center; | ||||
| 	overflow: hidden; | ||||
| } | ||||
| .audio-style { | ||||
| 	display: flex; | ||||
| 	align-items: center; | ||||
| 	justify-content: center; | ||||
| 	overflow: hidden; | ||||
| 	& > view { | ||||
| 		border: 2px solid transparent; | ||||
| 		border-radius: 50%; | ||||
| 	} | ||||
| } | ||||
| .small { | ||||
| 	border: 0px solid; | ||||
| 	width: 3px; | ||||
| 	height: 3px; | ||||
| } | ||||
| 
 | ||||
| .middle { | ||||
| 	width: 16px; | ||||
| 	height: 16px; | ||||
| 	margin-left: -11px; | ||||
| 	opacity: 1; | ||||
| } | ||||
| 
 | ||||
| .large { | ||||
| 	width: 24px; | ||||
| 	height: 24px; | ||||
| 	margin-left: -19px; | ||||
| 	opacity: 1; | ||||
| } | ||||
| .animation { | ||||
| 	.middle { | ||||
| 		animation: middle 1.2s ease-in-out infinite; | ||||
| 	} | ||||
| 	.large { | ||||
| 		animation: large 1.2s ease-in-out infinite; | ||||
| 	} | ||||
| } | ||||
| // 语音播放动画 | ||||
| @keyframes middle { | ||||
| 	0% { | ||||
| 		opacity: 0; | ||||
| 	} | ||||
| 	10% { | ||||
| 		opacity: 1; | ||||
| 	} | ||||
| 	100% { | ||||
| 		opacity: 0; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| @keyframes large { | ||||
| 	0% { | ||||
| 		opacity: 0; | ||||
| 	} | ||||
| 	50% { | ||||
| 		opacity: 1; | ||||
| 	} | ||||
| 	60% { | ||||
| 		opacity: 0; | ||||
| 	} | ||||
| 	100% { | ||||
| 		opacity: 0; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||