index.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. <template>
  2. <div>
  3. <div class="app-container">
  4. <div v-show="!dialog.visible && !detailState.show && !addState.show">
  5. <el-row :gutter="20">
  6. <el-col :lg="4" :xs="24" style="">
  7. <el-input v-model="deptName" placeholder="请输入部门名称" prefix-icon="Search" clearable />
  8. <el-tree
  9. ref="deptTreeRef"
  10. class="mt-2"
  11. node-key="id"
  12. :data="deptOptions"
  13. :props="{ label: 'label', children: 'children' }"
  14. :expand-on-click-node="false"
  15. :filter-node-method="filterNode"
  16. highlight-current
  17. default-expand-all
  18. @node-click="handleNodeClick"
  19. />
  20. </el-col>
  21. <el-col :lg="20" :xs="24">
  22. <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
  23. <div v-show="showSearch" class="mb-[10px]">
  24. <h1>茂名市责任人列表</h1>
  25. <el-form ref="queryFormRef" :model="queryParams" :inline="true">
  26. <el-form-item>
  27. <el-button type="primary">批量导出</el-button>
  28. <el-button type="primary" @click="handleUpload">批量导入</el-button>
  29. <el-button type="primary" @click="handleAdd">新建责任人</el-button>
  30. <el-button type="primary" @click="handleShowUploadDetails">查看导入情况</el-button>
  31. </el-form-item>
  32. <el-form-item label="姓名:" prop="Name" label-width="auto">
  33. <el-input v-model="queryParams.Name" placeholder="请输入联系人" clearable @keyup.enter="handleQuery" />
  34. </el-form-item>
  35. <el-form-item>
  36. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  37. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  38. </el-form-item>
  39. </el-form>
  40. </div>
  41. </transition>
  42. <el-table v-loading="loading" :data="userList" @selection-change="handleSelectionChange">
  43. <el-table-column type="selection" width="50" align="center" />
  44. <el-table-column v-if="columns[0].visible" key="userId" label="用户编号" align="center" prop="userId" />
  45. <el-table-column
  46. v-if="columns[1].visible"
  47. key="name"
  48. label="姓名"
  49. align="center"
  50. prop="name"
  51. :show-overflow-tooltip="true"
  52. width="120"
  53. />
  54. <el-table-column
  55. v-if="columns[2].visible"
  56. key="area_code"
  57. label="行政区划"
  58. align="center"
  59. prop="area_code"
  60. :show-overflow-tooltip="true"
  61. />
  62. <el-table-column
  63. v-if="columns[3].visible"
  64. key="unit_name"
  65. label="所属单位"
  66. align="center"
  67. prop="unit_name"
  68. :show-overflow-tooltip="true"
  69. />
  70. <el-table-column v-if="columns[4].visible" key="position" label="职务" align="center" prop="position" width="120" />
  71. <el-table-column v-if="columns[5].visible" key="phone" label="电话号码" align="center" prop="phone" width="120" />
  72. <el-table-column v-if="columns[6].visible" label="责任类型" align="center" prop="type_parent_list" width="160">
  73. <template #default="scope">
  74. <span v-for="(typeParent, index) in scope.row.type_parent_list" :key="index">
  75. {{ typeParent.type_parent }}
  76. </span>
  77. </template>
  78. </el-table-column>
  79. <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
  80. <template #default="scope">
  81. <el-text v-hasPermi="['system:menu:resetPwd']" class="common-btn-text-primary" @click="handleView(scope.row)">查看详情</el-text>
  82. <el-text v-hasPermi="['system:menu:edit']" class="common-btn-text-primary" @click="handleUpdate(scope.row)">修改</el-text>
  83. <el-text v-hasPermi="['system:menu:remove']" class="common-btn-text-danger" @click="handleDelete(scope.row)">删除</el-text>
  84. </template>
  85. </el-table-column>
  86. </el-table>
  87. <pagination
  88. v-show="total > 0"
  89. v-model:page="queryParams.page"
  90. v-model:limit="queryParams.pageSize"
  91. :total="total"
  92. @pagination="getList"
  93. />
  94. </el-col>
  95. </el-row>
  96. </div>
  97. <DataImport v-model="uploadShow" url="" file-name="责任人信息批量导入模板" steps-text="责任人信息" />
  98. <DataImportDetail v-model="showUploadDetails" />
  99. <detail v-if="detailState.show" :id="detailState.id" @close="handledetailClose"></detail>
  100. <add v-if="addState.show" :id="addState.id" @close="handleAddClose"></add>
  101. </div>
  102. </div>
  103. </template>
  104. <script setup lang="ts">
  105. import api from '@/api/system/user';
  106. import { UserForm, UserQuery, UserVO } from '@/api/system/user/types';
  107. import { deleteData, getTableList } from '@/api/PreventionResponsible/index';
  108. import { DeptVO } from '@/api/system/dept/types';
  109. import { RoleVO } from '@/api/system/role/types';
  110. import { PostQuery, PostVO } from '@/api/system/post/types';
  111. import { treeselect } from '@/api/system/dept';
  112. import { globalHeaders } from '@/utils/request';
  113. import { to } from 'await-to-js';
  114. import { optionselect } from '@/api/system/post';
  115. import detail from '@/views/setting/PreventionResponsible/detail.vue';
  116. import { reactive, ref } from 'vue';
  117. import { deleteMaterialRoot } from '@/api/comprehensiveGuarantee/materialReserveManagement/warehouseManagement';
  118. const router = useRouter();
  119. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  120. const userList = ref<UserVO[]>();
  121. const loading = ref(true);
  122. const showSearch = ref(true);
  123. const ids = ref<Array<number | string>>([]);
  124. const single = ref(true);
  125. const multiple = ref(true);
  126. const total = ref(0);
  127. const dateRange = ref<[DateModelType, DateModelType]>(['', '']);
  128. const deptName = ref('');
  129. const deptOptions = ref<DeptVO[]>([]);
  130. const initPassword = ref<string>('');
  131. const postOptions = ref<PostVO[]>([]);
  132. const roleOptions = ref<RoleVO[]>([]);
  133. // 列显隐信息
  134. const columns = ref<FieldOption[]>([
  135. { key: 0, label: `用户编号`, visible: false, children: [] },
  136. { key: 1, label: `姓名`, visible: true, children: [] },
  137. { key: 2, label: `行政区划`, visible: true, children: [] },
  138. { key: 3, label: `所属单位`, visible: true, children: [] },
  139. { key: 4, label: `职务`, visible: true, children: [] },
  140. { key: 5, label: `电话号码`, visible: true, children: [] },
  141. { key: 6, label: `责任类型`, visible: true, children: [] }
  142. ]);
  143. const deptTreeRef = ref<ElTreeInstance>();
  144. const queryFormRef = ref<ElFormInstance>();
  145. const userFormRef = ref<ElFormInstance>();
  146. const dialog = reactive<DialogOption>({
  147. visible: false,
  148. title: ''
  149. });
  150. const initFormData: UserForm = {
  151. id: undefined,
  152. name: '',
  153. area_code: '',
  154. unit_name: '',
  155. position: '',
  156. phone: '',
  157. type_parent_list: []
  158. };
  159. const initData: PageData<UserForm, UserQuery> = {
  160. form: { ...initFormData },
  161. queryParams: {
  162. page: 1,
  163. pageSize: 10,
  164. Name: ''
  165. },
  166. rules: {
  167. userName: [
  168. { required: true, message: '用户名称不能为空', trigger: 'blur' },
  169. {
  170. min: 2,
  171. max: 20,
  172. message: '用户名称长度必须介于 2 和 20 之间',
  173. trigger: 'blur'
  174. }
  175. ],
  176. nickName: [{ required: true, message: '用户昵称不能为空', trigger: 'blur' }],
  177. password: [
  178. { required: true, message: '用户密码不能为空', trigger: 'blur' },
  179. {
  180. min: 5,
  181. max: 20,
  182. message: '用户密码长度必须介于 5 和 20 之间',
  183. trigger: 'blur'
  184. },
  185. { pattern: /^[^<>"'|\\]+$/, message: '不能包含非法字符:< > " \' \\\ |', trigger: 'blur' }
  186. ],
  187. email: [
  188. {
  189. type: 'email',
  190. message: '请输入正确的邮箱地址',
  191. trigger: ['blur', 'change']
  192. }
  193. ],
  194. phonenumber: [
  195. {
  196. pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
  197. message: '请输入正确的手机号码',
  198. trigger: 'blur'
  199. }
  200. ],
  201. roleIds: [{ required: true, message: '用户角色不能为空', trigger: 'blur' }]
  202. }
  203. };
  204. const data = reactive<PageData<UserForm, UserQuery>>(initData);
  205. const { queryParams, form, rules } = toRefs<PageData<UserForm, UserQuery>>(data);
  206. /** 通过条件过滤节点 */
  207. const filterNode = (value: string, data: any) => {
  208. if (!value) return true;
  209. return data.label.indexOf(value) !== -1;
  210. };
  211. /** 根据名称筛选部门树 */
  212. watchEffect(
  213. () => {
  214. deptTreeRef.value?.filter(deptName.value);
  215. },
  216. {
  217. flush: 'post' // watchEffect会在DOM挂载或者更新之前就会触发,此属性控制在DOM元素更新后运行
  218. }
  219. );
  220. /** 查询部门下拉树结构 */
  221. const getTreeSelect = async () => {
  222. const res = await api.deptTreeSelect();
  223. deptOptions.value = res.data;
  224. };
  225. /** 查询用户列表 */
  226. const getList = async () => {
  227. loading.value = true;
  228. getTableList(queryParams.value)
  229. .then((res) => {
  230. userList.value = res.data;
  231. total.value = res.total;
  232. })
  233. .finally(() => {
  234. loading.value = false;
  235. });
  236. };
  237. /** 节点单击事件 */
  238. const handleNodeClick = (data: DeptVO) => {
  239. queryParams.value.deptId = data.id;
  240. handleQuery();
  241. };
  242. /** 搜索按钮操作 */
  243. const handleQuery = () => {
  244. queryParams.value.pageNum = 1;
  245. getList();
  246. };
  247. /** 重置按钮操作 */
  248. const resetQuery = () => {
  249. queryFormRef.value?.resetFields();
  250. queryParams.value.pageNum = 1;
  251. deptTreeRef.value?.setCurrentKey(undefined);
  252. handleQuery();
  253. };
  254. /** 删除按钮操作 */
  255. const handleDelete = (row?: UserVO) => {
  256. const userIds = row?.id || ids.value;
  257. proxy?.$modal.confirm('是否确认删除用户编号为"' + userIds + '"的数据项?').then(() => {
  258. deleteData(userIds);
  259. getList();
  260. proxy?.$modal.msgSuccess('删除成功');
  261. });
  262. };
  263. let detailState = reactive({
  264. show: false,
  265. id: ''
  266. });
  267. /** 查看详情按钮操作 */
  268. const handleView = async (row: UserVO) => {
  269. detailState.show = true;
  270. detailState.id = row.id;
  271. };
  272. const handledetailClose = () => {
  273. detailState.show = false;
  274. };
  275. let addState = reactive({
  276. show: false,
  277. id: ''
  278. });
  279. // 新建责任人按钮操作
  280. const handleAddClose = () => {
  281. addState.show = false;
  282. };
  283. const handleAdd = () => {
  284. addState.show = true;
  285. addState.id = row.id;
  286. };
  287. /** 选择条数 */
  288. const handleSelectionChange = (selection: UserVO[]) => {
  289. ids.value = selection.map((item) => item.userId);
  290. single.value = selection.length != 1;
  291. multiple.value = !selection.length;
  292. };
  293. /** 导出按钮操作 */
  294. const handleExport = () => {
  295. proxy?.download(
  296. 'system/user/export',
  297. {
  298. ...queryParams.value
  299. },
  300. `user_${new Date().getTime()}.xlsx`
  301. );
  302. };
  303. /** 初始化部门数据 */
  304. const initTreeData = async () => {
  305. // 判断部门的数据是否存在,存在不获取,不存在则获取
  306. if (deptOptions.value === undefined) {
  307. const { data } = await treeselect();
  308. deptOptions.value = data;
  309. }
  310. };
  311. /** 重置操作表单 */
  312. const reset = () => {
  313. form.value = { ...initFormData };
  314. userFormRef.value?.resetFields();
  315. };
  316. /** 取消按钮 */
  317. const cancel = () => {
  318. dialog.visible = false;
  319. reset();
  320. };
  321. /** 新增按钮操作 */
  322. // const handleAdd = async () => {
  323. // reset();
  324. // const { data } = await api.getUser();
  325. // dialog.visible = true;
  326. // dialog.title = '新增用户';
  327. // await initTreeData();
  328. // postOptions.value = data.posts;
  329. // roleOptions.value = data.roles;
  330. // // form.value.password = initPassword.value.toString();
  331. // };
  332. /** 修改按钮操作 */
  333. const handleUpdate = async (row?: UserForm) => {
  334. reset();
  335. const userId = row?.userId || ids.value[0];
  336. const { data } = await api.getUser(userId);
  337. dialog.visible = true;
  338. dialog.title = '修改用户';
  339. await initTreeData();
  340. Object.assign(form.value, data.user);
  341. postOptions.value = data.posts;
  342. roleOptions.value = data.roles;
  343. form.value.postIds = data.postIds;
  344. form.value.roleIds = data.roleIds;
  345. form.value.password = '';
  346. };
  347. /**
  348. * 重置表单
  349. */
  350. const resetForm = () => {
  351. userFormRef.value?.resetFields();
  352. userFormRef.value?.clearValidate();
  353. form.value.id = undefined;
  354. form.value.status = '1';
  355. };
  356. // 导入
  357. let uploadShow = ref(false);
  358. const handleUpload = () => {
  359. uploadShow.value = true;
  360. };
  361. let showUploadDetails = ref(false);
  362. const handleShowUploadDetails = () => {
  363. showUploadDetails.value = true;
  364. };
  365. onMounted(() => {
  366. getTreeSelect(); // 初始化部门数据
  367. getList(); // 初始化列表数据
  368. proxy?.getConfigKey('sys.user.initPassword').then((response) => {
  369. initPassword.value = response.data;
  370. });
  371. });
  372. async function handleDeptChange(value: number | string) {
  373. const response = await optionselect(value);
  374. postOptions.value = response.data;
  375. form.value.postIds = [];
  376. }
  377. </script>
  378. <style scoped lang="scss"></style>