index.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. <template>
  2. <div>
  3. <el-dialog v-model="userDialog.visible.value" :title="userDialog.title.value" width="80%" append-to-body>
  4. <el-row :gutter="20">
  5. <!-- 部门树 -->
  6. <el-col :lg="4" :xs="24" style="">
  7. <el-card shadow="hover">
  8. <el-input v-model="deptName" placeholder="请输入部门名称" prefix-icon="Search" clearable />
  9. <el-tree
  10. ref="deptTreeRef"
  11. class="mt-2"
  12. node-key="id"
  13. :data="deptOptions"
  14. :props="{ label: 'label', children: 'children' }"
  15. :expand-on-click-node="false"
  16. :filter-node-method="filterNode"
  17. highlight-current
  18. default-expand-all
  19. @node-click="handleNodeClick"
  20. />
  21. </el-card>
  22. </el-col>
  23. <el-col :lg="20" :xs="24">
  24. <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
  25. <div v-show="showSearch" class="mb-[10px]">
  26. <el-card shadow="hover">
  27. <el-form ref="queryFormRef" :model="queryParams" :inline="true" label-width="68px">
  28. <el-form-item label="用户名称" prop="userName">
  29. <el-input v-model="queryParams.userName" placeholder="请输入用户名称" clearable style="width: 200px" @keyup.enter="handleQuery" />
  30. </el-form-item>
  31. <el-form-item label="手机号码" prop="phonenumber">
  32. <el-input
  33. v-model="queryParams.phonenumber"
  34. placeholder="请输入手机号码"
  35. clearable
  36. style="width: 200px"
  37. @keyup.enter="handleQuery"
  38. />
  39. </el-form-item>
  40. <el-form-item>
  41. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  42. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  43. </el-form-item>
  44. </el-form>
  45. </el-card>
  46. </div>
  47. </transition>
  48. <el-card shadow="hover">
  49. <template v-if="prop.multiple" #header>
  50. <el-tag v-for="user in selectUserList" :key="user.userId" closable style="margin: 2px" @close="handleCloseTag(user)">
  51. {{ user.userName }}
  52. </el-tag>
  53. </template>
  54. <vxe-table
  55. ref="tableRef"
  56. height="400px"
  57. border
  58. show-overflow
  59. :data="userList"
  60. :loading="loading"
  61. :row-config="{ keyField: 'userId', isHover: true }"
  62. :checkbox-config="{ reserve: true, trigger: 'row', highlight: true, showHeader: prop.multiple }"
  63. @checkbox-all="handleCheckboxAll"
  64. @checkbox-change="handleCheckboxChange"
  65. >
  66. <vxe-column type="checkbox" width="50" align="center" />
  67. <vxe-column key="userId" title="用户编号" align="center" field="userId" />
  68. <vxe-column key="userName" title="用户名称" align="center" field="userName" />
  69. <vxe-column key="nickName" title="用户昵称" align="center" field="nickName" />
  70. <vxe-column key="deptName" title="部门" align="center" field="deptName" />
  71. <vxe-column key="phonenumber" title="手机号码" align="center" field="phonenumber" width="120" />
  72. <vxe-column key="status" title="状态" align="center">
  73. <template #default="scope">
  74. <dict-tag :options="sys_normal_disable" :value="scope.row.status"></dict-tag>
  75. </template>
  76. </vxe-column>
  77. <vxe-column title="创建时间" align="center" width="160">
  78. <template #default="scope">
  79. <span>{{ scope.row.createTime }}</span>
  80. </template>
  81. </vxe-column>
  82. </vxe-table>
  83. <pagination
  84. v-show="total > 0"
  85. v-model:page="queryParams.pageNum"
  86. v-model:limit="queryParams.pageSize"
  87. :total="total"
  88. @pagination="pageList"
  89. />
  90. </el-card>
  91. </el-col>
  92. </el-row>
  93. <template #footer>
  94. <el-button @click="close">取消</el-button>
  95. <el-button type="primary" @click="confirm">确定</el-button>
  96. </template>
  97. </el-dialog>
  98. </div>
  99. </template>
  100. <script setup lang="ts">
  101. import api from '@/api/system/user';
  102. import { UserQuery, UserVO } from '@/api/system/user/types';
  103. import { DeptVO } from '@/api/system/dept/types';
  104. import { VxeTableInstance } from 'vxe-table';
  105. import useDialog from '@/hooks/useDialog';
  106. interface PropType {
  107. modelValue?: UserVO[] | UserVO | undefined;
  108. multiple?: boolean;
  109. data?: string | number | (string | number)[];
  110. }
  111. const prop = withDefaults(defineProps<PropType>(), {
  112. multiple: true,
  113. modelValue: undefined,
  114. data: undefined
  115. });
  116. const emit = defineEmits(['update:modelValue', 'confirmCallBack']);
  117. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  118. const { sys_normal_disable } = toRefs<any>(proxy?.useDict('sys_normal_disable'));
  119. const userList = ref<UserVO[]>();
  120. const loading = ref(true);
  121. const showSearch = ref(true);
  122. const total = ref(0);
  123. const dateRange = ref<[DateModelType, DateModelType]>(['', '']);
  124. const deptName = ref('');
  125. const deptOptions = ref<DeptVO[]>([]);
  126. const selectUserList = ref<UserVO[]>([]);
  127. const deptTreeRef = ref<ElTreeInstance>();
  128. const queryFormRef = ref<ElFormInstance>();
  129. const tableRef = ref<VxeTableInstance<UserVO>>();
  130. const userDialog = useDialog({
  131. title: '用户选择'
  132. });
  133. const queryParams = ref<UserQuery>({
  134. pageNum: 1,
  135. pageSize: 10,
  136. userName: '',
  137. phonenumber: '',
  138. status: '',
  139. deptId: '',
  140. roleId: ''
  141. });
  142. const defaultSelectUserIds = computed(() => computedIds(prop.data));
  143. /** 根据名称筛选部门树 */
  144. watchEffect(
  145. () => {
  146. deptTreeRef.value?.filter(deptName.value);
  147. },
  148. {
  149. flush: 'post' // watchEffect会在DOM挂载或者更新之前就会触发,此属性控制在DOM元素更新后运行
  150. }
  151. );
  152. const confirm = () => {
  153. emit('update:modelValue', selectUserList.value);
  154. emit('confirmCallBack', selectUserList.value);
  155. userDialog.closeDialog();
  156. };
  157. const computedIds = (data) => {
  158. if (data instanceof Array) {
  159. return [...data];
  160. } else if (typeof data === 'string') {
  161. return data.split(',');
  162. } else if (typeof data === 'number') {
  163. return [data];
  164. } else {
  165. console.warn('<UserSelect> The data type of data should be array or string or number, but I received other');
  166. return [];
  167. }
  168. };
  169. /** 通过条件过滤节点 */
  170. const filterNode = (value: string, data: any) => {
  171. if (!value) return true;
  172. return data.label.indexOf(value) !== -1;
  173. };
  174. /** 查询部门下拉树结构 */
  175. const getTreeSelect = async () => {
  176. const res = await api.deptTreeSelect();
  177. deptOptions.value = res.data;
  178. };
  179. /** 查询用户列表 */
  180. const getList = async () => {
  181. loading.value = true;
  182. const res = await api.listUser(proxy?.addDateRange(queryParams.value, dateRange.value));
  183. loading.value = false;
  184. userList.value = res.rows;
  185. total.value = res.total;
  186. };
  187. const pageList = async () => {
  188. await getList();
  189. const users = userList.value.filter((item) => {
  190. return selectUserList.value.some((user) => user.userId === item.userId);
  191. });
  192. await tableRef.value.setCheckboxRow(users, true);
  193. };
  194. /** 节点单击事件 */
  195. const handleNodeClick = (data: DeptVO) => {
  196. queryParams.value.deptId = data.id;
  197. handleQuery();
  198. };
  199. /** 搜索按钮操作 */
  200. const handleQuery = () => {
  201. queryParams.value.pageNum = 1;
  202. getList();
  203. };
  204. /** 重置按钮操作 */
  205. const resetQuery = () => {
  206. dateRange.value = ['', ''];
  207. queryFormRef.value?.resetFields();
  208. queryParams.value.pageNum = 1;
  209. queryParams.value.deptId = undefined;
  210. deptTreeRef.value?.setCurrentKey(undefined);
  211. handleQuery();
  212. };
  213. const handleCheckboxChange = (checked) => {
  214. if (!prop.multiple && checked.checked) {
  215. tableRef.value.setCheckboxRow(selectUserList.value, false);
  216. selectUserList.value = [];
  217. }
  218. const row = checked.row;
  219. if (checked.checked) {
  220. selectUserList.value.push(row);
  221. } else {
  222. selectUserList.value = selectUserList.value.filter((item) => {
  223. return item.userId !== row.userId;
  224. });
  225. }
  226. };
  227. const handleCheckboxAll = (checked) => {
  228. const rows = userList.value;
  229. if (checked.checked) {
  230. rows.forEach((row) => {
  231. if (!selectUserList.value.some((item) => item.userId === row.userId)) {
  232. selectUserList.value.push(row);
  233. }
  234. });
  235. } else {
  236. selectUserList.value = selectUserList.value.filter((item) => {
  237. return !rows.some((row) => row.userId === item.userId);
  238. });
  239. }
  240. };
  241. const handleCloseTag = (user: UserVO) => {
  242. const userId = user.userId;
  243. // 使用split删除用户
  244. const index = selectUserList.value.findIndex((item) => item.userId === userId);
  245. const rows = selectUserList.value[index];
  246. tableRef.value?.setCheckboxRow(rows, false);
  247. selectUserList.value.splice(index, 1);
  248. };
  249. const initSelectUser = async () => {
  250. if (defaultSelectUserIds.value.length > 0) {
  251. const { data } = await api.optionSelect(defaultSelectUserIds.value);
  252. selectUserList.value = data;
  253. const users = userList.value.filter((item) => {
  254. return defaultSelectUserIds.value.includes(String(item.userId));
  255. });
  256. await nextTick(() => {
  257. tableRef.value.setCheckboxRow(users, true);
  258. });
  259. }
  260. };
  261. const close = () => {
  262. userDialog.closeDialog();
  263. };
  264. watch(
  265. () => userDialog.visible.value,
  266. (newValue: boolean) => {
  267. console.log(selectUserList.value)
  268. if (newValue) {
  269. initSelectUser();
  270. } else {
  271. tableRef.value.clearCheckboxReserve();
  272. tableRef.value.clearCheckboxRow();
  273. resetQuery();
  274. selectUserList.value = [];
  275. }
  276. }
  277. );
  278. onMounted(() => {
  279. getTreeSelect(); // 初始化部门数据
  280. getList(); // 初始化列表数据
  281. });
  282. defineExpose({
  283. open: userDialog.openDialog,
  284. close: userDialog.closeDialog
  285. });
  286. </script>
  287. <style lang="scss" scoped></style>