Compare commits
	
		
			3 Commits
		
	
	
		
			8f519f0efc
			...
			b93183eec3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b93183eec3 | |||
| 268c131aa7 | |||
| 5e32b9ae03 | 
							
								
								
									
										27
									
								
								src/api/deps/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,27 @@ | ||||
| import request from '@/service/index.js' | ||||
| 
 | ||||
| export const departmentV2TreeMy = (data) => { | ||||
|   return request({ | ||||
|     url: '/department/v2/tree/my', | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const departmentV2AllPosition = (data) => { | ||||
|   return request({ | ||||
|     url: '/department/v2/tree/all/position', | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const groupCreateDept = (data) => { | ||||
|   return request({ | ||||
|     url: '/api/v1/group/create-dept', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| @ -7,3 +7,21 @@ export const ServeGetUserSetting = (data) => { | ||||
|     data, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const userInfoApi = (data) => { | ||||
|   return request({ | ||||
|     url: '/user/info', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const userLeaderApi = (data) => { | ||||
|   return request({ | ||||
|     url: '/oa/user/other-info', | ||||
|     method: 'POST', | ||||
|     data, | ||||
|     baseURL:import.meta.env.VITE_EPR_BASEURL, | ||||
|   }) | ||||
| } | ||||
|  | ||||
							
								
								
									
										56
									
								
								src/components/checkBox/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,56 @@ | ||||
| <template> | ||||
|   <div @click="onCheck" > | ||||
|     <tm-image :width="size" :height="size" :src="imageSrc"></tm-image> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| import { defineProps,computed,defineEmits } from "vue"; | ||||
| import circle from "@/static/image/checkBox/circle@2x.png"; | ||||
| import zu6039 from "@/static/image/checkBox/zu6039@2x.png"; | ||||
| import zu6040 from "@/static/image/checkBox/zu6040@2x.png"; | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   modelValue: { | ||||
|     type: String, | ||||
|     default: "noChecked", // noChecked, checked, halfChecked | ||||
|   }, | ||||
|   size: { | ||||
|     type: Number, | ||||
|     default: 28, | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| const emit = defineEmits(['update:modelValue','change']); | ||||
| 
 | ||||
| const imageSrc = computed(() => { | ||||
|   switch (props.modelValue) { | ||||
|     case "noChecked": | ||||
|       return circle; | ||||
|     case "checked": | ||||
|       return zu6039; | ||||
|     case "halfChecked": | ||||
|       return zu6040; | ||||
|   } | ||||
| }); | ||||
| 
 | ||||
| const onCheck = () => { | ||||
|   switch (props.modelValue) { | ||||
|     case "noChecked": | ||||
|       emit('update:modelValue', 'checked'); | ||||
|       emit('change', 'checked'); | ||||
|       break; | ||||
|     case "checked": | ||||
|       emit('update:modelValue', 'noChecked'); | ||||
|       emit('change', 'noChecked'); | ||||
|       break; | ||||
|     case "halfChecked": | ||||
|       emit('update:modelValue', 'checked'); | ||||
|       emit('change', 'checked'); | ||||
|       break; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| </script> | ||||
| <style scoped lang='less'> | ||||
| </style> | ||||
| @ -38,6 +38,22 @@ | ||||
|         "enablePullDownRefresh": false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/chooseDeps/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh":false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/chooseGroupAdmin/index", | ||||
|       "type": "page", | ||||
|       "style": { | ||||
|         "navigationStyle": "custom", | ||||
|         "enablePullDownRefresh":false | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "path": "pages/chooseChat/index", | ||||
|       "type": "page", | ||||
|  | ||||
							
								
								
									
										520
									
								
								src/pages/chooseDeps/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,520 @@ | ||||
| <template> | ||||
|   <div class="outer-layer"> | ||||
|     <div> | ||||
|       <tm-navbar :hideBack="false" hideHome title="选择部门"> </tm-navbar> | ||||
|     </div> | ||||
|     <div class="root"> | ||||
|       <div | ||||
|         class="w-full h-[1134rpx] mb-[20rpx] pl-[32rpx] pr-[32rpx] pb-[20rpx] overflow-y-auto" | ||||
|       > | ||||
|         <div class="pl-[16rpx] pr-[16rpx] pt-[22rpx] pb-[24rpx] bg-[#FFFFFF]"> | ||||
|           <tm-input | ||||
|             placeholder="请输入…" | ||||
|             color="#F9F9FD" | ||||
|             :round="1" | ||||
|             prefix="tmicon-search" | ||||
|             prefixColor="#46299D" | ||||
|             searchBgColor="#F9F9FD" | ||||
|             focusColor="#F9F9FD" | ||||
|             v-model="searchVal" | ||||
|           ></tm-input> | ||||
|           <div | ||||
|             v-if="crumbs.length" | ||||
|             class="w-full overflow-x-auto mt-[22rpx] leading-[48rpx] text-[#2F2F2F] flex items-center no-scrollbar" | ||||
|             ref="crumbsContainer" | ||||
|             > | ||||
|             <div | ||||
|               v-for="(item, index) in crumbs" | ||||
|               class="flex items-center text-[28rpx] leading-[48rpx] whitespace-nowrap" | ||||
|               :class="[ | ||||
|                 index === crumbsIndex ? 'text-[#747474]' : 'text-[#46299D]', | ||||
|                 index === 0 ? '' : 'ml-[12rpx]', | ||||
|               ]" | ||||
|               @click="handleCrumbsClick(index)" | ||||
|             > | ||||
|               <div>{{ item.name }}</div> | ||||
|               <div | ||||
|                 v-if="index !== crumbs.length - 1" | ||||
|                 class="ml-[20rpx] flex items-center mb-[2rpx]" | ||||
|               > | ||||
|                 <tm-icon | ||||
|                   name="tmicon-angle-right" | ||||
|                   :font-size="20" | ||||
|                   :color="index !== crumbs.length - 1 ? '#7A58DE' : '#C1B4EA'" | ||||
|                 ></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div | ||||
|           class="pl-[32rpx] bg-[#FFFFFF] mt-[20rpx] h-[110rpx] flex items-center" | ||||
|           @click="()=>allCheck(allCheckStatus)" | ||||
|         > | ||||
|           <div> | ||||
|             <checkBox | ||||
|               :disabled="currentCrumbs?.sons?.length === 0" | ||||
|               :modelValue="allCheckStatus" | ||||
|             ></checkBox> | ||||
|           </div> | ||||
|           <div class="font-bold text-[28rpx] leading-[54rpx] ml-[20rpx]"> | ||||
|             全选 | ||||
|           </div> | ||||
|         </div> | ||||
|         <div | ||||
|           v-if="currentCrumbs?.sons?.length" | ||||
|           v-for="item in currentCrumbs?.sons" | ||||
|           class="pl-[32rpx] bg-[#FFFFFF] mt-[20rpx] h-[110rpx] flex items-center" | ||||
|         > | ||||
|           <div class="w-full flex items-center justify-between"> | ||||
|             <div | ||||
|               class="flex items-center" | ||||
|             > | ||||
|               <div> | ||||
|                 <checkBox | ||||
|                   :disabled="!item?.sons?.length" | ||||
|                   v-model="item.checkStatus" | ||||
|                   @change="(val) => checkItemChange(item, val)" | ||||
|                 ></checkBox> | ||||
|               </div> | ||||
|               <div class="font-bold text-[28rpx] leading-[54rpx] ml-[20rpx]"> | ||||
|                 {{ item.name }} | ||||
|               </div> | ||||
|             </div> | ||||
|             <div v-if="item.sons?.length" class="flex items-center mr-[32rpx]"> | ||||
|               <div class="vDivider mr-[32rpx]"></div> | ||||
|               <div @click="() => toNextLevel(item)" class="flex items-center"> | ||||
|                 <div class="mr-[12rpx]"> | ||||
|                   <tm-image | ||||
|                     :width="26" | ||||
|                     :height="26" | ||||
|                     :src="item.checkStatus !== 'checked' ? downDep : downDepDis" | ||||
|                   ></tm-image> | ||||
|                 </div> | ||||
|                 <div | ||||
|                   class="text-[28rpx] leading-[54rpx] font-bold" | ||||
|                   :class=" | ||||
|                     item.checkStatus !== 'checked' | ||||
|                       ? 'text-[#46299D]' | ||||
|                       : 'text-[#C1B4EA]' | ||||
|                   " | ||||
|                 > | ||||
|                   下级 | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <div class="h-[162rpx] pl-[32rpx] pr-[32rpx] bg-[#FFFFFF]"> | ||||
|         <div class="mt-[14rpx] flex justify-between"> | ||||
|           <div class="flex flex-col"> | ||||
|             <div @click="openDrawer" class="flex items-center text-[28rpx] leading-[60rpx] text-[#000000] font-bold"> | ||||
|               <div>已选择部门数:</div> | ||||
|               <div>{{ allCheckedList.length }}</div> | ||||
|               <div class="ml-[28rpx]"> | ||||
|                 <tm-icon :fontSize="24" color="#46299D" name="tmicon-angle-up"></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="text-[24rpx] leading-[24rpx] text-[#7A58DE] w-[280rpx] truncate"> | ||||
|               {{ allCheckedList.map(v=>v.name).toString() }} | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="btnBox"> | ||||
|             <tm-button | ||||
|               @click="handleConfirm" | ||||
|               color="#46299D" | ||||
|               :disabled="!allCheckedList.length" | ||||
|               disabledColor="#E6E6E6" | ||||
|               :margin="[0]" | ||||
|               :shadow="0" | ||||
|               :width="264" | ||||
|               :height="76" | ||||
|               size="large" | ||||
|               label="确定" | ||||
|             > | ||||
|             </tm-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <tm-drawer  | ||||
|       placement="bottom"  | ||||
|       v-model:show="showWin"  | ||||
|       :height="800"  | ||||
|       :hideHeader="true" | ||||
|       :round="5" | ||||
|     > | ||||
|       <div class="flex flex-col w-full h-full pt-[36rpx] pl-[32rpx] pr-[32rpx] leading-[60rpx]" > | ||||
|         <div class="text-[32rpx] font-bold flex items-center justify-between" > | ||||
|           <div class="flex items-center ml-[10rpx]"> | ||||
|             <div>已选择部门数:</div> | ||||
|             <div>{{ allCheckedList.length }}</div> | ||||
|           </div> | ||||
|           <div  | ||||
|             class="text-[#7A58DE] mr-[10rpx]" | ||||
|             @click="()=>showWin = false" | ||||
|           > | ||||
|             确定 | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="flex-1 pb-[20rpx] overflow-y-auto pt-[30rpx]" > | ||||
|           <div | ||||
|             v-for="(item,index) in allCheckedList" | ||||
|             class="flex flex-col" | ||||
|           > | ||||
|             <div v-if="index ===0" class="divider" ></div> | ||||
|             <div  | ||||
|               class="flex items-center justify-between mt-[36rpx] font-bold text-[#000000] leading-[54rpx] mb-[34rpx]"  | ||||
|             > | ||||
|               <div class="text-[28rpx] ml-[10rpx]">{{ item.name }}</div> | ||||
|               <div class="diyBtn"> | ||||
|                 <tm-button @click="()=>deleteNode(item)" :disabled="userDepIds.includes(item.ID)" :margin="[10]" :shadow="0" text size="small" :width="106" :height="50" outlined label="移除"></tm-button> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="divider" ></div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </tm-drawer> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import { ref, watch, computed, onMounted,nextTick } from "vue"; | ||||
| import { onShow, onLoad } from "@dcloudio/uni-app"; | ||||
| import { useChatList } from "@/store/chatList/index.js"; | ||||
| import { useAuth } from "@/store/auth"; | ||||
| import { useTalkStore, useUserStore } from "@/store"; | ||||
| import { useGroupTypeStore } from "@/store/groupType"; | ||||
| import downDep from "@/static/image/chatList/downDep.png"; | ||||
| import downDepDis from "@/static/image/chatList/downDepDis.png"; | ||||
| import checkBox from "@/components/checkBox/index.vue"; | ||||
| import lodash from 'lodash' | ||||
| 
 | ||||
| const { | ||||
|   groupActiveIndex, | ||||
|   getDepsTreeMy, | ||||
|   depTreeMyList, | ||||
|   crumbs, | ||||
|   crumbsIndex, | ||||
|   depCheckedKeys, | ||||
| } = useGroupTypeStore(); | ||||
| const userStore = useUserStore() | ||||
| 
 | ||||
| const searchVal = ref(""); | ||||
| const crumbsContainer = ref(null); | ||||
| const showWin = ref(false); | ||||
| 
 | ||||
| const getAllCheckedNodes = (node, checkedNodes = []) => { | ||||
|   if (node.checkStatus === 'checked') { | ||||
|     checkedNodes.push(node); | ||||
|   } | ||||
|   if (node.sons && Array.isArray(node.sons)) { | ||||
|     node.sons.forEach(son => getAllCheckedNodes(son, checkedNodes)); | ||||
|   } | ||||
|   return checkedNodes; | ||||
| }; | ||||
| 
 | ||||
| const userDepIds = computed(() => { | ||||
|   return userStore.deps.map(v=>v.dept_id); | ||||
| }); | ||||
| 
 | ||||
| const allCheckedList = computed(() => { | ||||
|   const checkedNodes = []; | ||||
|   depTreeMyList.value.forEach(node => getAllCheckedNodes(node, checkedNodes)); | ||||
|   console.log("checkedNodes", checkedNodes); | ||||
|    | ||||
|   return checkedNodes; | ||||
| }) | ||||
| 
 | ||||
| const currentCrumbs = computed(() => { | ||||
|   if (crumbs.value[crumbsIndex.value] ) { | ||||
|     if (searchVal.value && searchVal.value !== "") { | ||||
|       let filterSons = crumbs.value[crumbsIndex.value].sons.filter((item) => | ||||
|         item.name.includes(searchVal.value) | ||||
|       ); | ||||
|         return { | ||||
|           ...crumbs.value[crumbsIndex.value], | ||||
|           sons: filterSons, | ||||
|         }; | ||||
|     } | ||||
|     console.log("allCheckedList", crumbs.value[crumbsIndex.value]); | ||||
|     return crumbs.value[crumbsIndex.value]; | ||||
|   } | ||||
|   return {} | ||||
| }); | ||||
| 
 | ||||
| const allCheckStatus = computed(() => { | ||||
|   if (!currentCrumbs.value.sons) { | ||||
|     return "noChecked"; | ||||
|   } | ||||
|   const allChecked = currentCrumbs.value.sons.every((son) => son.checkStatus === "checked"); | ||||
|   const someChecked =currentCrumbs.value.sons.some( | ||||
|       (son) => | ||||
|         son.checkStatus === "checked" || son.checkStatus === "halfChecked" | ||||
|     ); | ||||
|     if (allChecked) { | ||||
|     return "checked"; | ||||
|   } else if (someChecked) { | ||||
|     return "halfChecked"; | ||||
|   } else { | ||||
|     return "noChecked"; | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const findNodeById = (node, targetId) => { | ||||
|   if (node.ID === targetId) { | ||||
|     return node; | ||||
|   } | ||||
|   if (node.sons && Array.isArray(node.sons)) { | ||||
|     for (const son of node.sons) { | ||||
|       const found = findNodeById(son, targetId); | ||||
|       if (found) { | ||||
|         return found; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   return null; | ||||
| }; | ||||
| 
 | ||||
| const findParentNode = (node, targetId) => { | ||||
|   if (!node.sons || !Array.isArray(node.sons)) return null; | ||||
|   for (const son of node.sons) { | ||||
|     if (son.ID === targetId) { | ||||
|       return node; | ||||
|     } | ||||
|     const parent = findParentNode(son, targetId); | ||||
|     if (parent) { | ||||
|       return parent; | ||||
|     } | ||||
|   } | ||||
|   return null; | ||||
| }; | ||||
| 
 | ||||
| // 检查并更新父节点的状态 | ||||
| const updateParentStatus = (node) => { | ||||
|   const parent = findParentNode(depTreeMyList.value[0], node.ID); | ||||
|   if (!parent) return; | ||||
|   if (parent.checkStatus !== "checked"){ | ||||
|     const allChecked = | ||||
|       parent.sons && | ||||
|       Array.isArray(parent.sons) && | ||||
|       parent.sons.every((son) => son.checkStatus === "checked"); | ||||
|     const someChecked = | ||||
|       parent.sons && | ||||
|       Array.isArray(parent.sons) && | ||||
|       parent.sons.some( | ||||
|         (son) => | ||||
|           son.checkStatus === "checked" || son.checkStatus === "halfChecked" | ||||
|       ); | ||||
|    | ||||
|     if (allChecked) { | ||||
|       parent.checkStatus = "halfChecked"; | ||||
|     } else if (someChecked) { | ||||
|       parent.checkStatus = "halfChecked"; | ||||
|     } else { | ||||
|       parent.checkStatus = "noChecked"; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   updateParentStatus(parent); | ||||
| }; | ||||
| //根据子节点状态更新当前节点状态 | ||||
| const updateNodeStatus = (node) => { | ||||
|   if (!node.sons || !Array.isArray(node.sons)) return; | ||||
|   const allChecked = node.sons.every(son => son.checkStatus === 'checked'); | ||||
|   const someChecked = node.sons.some(son => son.checkStatus === 'checked' || son.checkStatus === 'halfChecked'); | ||||
| 
 | ||||
|   if (allChecked) { | ||||
|     node.checkStatus = 'halfChecked'; | ||||
|   } else if (someChecked) { | ||||
|     node.checkStatus = 'halfChecked'; | ||||
|   } else { | ||||
|     node.checkStatus = 'noChecked'; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| // 更新当前节点及其所有子节点的状态 | ||||
| const updateCheckStatus = (node, status) => { | ||||
|   node.checkStatus = status; | ||||
|   if (node.sons && Array.isArray(node.sons) && node.sons.length > 0) { | ||||
|     node.sons.forEach((son) => updateCheckStatus(son, status)); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const checkItemChange = (item, val) => { | ||||
|   // 更新当前节点及其子节点的状态 | ||||
|   updateCheckStatus(item, val); | ||||
| 
 | ||||
|   // 更新父节点的状态 | ||||
|   updateParentStatus(item); | ||||
| }; | ||||
| 
 | ||||
| const deleteNode = (item) => { | ||||
|   const treeNode = findNodeById(depTreeMyList.value[0], item.ID); | ||||
|   if (!treeNode) return; | ||||
|   treeNode.checkStatus = "noChecked"; | ||||
|   updateNodeStatus(treeNode); | ||||
|   updateParentStatus(treeNode); | ||||
| }; | ||||
| 
 | ||||
| const toNextLevel = async (item) => { | ||||
|   if (item.checkStatus !== "checked") { | ||||
|     crumbs.value.push(item); | ||||
|     crumbsIndex.value++; | ||||
|     await nextTick();  | ||||
|     if (crumbsContainer.value) { | ||||
|       crumbsContainer.value.scrollLeft = crumbsContainer.value.scrollWidth; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const handleCrumbsClick = (index) => { | ||||
|   crumbsIndex.value = index; | ||||
|   crumbs.value = crumbs.value.slice(0, index + 1); | ||||
| }; | ||||
| 
 | ||||
| const allCheck = (status) => { | ||||
|   let statusT = 'noChecked'; | ||||
|   if (status === "checked") { | ||||
|     statusT = "noChecked"; | ||||
|   } else { | ||||
|     statusT = "checked"; | ||||
|   } | ||||
|   currentCrumbs.value.sons.forEach((item) => { | ||||
|     const itemT = findNodeById(depTreeMyList.value[0], item.ID) | ||||
|     if (!itemT) return; | ||||
|     checkItemChange(itemT, statusT); | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| const openDrawer = () => { | ||||
|   showWin.value = true; | ||||
|   if (allCheckedList.length>0) { | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| // watch(() => depTreeMyList.value, (newValue, oldValue) => { | ||||
| //   console.log("depTreeMyList", newValue); | ||||
|    | ||||
| // }, { deep: true }); | ||||
| 
 | ||||
| // watch(() => searchVal.value, (newValue, oldValue) => { | ||||
| //   console.log("searchVal", newValue); | ||||
| // }); | ||||
| 
 | ||||
| const handleConfirm = () => { | ||||
|   depCheckedKeys.value = lodash.cloneDeep(allCheckedList.value)  | ||||
|   console.log("depCheckedKeys", depCheckedKeys.value); | ||||
|    | ||||
|   uni.navigateBack(); | ||||
| } | ||||
| 
 | ||||
| const initCheckedKeys = () => { | ||||
|   depCheckedKeys.value.forEach((item) => { | ||||
|     const node = findNodeById(depTreeMyList.value[0], item.ID); | ||||
|     if (node) { | ||||
|       node.checkStatus = "checked"; | ||||
|       updateParentStatus(node); | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| const init = async () => { | ||||
|   crumbsIndex.value = 0; | ||||
|   await getDepsTreeMy(); | ||||
|   crumbs.value = depTreeMyList.value.length ? [depTreeMyList.value[0]] : []; | ||||
|   initCheckedKeys(); | ||||
| }; | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   init(); | ||||
| }); | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| uni-page-body, | ||||
| page { | ||||
|   height: 100%; | ||||
| } | ||||
| .outer-layer { | ||||
|   overflow-y: auto; | ||||
|   flex: 1; | ||||
|   background-image: url("@/static/image/clockIn/z3280@3x.png"); | ||||
|   background-size: cover; | ||||
|   padding-bottom: 0; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
| } | ||||
| .root { | ||||
|   flex: 1; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   justify-content: space-between; | ||||
|   padding-top: 18rpx; | ||||
|   overflow: hidden; | ||||
| } | ||||
| .divider { | ||||
|   height: 1rpx; | ||||
|   background-color: #707070; | ||||
|   opacity: 0.1; | ||||
| } | ||||
| .vDivider { | ||||
|   width: 1rpx; | ||||
|   height: 48rpx; | ||||
|   background-color: #b4b4b4; | ||||
| } | ||||
| 
 | ||||
| .avatar-placeholder { | ||||
|   width: 192rpx; | ||||
|   height: 192rpx; | ||||
|   background-color: #e0e0e0; | ||||
|   border-radius: 50%; | ||||
|   margin-bottom: 40rpx; | ||||
| } | ||||
| .groupCard { | ||||
|   height: 272rpx; | ||||
|   width: 100%; | ||||
|   background-size: cover; | ||||
|   background-position: center; | ||||
|   background-repeat: no-repeat; | ||||
|   border-radius: 12rpx; | ||||
|   &.firstPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6033@2x.png"); | ||||
|   } | ||||
|   &.secondPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6031@2x.png"); | ||||
|     margin-top: 28rpx; | ||||
|     margin-bottom: 28rpx; | ||||
|   } | ||||
|   &.thirdPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6032@2x.png"); | ||||
|   } | ||||
|   &.activePanel { | ||||
|     box-shadow: 0 0 0 3rpx #46299d; | ||||
|   } | ||||
| } | ||||
| .btnBox { | ||||
|   :deep(uni-button[disabled="true"]) { | ||||
|     color: #bebebe !important; | ||||
|   } | ||||
| } | ||||
| .no-scrollbar { | ||||
|   -ms-overflow-style: none;  /* IE and Edge */ | ||||
|   scrollbar-width: none;  /* Firefox */ | ||||
| } | ||||
| 
 | ||||
| .no-scrollbar::-webkit-scrollbar { | ||||
|   display: none;  /* Chrome, Safari, and Opera */ | ||||
| } | ||||
| .diyBtn { | ||||
|   :deep(uni-button) { | ||||
|     color: #191919 !important; | ||||
|     border: 1rpx solid #D6D6D8 !important; | ||||
|     background-color: #FFFFFF !important; | ||||
|     &[disabled="true"] { | ||||
|       color: #BEBEBE !important; | ||||
|       border: 1rpx solid #E6E6E6 !important; | ||||
|       background-color: #E6E6E6 !important; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										532
									
								
								src/pages/chooseGroupAdmin/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,532 @@ | ||||
| <template> | ||||
|   <div class="outer-layer"> | ||||
|     <div> | ||||
|       <tm-navbar :hideBack="false" hideHome title="群管理员"> </tm-navbar> | ||||
|     </div> | ||||
|     <div class="root"> | ||||
|       <div | ||||
|         class="w-full h-[1134rpx] mb-[20rpx] pl-[32rpx] pr-[32rpx] pb-[20rpx] overflow-y-auto" | ||||
|       > | ||||
|         <div class="pl-[16rpx] pr-[16rpx] pt-[22rpx] pb-[24rpx] bg-[#FFFFFF]"> | ||||
|           <tm-input | ||||
|             placeholder="请输入…" | ||||
|             color="#F9F9FD" | ||||
|             :round="1" | ||||
|             prefix="tmicon-search" | ||||
|             prefixColor="#46299D" | ||||
|             searchBgColor="#F9F9FD" | ||||
|             focusColor="#F9F9FD" | ||||
|             v-model="searchVal" | ||||
|           ></tm-input> | ||||
|           <div | ||||
|             v-if="crumbs.length" | ||||
|             class="w-full overflow-x-auto mt-[22rpx] leading-[48rpx] text-[#2F2F2F] flex items-center no-scrollbar" | ||||
|             ref="crumbsContainer" | ||||
|             > | ||||
|             <div | ||||
|               v-for="(item, index) in crumbs" | ||||
|               class="flex items-center text-[28rpx] leading-[48rpx] whitespace-nowrap" | ||||
|               :class="[ | ||||
|                 index === crumbsIndex ? 'text-[#747474]' : 'text-[#46299D]', | ||||
|                 index === 0 ? '' : 'ml-[12rpx]', | ||||
|               ]" | ||||
|               @click="handleCrumbsClick(index)" | ||||
|             > | ||||
|               <div>{{ item.name }}</div> | ||||
|               <div | ||||
|                 v-if="index !== crumbs.length - 1" | ||||
|                 class="ml-[20rpx] flex items-center mb-[2rpx]" | ||||
|               > | ||||
|                 <tm-icon | ||||
|                   name="tmicon-angle-right" | ||||
|                   :font-size="20" | ||||
|                   :color="index !== crumbs.length - 1 ? '#7A58DE' : '#C1B4EA'" | ||||
|                 ></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <!-- <div | ||||
|           class="pl-[32rpx] bg-[#FFFFFF] mt-[20rpx] h-[110rpx] flex items-center" | ||||
|           @click="()=>allCheck(allCheckStatus)" | ||||
|         > | ||||
|           <div> | ||||
|             <checkBox | ||||
|               :disabled="currentCrumbs?.sons?.length === 0" | ||||
|               :modelValue="allCheckStatus" | ||||
|             ></checkBox> | ||||
|           </div> | ||||
|           <div class="font-bold text-[28rpx] leading-[54rpx] ml-[20rpx]"> | ||||
|             全选 | ||||
|           </div> | ||||
|         </div> --> | ||||
|         <div | ||||
|           v-if="currentCrumbs?.sons?.length" | ||||
|           v-for="item in currentCrumbs?.sons" | ||||
|           class="pl-[32rpx] bg-[#FFFFFF] mt-[20rpx] h-[110rpx] flex items-center" | ||||
|         > | ||||
|           <div class="w-full flex items-center justify-between"> | ||||
|             <div | ||||
|               class="flex items-center" | ||||
|             > | ||||
|               <div> | ||||
|                 <!-- <checkBox | ||||
|                   :disabled="!item?.sons?.length" | ||||
|                   v-model="item.checkStatus" | ||||
|                   @change="(val) => checkItemChange(item, val)" | ||||
|                 ></checkBox> --> | ||||
|               </div> | ||||
|               <div class="font-bold text-[28rpx] leading-[54rpx] ml-[20rpx]"> | ||||
|                 {{ item.name }} | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="flex items-center mr-[32rpx]"> | ||||
|               <div class="vDivider mr-[32rpx]"></div> | ||||
|               <div @click="() => toNextLevel(item)" class="flex items-center"> | ||||
|                 <div class="mr-[12rpx]"> | ||||
|                   <tm-image | ||||
|                     :width="26" | ||||
|                     :height="26" | ||||
|                     :src="item.checkStatus !== 'checked' ? downDep : downDepDis" | ||||
|                   ></tm-image> | ||||
|                 </div> | ||||
|                 <div | ||||
|                   class="text-[28rpx] leading-[54rpx] font-bold" | ||||
|                   :class=" | ||||
|                     item.checkStatus !== 'checked' | ||||
|                       ? 'text-[#46299D]' | ||||
|                       : 'text-[#C1B4EA]' | ||||
|                   " | ||||
|                 > | ||||
|                   下级 | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div | ||||
|           v-if="currentCrumbs?.positions?.length" | ||||
|           v-for="item in currentCrumbs?.positions" | ||||
|           class="pl-[32rpx] bg-[#FFFFFF] mt-[20rpx] h-[110rpx] flex items-center" | ||||
|         > | ||||
|           <div class="w-full flex items-center justify-between"> | ||||
|             <div | ||||
|               class="flex items-center" | ||||
|             > | ||||
|               <div> | ||||
|                 <checkBox | ||||
|                   :disabled="!item?.sons?.length" | ||||
|                   :modelValue="choosePostList.find(v=>v.ID === item.ID)?'checked':'noChecked'" | ||||
|                   @change="(val) => checkItemChange(item, val)" | ||||
|                 ></checkBox> | ||||
|               </div> | ||||
|               <div class="font-bold text-[28rpx] leading-[54rpx] ml-[20rpx]"> | ||||
|                 {{ item.name }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <div class="h-[162rpx] pl-[32rpx] pr-[32rpx] bg-[#FFFFFF]"> | ||||
|         <div class="mt-[14rpx] flex justify-between"> | ||||
|           <div class="flex flex-col"> | ||||
|             <div @click="openDrawer" class="flex items-center text-[28rpx] leading-[60rpx] text-[#000000] font-bold"> | ||||
|               <div>已选择岗位数:</div> | ||||
|               <div>{{ choosePostList.length }}</div> | ||||
|               <div class="ml-[28rpx]"> | ||||
|                 <tm-icon :fontSize="24" color="#46299D" name="tmicon-angle-up"></tm-icon> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="text-[24rpx] leading-[24rpx] text-[#7A58DE] w-[280rpx] truncate"> | ||||
|               {{ choosePostList.map(v=>v.name).toString() }} | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="btnBox"> | ||||
|             <tm-button | ||||
|               @click="handleConfirm" | ||||
|               color="#46299D" | ||||
|               :disabled="!choosePostList.length" | ||||
|               disabledColor="#E6E6E6" | ||||
|               :margin="[0]" | ||||
|               :shadow="0" | ||||
|               :width="264" | ||||
|               :height="76" | ||||
|               size="large" | ||||
|               label="确定" | ||||
|             > | ||||
|             </tm-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <tm-drawer  | ||||
|       placement="bottom"  | ||||
|       v-model:show="showWin"  | ||||
|       :height="800"  | ||||
|       :hideHeader="true" | ||||
|       :round="5" | ||||
|     > | ||||
|       <div class="flex flex-col w-full h-full pt-[36rpx] pl-[32rpx] pr-[32rpx] leading-[60rpx]" > | ||||
|         <div class="text-[32rpx] font-bold flex items-center justify-between" > | ||||
|           <div class="flex items-center ml-[10rpx]"> | ||||
|             <div>已选择岗位数:</div> | ||||
|             <div>{{ choosePostList.length }}</div> | ||||
|           </div> | ||||
|           <div  | ||||
|             class="text-[#7A58DE] mr-[10rpx]" | ||||
|             @click="()=>showWin = false" | ||||
|           > | ||||
|             确定 | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="flex-1 pb-[20rpx] overflow-y-auto pt-[30rpx]" > | ||||
|           <div | ||||
|             v-for="(item,index) in choosePostList" | ||||
|             class="flex flex-col" | ||||
|           > | ||||
|             <div v-if="index ===0" class="divider" ></div> | ||||
|             <div  | ||||
|               class="flex items-center justify-between mt-[36rpx] font-bold text-[#000000] leading-[54rpx] mb-[34rpx]"  | ||||
|             > | ||||
|               <div class="text-[28rpx] ml-[10rpx]">{{ item.departmentName }}-{{ item.name }}</div> | ||||
|               <div class="diyBtn"> | ||||
|                 <tm-button @click="()=>deleteNode(item)" :disabled="userDepIds.includes(item.ID)" :margin="[10]" :shadow="0" text size="small" :width="106" :height="50" outlined label="移除"></tm-button> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="divider" ></div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </tm-drawer> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup> | ||||
| import { ref, watch, computed, onMounted,nextTick } from "vue"; | ||||
| import { onShow, onLoad } from "@dcloudio/uni-app"; | ||||
| import { useChatList } from "@/store/chatList/index.js"; | ||||
| import { useAuth } from "@/store/auth"; | ||||
| import { useTalkStore, useUserStore } from "@/store"; | ||||
| import { useGroupTypeStore } from "@/store/groupType"; | ||||
| import downDep from "@/static/image/chatList/downDep.png"; | ||||
| import downDepDis from "@/static/image/chatList/downDepDis.png"; | ||||
| import checkBox from "@/components/checkBox/index.vue"; | ||||
| import lodash from 'lodash' | ||||
| 
 | ||||
| const { | ||||
|   groupActiveIndex, | ||||
|   getPositionsTree, | ||||
|   postTreeList, | ||||
|   crumbs, | ||||
|   crumbsIndex, | ||||
|   depCheckedKeys, | ||||
|   groupAdmins, | ||||
| } = useGroupTypeStore(); | ||||
| const userStore = useUserStore() | ||||
| 
 | ||||
| // 1 groupAdmin 2 groupMember | ||||
| const pageType = ref(1);  | ||||
| const searchVal = ref(""); | ||||
| const crumbsContainer = ref(null); | ||||
| const showWin = ref(false); | ||||
| const choosePostList = ref([]) | ||||
| 
 | ||||
| const getAllCheckedNodes = (node, checkedNodes = []) => { | ||||
|   if (node.checkStatus === 'checked') { | ||||
|     checkedNodes.push(node); | ||||
|   } | ||||
|   if (node.sons && Array.isArray(node.sons)) { | ||||
|     node.sons.forEach(son => getAllCheckedNodes(son, checkedNodes)); | ||||
|   } | ||||
|   return checkedNodes; | ||||
| }; | ||||
| 
 | ||||
| const userDepIds = computed(() => { | ||||
|   return userStore.deps.map(v=>v.dept_id); | ||||
| }); | ||||
| 
 | ||||
| const currentCrumbs = computed(() => { | ||||
|   if (crumbs.value[crumbsIndex.value] ) { | ||||
|     if (searchVal.value && searchVal.value !== "") { | ||||
|       let filterSons = crumbs.value[crumbsIndex.value].sons.filter((item) => | ||||
|         item.name.includes(searchVal.value) | ||||
|       ); | ||||
|         return { | ||||
|           ...crumbs.value[crumbsIndex.value], | ||||
|           sons: filterSons, | ||||
|         }; | ||||
|     } | ||||
|     console.log("allCheckedList", crumbs.value[crumbsIndex.value]); | ||||
|     return crumbs.value[crumbsIndex.value]; | ||||
|   } | ||||
|   return {} | ||||
| }); | ||||
| 
 | ||||
| // const allCheckStatus = computed(() => { | ||||
| //   if (!currentCrumbs.value.sons) { | ||||
| //     return "noChecked"; | ||||
| //   } | ||||
| //   const allChecked = currentCrumbs.value.sons.every((son) => son.checkStatus === "checked"); | ||||
| //   const someChecked =currentCrumbs.value.sons.some( | ||||
| //       (son) => | ||||
| //         son.checkStatus === "checked" || son.checkStatus === "halfChecked" | ||||
| //     ); | ||||
| //     if (allChecked) { | ||||
| //     return "checked"; | ||||
| //   } else if (someChecked) { | ||||
| //     return "halfChecked"; | ||||
| //   } else { | ||||
| //     return "noChecked"; | ||||
| //   } | ||||
| // }) | ||||
| 
 | ||||
| const findNodeById = (node, targetId) => { | ||||
|   if (node.ID === targetId) { | ||||
|     return node; | ||||
|   } | ||||
|   if (node.sons && Array.isArray(node.sons)) { | ||||
|     for (const son of node.sons) { | ||||
|       const found = findNodeById(son, targetId); | ||||
|       if (found) { | ||||
|         return found; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   return null; | ||||
| }; | ||||
| 
 | ||||
| const findParentNode = (node, targetId) => { | ||||
|   if (!node.sons || !Array.isArray(node.sons)) return null; | ||||
|   for (const son of node.sons) { | ||||
|     if (son.ID === targetId) { | ||||
|       return node; | ||||
|     } | ||||
|     const parent = findParentNode(son, targetId); | ||||
|     if (parent) { | ||||
|       return parent; | ||||
|     } | ||||
|   } | ||||
|   return null; | ||||
| }; | ||||
| 
 | ||||
| // 检查并更新父节点的状态 | ||||
| const updateParentStatus = (node) => { | ||||
|   const parent = findParentNode(postTreeList.value[0], node.ID); | ||||
|   if (!parent) return; | ||||
|   if (parent.checkStatus !== "checked"){ | ||||
|     const allChecked = | ||||
|       parent.sons && | ||||
|       Array.isArray(parent.sons) && | ||||
|       parent.sons.every((son) => son.checkStatus === "checked"); | ||||
|     const someChecked = | ||||
|       parent.sons && | ||||
|       Array.isArray(parent.sons) && | ||||
|       parent.sons.some( | ||||
|         (son) => | ||||
|           son.checkStatus === "checked" || son.checkStatus === "halfChecked" | ||||
|       ); | ||||
|    | ||||
|     if (allChecked) { | ||||
|       parent.checkStatus = "halfChecked"; | ||||
|     } else if (someChecked) { | ||||
|       parent.checkStatus = "halfChecked"; | ||||
|     } else { | ||||
|       parent.checkStatus = "noChecked"; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   updateParentStatus(parent); | ||||
| }; | ||||
| //根据子节点状态更新当前节点状态 | ||||
| const updateNodeStatus = (node) => { | ||||
|   if (!node.sons || !Array.isArray(node.sons)) return; | ||||
|   const allChecked = node.sons.every(son => son.checkStatus === 'checked'); | ||||
|   const someChecked = node.sons.some(son => son.checkStatus === 'checked' || son.checkStatus === 'halfChecked'); | ||||
| 
 | ||||
|   if (allChecked) { | ||||
|     node.checkStatus = 'halfChecked'; | ||||
|   } else if (someChecked) { | ||||
|     node.checkStatus = 'halfChecked'; | ||||
|   } else { | ||||
|     node.checkStatus = 'noChecked'; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| // 更新当前节点及其所有子节点的状态 | ||||
| const updateCheckStatus = (node, status) => { | ||||
|   node.checkStatus = status; | ||||
|   if (node.sons && Array.isArray(node.sons) && node.sons.length > 0) { | ||||
|     node.sons.forEach((son) => updateCheckStatus(son, status)); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const checkItemChange = (item, val) => { | ||||
|   if (val === 'checked') { | ||||
|     let depItem = findNodeById(postTreeList.value[0],item.departmentID) | ||||
|     choosePostList.value.push({ | ||||
|       ...item, | ||||
|       departmentName:depItem.name, | ||||
|     }) | ||||
|   }else{ | ||||
|     choosePostList.value = choosePostList.value.filter(v=>v.ID !== item.ID) | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const deleteNode = (item) => { | ||||
|   choosePostList.value = choosePostList.value.filter(v=>v.ID !== item.ID) | ||||
| }; | ||||
| 
 | ||||
| const toNextLevel = async (item) => { | ||||
|   if (item.checkStatus !== "checked") { | ||||
|     crumbs.value.push(item); | ||||
|     crumbsIndex.value++; | ||||
|     await nextTick();  | ||||
|     if (crumbsContainer.value) { | ||||
|       crumbsContainer.value.scrollLeft = crumbsContainer.value.scrollWidth; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const handleCrumbsClick = (index) => { | ||||
|   crumbsIndex.value = index; | ||||
|   crumbs.value = crumbs.value.slice(0, index + 1); | ||||
| }; | ||||
| 
 | ||||
| // const allCheck = (status) => { | ||||
| //   let statusT = 'noChecked'; | ||||
| //   if (status === "checked") { | ||||
| //     statusT = "noChecked"; | ||||
| //   } else { | ||||
| //     statusT = "checked"; | ||||
| //   } | ||||
| //   currentCrumbs.value.sons.forEach((item) => { | ||||
| //     const itemT = findNodeById(postTreeList.value[0], item.ID) | ||||
| //     if (!itemT) return; | ||||
| //     checkItemChange(itemT, statusT); | ||||
| //   }); | ||||
| // }; | ||||
| 
 | ||||
| const openDrawer = () => { | ||||
|   showWin.value = true; | ||||
|   if (allCheckedList.length>0) { | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| watch(() => postTreeList.value, (newValue, oldValue) => { | ||||
|   console.log("postTreeList", newValue); | ||||
|    | ||||
| }, { deep: true }); | ||||
| 
 | ||||
| // watch(() => searchVal.value, (newValue, oldValue) => { | ||||
| //   console.log("searchVal", newValue); | ||||
| // }); | ||||
| 
 | ||||
| const handleConfirm = () => { | ||||
|   groupAdmins.value = lodash.cloneDeep(choosePostList.value)  | ||||
|   console.log("groupAdmins", groupAdmins.value); | ||||
|    | ||||
|   uni.navigateBack(); | ||||
| } | ||||
| 
 | ||||
| const initCheckedKeys = () => { | ||||
|   choosePostList.value = groupAdmins.value | ||||
| }; | ||||
| 
 | ||||
| const init = async () => { | ||||
|   crumbsIndex.value = 0; | ||||
|   await getPositionsTree(); | ||||
|   crumbs.value = postTreeList.value.length ? [postTreeList.value[0]] : []; | ||||
|   initCheckedKeys(); | ||||
| }; | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   init(); | ||||
| }); | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| uni-page-body, | ||||
| page { | ||||
|   height: 100%; | ||||
| } | ||||
| .outer-layer { | ||||
|   overflow-y: auto; | ||||
|   flex: 1; | ||||
|   background-image: url("@/static/image/clockIn/z3280@3x.png"); | ||||
|   background-size: cover; | ||||
|   padding-bottom: 0; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
| } | ||||
| .root { | ||||
|   flex: 1; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   justify-content: space-between; | ||||
|   padding-top: 18rpx; | ||||
|   overflow: hidden; | ||||
| } | ||||
| .divider { | ||||
|   height: 1rpx; | ||||
|   background-color: #707070; | ||||
|   opacity: 0.1; | ||||
| } | ||||
| .vDivider { | ||||
|   width: 1rpx; | ||||
|   height: 48rpx; | ||||
|   background-color: #b4b4b4; | ||||
| } | ||||
| 
 | ||||
| .avatar-placeholder { | ||||
|   width: 192rpx; | ||||
|   height: 192rpx; | ||||
|   background-color: #e0e0e0; | ||||
|   border-radius: 50%; | ||||
|   margin-bottom: 40rpx; | ||||
| } | ||||
| .groupCard { | ||||
|   height: 272rpx; | ||||
|   width: 100%; | ||||
|   background-size: cover; | ||||
|   background-position: center; | ||||
|   background-repeat: no-repeat; | ||||
|   border-radius: 12rpx; | ||||
|   &.firstPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6033@2x.png"); | ||||
|   } | ||||
|   &.secondPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6031@2x.png"); | ||||
|     margin-top: 28rpx; | ||||
|     margin-bottom: 28rpx; | ||||
|   } | ||||
|   &.thirdPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6032@2x.png"); | ||||
|   } | ||||
|   &.activePanel { | ||||
|     box-shadow: 0 0 0 3rpx #46299d; | ||||
|   } | ||||
| } | ||||
| .btnBox { | ||||
|   :deep(uni-button[disabled="true"]) { | ||||
|     color: #bebebe !important; | ||||
|   } | ||||
| } | ||||
| .no-scrollbar { | ||||
|   -ms-overflow-style: none;  /* IE and Edge */ | ||||
|   scrollbar-width: none;  /* Firefox */ | ||||
| } | ||||
| 
 | ||||
| .no-scrollbar::-webkit-scrollbar { | ||||
|   display: none;  /* Chrome, Safari, and Opera */ | ||||
| } | ||||
| .diyBtn { | ||||
|   :deep(uni-button) { | ||||
|     color: #191919 !important; | ||||
|     border: 1rpx solid #D6D6D8 !important; | ||||
|     background-color: #FFFFFF !important; | ||||
|     &[disabled="true"] { | ||||
|       color: #BEBEBE !important; | ||||
|       border: 1rpx solid #E6E6E6 !important; | ||||
|       background-color: #E6E6E6 !important; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @ -1,12 +1,12 @@ | ||||
| <template> | ||||
|   <div class="outer-layer"> | ||||
|     <div> | ||||
|       <tm-navbar :hideBack="false" hideHome title="选择群类型" :leftWidth="320"> | ||||
|       <tm-navbar :hideBack="false" hideHome title="选择群类型"> | ||||
|       </tm-navbar> | ||||
|     </div> | ||||
|     <div class="root"> | ||||
|       <div class="w-full pl-[32rpx] pr-[32rpx] overflow-y-auto" > | ||||
|         <div class=" pl-[32rpx] pr-[32rpx] pt-[44rpx] bg-[#FFFFFF]" > | ||||
|       <div class="w-full h-[1134rpx] mb-[20rpx] pl-[32rpx] pr-[32rpx] overflow-y-auto" > | ||||
|         <div class=" pl-[32rpx] pr-[32rpx] pt-[44rpx] pb-[42rpx] bg-[#FFFFFF]" > | ||||
|           <div class="text-[40rpx] leading-[54rpx] text-[#2F2F2F] font-bold" > | ||||
|             群类型保存后将不可修改 | ||||
|           </div> | ||||
| @ -14,24 +14,72 @@ | ||||
|             请创建过程中正确选择 | ||||
|           </div> | ||||
|           <div class="mt-[54rpx] w-full h-[872rpx]" > | ||||
|             <div class="groupCard"> | ||||
| 
 | ||||
|             <div @click="groupActiveIndex = 0" class="groupCard firstPanel" :class="groupActiveIndex === 0?'activePanel':''"> | ||||
|               <div class="w-full h-full pt-[64rpx] pl-[36rpx]"> | ||||
|                 <div class="text-[36rpx] leading-[54rpx] text-[#2F2F2F] font-bold" > | ||||
|                   普通群 | ||||
|                 </div> | ||||
|                 <div class="text-[24rpx] leading-[36rpx] text-[#939393] w-[216rpx]" > | ||||
|                   员工线上沟通专用群 | ||||
|                   离职后自动退群 | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div @click="groupActiveIndex = 1" class="groupCard secondPanel" :class="groupActiveIndex === 1?'activePanel':''"> | ||||
|               <div class="w-full h-full pr-[36rpx] flex flex-col items-end justify-center"> | ||||
|                 <div class="text-[36rpx] leading-[54rpx] text-[#2F2F2F] font-bold" > | ||||
|                   部门群 | ||||
|                 </div> | ||||
|                 <div class="text-[24rpx] leading-[36rpx] text-[#939393] w-[288rpx] text-end" > | ||||
|                   指定部门员工入职自动进群 | ||||
|                   离职后自动退群 | ||||
|                 </div> | ||||
|                 <div v-if="!depCheckedKeys.length" @click="chooseDep" class="text-[24rpx] leading-[36rpx] text-[#C1B4EA] flex items-center mt-[16rpx]" > | ||||
|                   <div :class="groupActiveIndex === 1?'text-[#7A58DE]':'text-[#C1B4EA]'" >选择部门</div> | ||||
|                   <div class="ml-[20rpx]"> | ||||
|                     <tm-icon name="tmicon-angle-right" :font-size="18" :color="groupActiveIndex === 1?'#7A58DE':'#C1B4EA'"></tm-icon> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                 <div v-else class="text-[24rpx] leading-[36rpx] flex flex-col justify-center items-end mt-[16rpx] mb-[22rpx]" > | ||||
|                   <div class="max-w-[336rpx] truncate" > | ||||
|                     {{ depCheckedKeys.map(v=>v.name).toString() }} | ||||
|                   </div> | ||||
|                   <div @click="chooseDep" class="flex items-center" > | ||||
|                     <div class="text-[#7A58DE]" >查看全部</div> | ||||
|                     <div class="ml-[20rpx]"> | ||||
|                       <tm-icon name="tmicon-angle-right" :font-size="18" :color="groupActiveIndex === 1?'#7A58DE':'#C1B4EA'"></tm-icon> | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div @click="groupActiveIndex = 2" class="groupCard thirdPanel" :class="groupActiveIndex === 2?'activePanel':''"> | ||||
|               <div class="w-full h-full pt-[64rpx] pl-[36rpx]"> | ||||
|                 <div class="text-[36rpx] leading-[54rpx] text-[#2F2F2F] font-bold" > | ||||
|                   项目群 | ||||
|                 </div> | ||||
|                 <div class="text-[24rpx] leading-[36rpx] text-[#939393] w-[216rpx]" > | ||||
|                   项目成员沟通专用群 | ||||
|                   离职后自动退群 | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <div class="h-[162rpx] flex justify-center bg-[#FFFFFF]"> | ||||
|         <div class="mt-[14rpx]" > | ||||
|         <div class="mt-[14rpx] btnBox" > | ||||
|           <tm-button  | ||||
|             @click="handleConfirm" | ||||
|             color="#46299D"  | ||||
|             disabled="true" | ||||
|             :disabled="confirmBtnStatus" | ||||
|             disabledColor="#E6E6E6" | ||||
|             :margin="[0]" | ||||
|             :shadow="0"  | ||||
|             :width="426" | ||||
|             :height="76" | ||||
|             size="large"  | ||||
|             label="按钮" | ||||
|             label="确定" | ||||
|           > | ||||
|           </tm-button> | ||||
|         </div> | ||||
| @ -45,18 +93,40 @@ import { onShow, onLoad } from "@dcloudio/uni-app"; | ||||
| import { useChatList } from "@/store/chatList/index.js"; | ||||
| import { useAuth } from "@/store/auth"; | ||||
| import { useTalkStore, useUserStore } from "@/store"; | ||||
| import addCircle from "@/static/image/chatList/addCircle.png"; | ||||
| import cahtPopover from "@/static/image/chatList/cahtPopover.png"; | ||||
| import zu4992 from "@/static/image/chatList/zu4992@2x.png"; | ||||
| import zu4991 from "@/static/image/chatList/zu4991@2x.png"; | ||||
| import zu4989 from "@/static/image/chatList/zu4989@2x.png"; | ||||
| import { useGroupTypeStore } from "@/store/groupType"; | ||||
| 
 | ||||
| const talkStore = useTalkStore(); | ||||
| const userStore = useUserStore(); | ||||
| const { userInfo } = useAuth(); | ||||
| const { groupActiveIndex,depCheckedKeys } = useGroupTypeStore(); | ||||
| 
 | ||||
| const groupChatName = ref(""); | ||||
| const groupChatType = ref(''); | ||||
| const confirmBtnStatus = computed(() => { | ||||
|   let disabledT = false; | ||||
|   switch (groupActiveIndex.value) { | ||||
|     case 0: | ||||
|       break; | ||||
|     case 1: | ||||
|       if (!depCheckedKeys.value.length) { | ||||
|         disabledT = true; | ||||
|       } | ||||
|       break; | ||||
|     case 2: | ||||
|       break; | ||||
|     default: | ||||
|       break; | ||||
|   } | ||||
|   return disabledT | ||||
| }) | ||||
| 
 | ||||
| const chooseDep = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/chooseDeps/index' | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| const handleConfirm = () => { | ||||
|   uni.navigateBack() | ||||
|   // uni.navigateTo({ | ||||
|   //   url: '/pages/creatGroupChat/index' | ||||
|   // }); | ||||
| }; | ||||
| 
 | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| @ -94,7 +164,31 @@ page { | ||||
|   margin-bottom: 40rpx; | ||||
| } | ||||
| .groupCard { | ||||
|   display: block; | ||||
|   height: 272rpx; | ||||
|   width: 100%; | ||||
|   background-size: cover; | ||||
|   background-position: center; | ||||
|   background-repeat: no-repeat; | ||||
|   border-radius: 12rpx; | ||||
|   &.firstPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6033@2x.png"); | ||||
|   } | ||||
|   &.secondPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6031@2x.png"); | ||||
|     margin-top: 28rpx; | ||||
|     margin-bottom: 28rpx; | ||||
|   } | ||||
|   &.thirdPanel { | ||||
|     background-image: url("@/static/image/chatList/zu6032@2x.png"); | ||||
|   } | ||||
|   &.activePanel { | ||||
|     box-shadow: 0 0 0 3rpx #46299D; | ||||
|   } | ||||
| } | ||||
| .btnBox { | ||||
|   :deep(uni-button[disabled="true"]){ | ||||
|     color: #BEBEBE !important; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| </style> | ||||
|  | ||||
| @ -5,9 +5,9 @@ | ||||
|       </tm-navbar> | ||||
|     </div> | ||||
|     <div class="root"> | ||||
|       <div class="w-full pl-[32rpx] pr-[32rpx]" > | ||||
|       <div class="w-full pl-[32rpx] pr-[32rpx] mb-[20rpx]" > | ||||
|         <div class="w-full mt-[60rpx] flex justify-center" > | ||||
|           <div v-if="!groupChatType" class="avatar-placeholder"> | ||||
|           <div v-if="groupActiveIndex === -1" class="avatar-placeholder"> | ||||
|              | ||||
|           </div> | ||||
|           <div class="mb-[40rpx]" v-else> | ||||
| @ -19,51 +19,108 @@ | ||||
|             ></tm-image> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="input-group"> | ||||
|         <div class="input-group flex items-center justify-between" > | ||||
|           <div class="input-item"> | ||||
|             群名称 | ||||
|           </div> | ||||
|           <div class="input-box" > | ||||
|             <tm-input  | ||||
|               v-model="groupName" | ||||
|               :followTheme="false"  | ||||
|               color="#fff"  | ||||
|               fontColor="#747474"  | ||||
|               placeholderStyle="color: #B4B4B4" | ||||
|               focusColor="#FFF"  | ||||
|               :fontSize="28" | ||||
|               :maxlength="20" | ||||
|               :height="40" | ||||
|               :transprent="true"  | ||||
|               placeholder="请输入群名称(1~20个字)" | ||||
|               :padding="[0]" | ||||
|               :padding="[0,0]" | ||||
|               align="right" | ||||
|             > | ||||
|             </tm-input> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="input-group mt-[20rpx]"> | ||||
|         <div 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="chooseGroupType" class="left-box" > | ||||
|               <div class="text-[#B4B4B4] text-[28rpx]" > | ||||
|               请选择群类型 | ||||
|                 <span v-if="groupActiveIndex ===-1">请选择群类型</span>  | ||||
|                 <span v-else-if="groupActiveIndex ===0">普通群</span> | ||||
|                 <span v-else-if="groupActiveIndex ===1">部门群</span> | ||||
|                 <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> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div v-if="depCheckedKeys.length" 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&&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 = !depsNoExpanded" class="w-[100rpx]"> | ||||
|                 {{ depsNoExpanded? '展开':'收起' }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <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> | ||||
|                 <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" | ||||
|               :class="[ | ||||
|                 index !==0 ? 'mt-[10rpx]':'', | ||||
|                 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 = !depsNoExpanded" class="w-[100rpx]"> | ||||
|                 {{ depsNoExpanded? '展开':'收起' }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
| 
 | ||||
|       </div> | ||||
|       <div class="h-[162rpx] flex justify-center bg-[#FFFFFF]"> | ||||
|         <div class="mt-[14rpx]" > | ||||
|         <div class="mt-[14rpx] btnBox" > | ||||
|           <tm-button  | ||||
|             @click="handleConfirm" | ||||
|             color="#46299D"  | ||||
|             disabled="true" | ||||
|             :disabled="confirmBtnStatus" | ||||
|             disabledColor="#E6E6E6" | ||||
|             :margin="[0]" | ||||
|             :shadow="0"  | ||||
|             :width="426" | ||||
|             :height="76" | ||||
|             size="large"  | ||||
|             label="按钮" | ||||
|             label="发起群聊" | ||||
|           > | ||||
|           </tm-button> | ||||
|         </div> | ||||
| @ -82,26 +139,28 @@ import cahtPopover from "@/static/image/chatList/cahtPopover.png"; | ||||
| import zu4992 from "@/static/image/chatList/zu4992@2x.png"; | ||||
| import zu4991 from "@/static/image/chatList/zu4991@2x.png"; | ||||
| import zu4989 from "@/static/image/chatList/zu4989@2x.png"; | ||||
| import { useGroupTypeStore } from "@/store/groupType"; | ||||
| 
 | ||||
| const { groupName,groupActiveIndex,depCheckedKeys,groupAdmins,createDepGroup,resetGroupInfo } = useGroupTypeStore(); | ||||
| const talkStore = useTalkStore(); | ||||
| const userStore = useUserStore(); | ||||
| const { userInfo } = useAuth(); | ||||
| 
 | ||||
| const groupChatName = ref(""); | ||||
| const groupChatType = ref(''); | ||||
| const depsNoExpanded = ref(true); | ||||
| 
 | ||||
| const avatarImg = computed(() => { | ||||
|   let srcT = ""; | ||||
|   switch (groupChatType.value) { | ||||
|     case '普通群': | ||||
|   switch (groupActiveIndex.value) { | ||||
|     case 0: | ||||
|       srcT = zu4992; | ||||
|       break; | ||||
|     case '项目群': | ||||
|       srcT = zu4991; | ||||
|       break; | ||||
|     case '部门群': | ||||
|     case 1: | ||||
|       srcT = zu4989; | ||||
|       break; | ||||
|     case 2: | ||||
|       srcT = zu4991; | ||||
|       break; | ||||
|     default: | ||||
|       srcT = zu4992; | ||||
|   } | ||||
| @ -114,6 +173,55 @@ const chooseGroupType = () => { | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| const chooseGroupAdmin = () => { | ||||
|   uni.navigateTo({ | ||||
|     url: '/pages/chooseGroupAdmin/index' | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| const handleConfirm = async () => { | ||||
|   if (groupActiveIndex.value === 0) { | ||||
|      | ||||
|   }else if(groupActiveIndex.value === 1){ | ||||
|     const res = await createDepGroup() | ||||
|     if (res.code === 200) { | ||||
|       resetGroupInfo() | ||||
|       uni.navigateBack() | ||||
|     } | ||||
| 
 | ||||
|   }else { | ||||
| 
 | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const confirmBtnStatus = computed(() => { | ||||
|   let disabledT = false; | ||||
|   if (groupName.value === "" || !groupActiveIndex.value) { | ||||
|     return true; | ||||
|   } | ||||
|   switch (groupActiveIndex.value) { | ||||
|     case 0: | ||||
|       break; | ||||
|     case 1: | ||||
|       if (!depCheckedKeys.value.length) { | ||||
|         disabledT = true; | ||||
|       } | ||||
|       if (!groupAdmins.value.length) { | ||||
|         disabledT = true; | ||||
|       } | ||||
|       break; | ||||
|     case 2: | ||||
|       break; | ||||
|     default: | ||||
|       break; | ||||
|   } | ||||
|   return disabledT | ||||
| }) | ||||
| 
 | ||||
| onShow(() => { | ||||
|   depsNoExpanded.value = true; | ||||
| }) | ||||
| 
 | ||||
| 
 | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| @ -149,27 +257,34 @@ page { | ||||
|   margin-bottom: 40rpx; | ||||
| } | ||||
| .input-group { | ||||
|   height: 110rpx; | ||||
|   background-color: #fff; | ||||
|   padding-top: 38rpx; | ||||
|   padding-bottom: 32rpx; | ||||
|   padding-left: 32rpx; | ||||
|   padding-right: 40rpx; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
|   // display: flex; | ||||
|   // align-items: center; | ||||
|   // justify-content: space-between; | ||||
| } | ||||
| .input-item { | ||||
|   line-height: 54rpx; | ||||
|   line-height: 40rpx; | ||||
|   font-size: 28rpx; | ||||
|   color: #000; | ||||
|   font-weight: bold; | ||||
| } | ||||
| .input-box { | ||||
|   margin-left: 84rpx; | ||||
|   line-height: 40rpx; | ||||
|   width: 404rpx; | ||||
| 
 | ||||
| } | ||||
| .left-box { | ||||
|   margin-left: 290rpx; | ||||
|   display: flex; | ||||
|   align-items: center; | ||||
| } | ||||
| .btnBox { | ||||
|   :deep(uni-button[disabled="true"]) { | ||||
|     color: #bebebe !important; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -16,7 +16,7 @@ | ||||
|           </div> | ||||
|         </template> | ||||
|         <template v-slot:right> | ||||
|           <div class="mr-[48rpx]"> | ||||
|           <div class="mr-[48rpx] popoverBox"> | ||||
|             <tm-popover position="br" color="#333333" :width="260"> | ||||
|               <tm-image | ||||
|                 :width="41" | ||||
| @ -160,4 +160,9 @@ page { | ||||
|   background-color: #7C7C7C; | ||||
|   opacity: 0.6; | ||||
| } | ||||
| .popoverBox { | ||||
|   :deep(.popover-bcc){ | ||||
|     transform:translateX(20rpx) translateY(40rpx); | ||||
|   } | ||||
| } | ||||
| </style> | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								src/static/image/chatList/downDep.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 251 B | 
							
								
								
									
										
											BIN
										
									
								
								src/static/image/chatList/downDepDis.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 245 B | 
| Before Width: | Height: | Size: 40 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/static/image/chatList/zu6033@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 40 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/static/image/checkBox/circle@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 894 B | 
							
								
								
									
										
											BIN
										
									
								
								src/static/image/checkBox/zu6039@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 932 B | 
							
								
								
									
										
											BIN
										
									
								
								src/static/image/checkBox/zu6040@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 695 B | 
| @ -1,10 +1,11 @@ | ||||
| import {createGlobalState,useStorage} from '@vueuse/core' | ||||
| import {uniStorage} from "@/utils/uniStorage.js" | ||||
| import { userInfoApi } from "@/api/user"; | ||||
| 
 | ||||
| import {ref} from 'vue' | ||||
| export const useAuth = createGlobalState(() => { | ||||
|   // const token = useStorage('token', '', uniStorage)
 | ||||
|   const token = ref('79b5c732d96d2b27a48a99dfd4a5566c43aaa5796242e854ebe3ffc198d6876b9628e7b764d9af65ab5dbb2d517ced88170491b74b048c0ba827c0d3741462cb89dc59ed46653a449af837a8262941ca1430937103230a1e32a1715f569f3efdbe6f8cb8b7b8642bd679668081b9b08f693d1b5be6002d936ec51e1e3e0c4927de9e32ac99a109b326e5d2bda27ec87624bb416ec70d2a95a2e190feeba9f0d6bae8571b3dfe89c824712344759a8f2bff9d70747c52525cf6a5614f9c770bca461a9b9c247b6dca97bcf83bbaf99bb726752c4fe1e9a4aa7de5c4cf3e88a3e480801280d45cdc124f9d8221105d8529c7c268c1251c3477bff5c051043bf980d1ae32bb67035d540ba9325759be69093c01497a2ece6c54d7ba2c2ccd07bb67242ccd7b216049b2afefd767f7269fa45ebb81dff8bec485f539e91611a2908af4407a64e67e6a1cb533d30ae71e630f') | ||||
|   const token = ref('79b5c732d96d2b27a48a99dfd4a5566c43aaa5796242e854ebe3ffc198d6876b9628e7b764d9af65ab5dbb2d517ced88170491b74b048c0ba827c0d3741462cb89dc59ed46653a449af837a8262941ca1430937103230a1e32a1715f569f3efdbe6f8cb8b7b8642bd679668081b9b08f693d1b5be6002d936ec51e1e3e0c4927de9e32ac99a109b326e5d2bda27ec87624bb416ec70d2a95a2e190feeba9f0d6bae8571b3dfe89c824712344759a8f2bff9d70747c52525cf6a5614f9c770bca461a9b9c247b6dca97bcf83bbaf99bb726752c4fe1e9a4aa7de5c4cf3e88a3e480801280d45cdc124f9d8221105d8529c7c268c1251c3477bff5c051043bf9804c4f7096c5178cc01261bbedf04108b969333baf0bd5226c426de3591fc50b70691f2a5e275fa67ea3a7301df448929c13e4b71bf2182b6bfc270e21c142aa693844a0a175637f907e4438786a6e5e50') | ||||
|   const refreshToken = useStorage('refreshToken', '', uniStorage) | ||||
|   const userInfo = useStorage('userInfo', {}, uniStorage) | ||||
|   const leaderList = useStorage('leaderList', [], uniStorage) | ||||
| @ -20,25 +21,25 @@ export const useAuth = createGlobalState(() => { | ||||
|       userInfo.value=res.data | ||||
|     } | ||||
|   } | ||||
|   const getUserLeader=async ()=>{ | ||||
|     const data={ | ||||
|       departmentId:userInfo.value.PositionUsers?.map(x=>x.DepartmentId) | ||||
|     } | ||||
|   // const getUserLeader=async ()=>{
 | ||||
|   //   const data={
 | ||||
|   //     departmentId:userInfo.value.PositionUsers?.map(x=>x.DepartmentId)
 | ||||
|   //   }
 | ||||
| 
 | ||||
|     const res=  await userLeaderApi(data) | ||||
|     if (res.status===0){ | ||||
|       isLeader.value=!!res.data.departmentLeaders?.find((x) => { | ||||
|         return x.userID === userInfo.value.ID | ||||
|       }) | ||||
|       leaderList.value=res.data.departmentLeaders | ||||
|       myCompany.value=res.data.company | ||||
|     } | ||||
|   //   const res=  await userLeaderApi(data)
 | ||||
|   //   if (res.status===0){
 | ||||
|   //     isLeader.value=!!res.data.departmentLeaders?.find((x) => {
 | ||||
|   //       return x.userID === userInfo.value.ID
 | ||||
|   //     })
 | ||||
|   //     leaderList.value=res.data.departmentLeaders
 | ||||
|   //     myCompany.value=res.data.company
 | ||||
|   //   }
 | ||||
| 
 | ||||
|   } | ||||
|   // }
 | ||||
|   return { | ||||
|     leaderList, | ||||
|     myCompany, | ||||
|     getUserLeader, | ||||
|     // getUserLeader,
 | ||||
|     getUserInfo, | ||||
|     userInfo, | ||||
|     token, | ||||
|  | ||||
							
								
								
									
										81
									
								
								src/store/groupType/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,81 @@ | ||||
| import { defineStore } from 'pinia' | ||||
| import { useDialogueStore } from '@/store' | ||||
| import lodash from 'lodash' | ||||
| 
 | ||||
| import {ref} from 'vue' | ||||
| import {createGlobalState,useStorage} from '@vueuse/core' | ||||
| import {uniStorage} from "@/utils/uniStorage.js" | ||||
| import {departmentV2TreeMy,departmentV2AllPosition,groupCreateDept} from '@/api/deps/index.js' | ||||
| 
 | ||||
| export const useGroupTypeStore = createGlobalState(() => { | ||||
|   const groupName = ref('') | ||||
|   const groupActiveIndex = ref(-1) // 当前激活的分组索引
 | ||||
|   const depTreeMyList = ref([]) | ||||
|   const postTreeList = ref([]) | ||||
|   const crumbs = ref([]) | ||||
|   const crumbsIndex = ref(0) | ||||
|   const depCheckedKeys = ref([]) // 选中的部门keys
 | ||||
|   const groupAdmins = ref([]) // 选中的管理员
 | ||||
|    | ||||
|   // const halfCheckedKeys = ref([])
 | ||||
| 
 | ||||
|   const getDepsTreeMy= async()=>{ | ||||
|     const res = await departmentV2TreeMy() | ||||
|     if(res.status===0){ | ||||
|       depTreeMyList.value = res.data.nodes | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const getPositionsTree= async()=>{ | ||||
|     const res = await departmentV2AllPosition() | ||||
|     if(res.status===0){ | ||||
|       postTreeList.value = res.data.nodes | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const resetGroupInfo = () => { | ||||
|     groupName.value = ''; | ||||
|     groupActiveIndex.value = -1; | ||||
|     depTreeMyList.value = []; | ||||
|     postTreeList.value = []; | ||||
|     crumbs.value = []; | ||||
|     crumbsIndex.value = 0 | ||||
|     depCheckedKeys.value = [] | ||||
|     groupAdmins.value = [] | ||||
|   } | ||||
| 
 | ||||
|   const createDepGroup = async (param) => { | ||||
|     const res = await groupCreateDept({ | ||||
|       name:groupName.value, | ||||
|       deptInfos:depCheckedKeys.value.map(v=>{ | ||||
|         return { | ||||
|           dept_id:v.ID, | ||||
|           dept_name:v.name | ||||
|         } | ||||
|       }), | ||||
|       positionInfos:groupAdmins.value.map(v=>{ | ||||
|         return { | ||||
|           position_id:v.ID, | ||||
|           position_name:v.name | ||||
|         } | ||||
|       }) | ||||
|     }) | ||||
|     return res; | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|   return { | ||||
|     groupName, | ||||
|     groupActiveIndex, | ||||
|     depTreeMyList, | ||||
|     postTreeList, | ||||
|     getDepsTreeMy, | ||||
|     getPositionsTree, | ||||
|     crumbs, | ||||
|     crumbsIndex, | ||||
|     depCheckedKeys, | ||||
|     groupAdmins, | ||||
|     resetGroupInfo, | ||||
|     createDepGroup, | ||||
|   } | ||||
| }) | ||||
| @ -2,6 +2,7 @@ import { defineStore } from 'pinia' | ||||
| import { ServeGetUserSetting } from '@/api/user/index.js' | ||||
| // import { ServeFindFriendApplyNum } from '@/api/contact'
 | ||||
| import { ServeGroupApplyUnread } from '@/api/group' | ||||
| import {useAuth} from '@/store/auth' | ||||
| // import { delAccessToken } from '@/utils/auth'
 | ||||
| // import { storage } from '@/utils/storage'
 | ||||
| 
 | ||||
| @ -20,7 +21,9 @@ export const useUserStore = defineStore('chatUser', { | ||||
|       online: false, // 在线状态
 | ||||
|       isQiye: false, | ||||
|       isContactApply: false, | ||||
|       isGroupApply: false | ||||
|       isGroupApply: false, | ||||
|       deps:[], | ||||
|       positions:[], | ||||
|     } | ||||
|   }, | ||||
|   getters: {}, | ||||
| @ -40,6 +43,9 @@ export const useUserStore = defineStore('chatUser', { | ||||
|     loadSetting() { | ||||
|       ServeGetUserSetting().then(({ code, data }) => { | ||||
|         if (code == 200) { | ||||
|           const {getUserInfo,userInfo} = useAuth() | ||||
|           userInfo.value.ID = data.user_info.erp_user_id | ||||
|           getUserInfo() | ||||
|           this.nickname = data.user_info.nickname | ||||
|           this.uid = data.user_info.uid | ||||
|           this.avatar = data.user_info.avatar | ||||
| @ -49,6 +55,8 @@ export const useUserStore = defineStore('chatUser', { | ||||
|           this.email = data.user_info.email || '' | ||||
|           this.motto = data.user_info.motto | ||||
|           this.isQiye = data.user_info.is_qiye || false | ||||
|           this.deps = data.user_info.user_depts || [] | ||||
|           this.positions = data.user_info.user_positions || [] | ||||
| 
 | ||||
|           // storage.set('user_info', data)
 | ||||
|         } | ||||
|  | ||||
| @ -32,7 +32,8 @@ | ||||
| 				:class="[ | ||||
| 					isNvue ? 'fixed' : 'absolute', | ||||
| 					props.position == 'tc' || props.position == 'tl' || props.position == 'tr' ? 'popover-tcc' : '', | ||||
| 					props.position == 'bc' || props.position == 'bl' || props.position == 'br' ? 'popover-bcc' : '' | ||||
| 					props.position == 'bc' || props.position == 'bl' || props.position == 'br' ? 'popover-bcc' : '', | ||||
| 					 | ||||
| 				]" | ||||
| 				:style="[ | ||||
| 					isNvue && props.position == 'tc' | ||||
|  | ||||