editUser.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. <template>
  2. <div class="sys-user-container">
  3. <el-dialog v-model="state.isShowDialog" draggable :close-on-click-modal="false" width="700px">
  4. <template #header>
  5. <div style="color: #fff">
  6. <el-icon size="16" style="margin-right: 3px; display: inline; vertical-align: middle"> <ele-Edit /> </el-icon>
  7. <span>{{ props.title }}</span>
  8. </div>
  9. </template>
  10. <el-tabs v-loading="state.loading" v-model="state.selectedTabName">
  11. <el-tab-pane label="基础信息" style="height: 550px; overflow-y: auto; overflow-x: hidden">
  12. <el-form :model="state.ruleForm" ref="ruleFormRef" label-width="auto">
  13. <el-row :gutter="35">
  14. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  15. <el-form-item label="账号名称" prop="account" :rules="[{ required: true, message: '账号名称不能为空', trigger: 'blur' }]">
  16. <el-input v-model="state.ruleForm.account" placeholder="账号名称" clearable />
  17. </el-form-item>
  18. </el-col>
  19. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  20. <el-form-item label="昵称">
  21. <el-input v-model="state.ruleForm.nickName" placeholder="昵称" clearable />
  22. </el-form-item>
  23. </el-col>
  24. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  25. <el-form-item label="手机号码" prop="phone" :rules="[{ required: true, message: '手机号码不能为空', trigger: 'blur' }]">
  26. <el-input v-model="state.ruleForm.phone" placeholder="手机号码" clearable />
  27. </el-form-item>
  28. </el-col>
  29. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  30. <el-form-item label="真实姓名" prop="realName" :rules="[{ required: true, message: '真实姓名不能为空', trigger: 'blur' }]">
  31. <el-input v-model="state.ruleForm.realName" placeholder="真实姓名" clearable />
  32. </el-form-item>
  33. </el-col>
  34. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  35. <el-form-item label="角色集合" prop="roleIdList" :rules="[{ required: true, message: '角色集合不能为空', trigger: 'blur' }]">
  36. <el-select v-model="state.ruleForm.roleIdList" multiple value-key="id" clearable placeholder="角色集合" collapse-tags collapse-tags-tooltip class="w100" filterable>
  37. <el-option v-for="item in state.roleData" :key="item.id" :label="item.name" :value="item.id" />
  38. </el-select>
  39. </el-form-item>
  40. </el-col>
  41. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  42. <el-form-item label="账号类型" prop="accountType" :rules="[{ required: true, message: '账号类型不能为空', trigger: 'blur' }]">
  43. <g-sys-dict
  44. v-model="state.ruleForm.accountType"
  45. :on-item-filter="(data: any) => data.name != 'SuperAdmin' && (data.name == 'SysAdmin' ? [888, 999].includes(userInfos.accountType) : true)"
  46. code="AccountTypeEnum"
  47. render-as="select" />
  48. </el-form-item>
  49. </el-col>
  50. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  51. <el-form-item label="邮箱">
  52. <el-input v-model="state.ruleForm.email" placeholder="邮箱" clearable />
  53. </el-form-item>
  54. </el-col>
  55. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb5">
  56. <el-form-item label="排序">
  57. <el-input-number v-model="state.ruleForm.orderNo" placeholder="排序" class="w100" />
  58. </el-form-item>
  59. </el-col>
  60. <el-divider border-style="dashed" content-position="center">
  61. <div style="color: #b1b3b8">机构组织</div>
  62. </el-divider>
  63. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  64. <el-form-item label="所属机构" prop="orgId" :rules="[{ required: true, message: '所属机构不能为空', trigger: 'blur' }]">
  65. <el-cascader :options="props.orgData" :props="cascaderProps" placeholder="所属机构" clearable filterable class="w100" v-model="state.ruleForm.orgId">
  66. <template #default="{ node, data }">
  67. <span>{{ data.name }}</span>
  68. <span v-if="!node.isLeaf"> ({{ data.children.length }}) </span>
  69. </template>
  70. </el-cascader>
  71. </el-form-item>
  72. </el-col>
  73. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  74. <el-form-item label="职位" prop="posId" :rules="[{ required: true, message: '职位名称不能为空', trigger: 'blur' }]">
  75. <el-select v-model="state.ruleForm.posId" placeholder="职位" class="w100">
  76. <el-option v-for="d in state.posData" :key="d.id" :label="d.name" :value="d.id" />
  77. </el-select>
  78. </el-form-item>
  79. </el-col>
  80. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  81. <el-form-item label="工号">
  82. <el-input v-model="state.ruleForm.jobNum" placeholder="工号" clearable />
  83. </el-form-item>
  84. </el-col>
  85. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  86. <el-form-item label="域账号">
  87. <el-input v-model="state.ruleForm.domainAccount" placeholder="域账号" clearable />
  88. </el-form-item>
  89. </el-col>
  90. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  91. <el-form-item label="入职日期">
  92. <el-date-picker v-model="state.ruleForm.joinDate" type="date" placeholder="入职日期" format="YYYY-MM-DD" value-format="YYYY-MM-DD" class="w100" />
  93. </el-form-item>
  94. </el-col>
  95. <el-divider border-style="dashed" content-position="center">
  96. <div style="color: #b1b3b8">附属机构</div>
  97. </el-divider>
  98. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  99. <el-button icon="ele-Plus" type="primary" plain @click="addExtOrgRow"> 增加附属机构 </el-button>
  100. <span style="font-size: 12px; color: gray; padding-left: 5px"> 具有相应组织机构的数据权限 </span>
  101. </el-col>
  102. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  103. <template v-if="state.ruleForm.extOrgIdList != undefined && state.ruleForm.extOrgIdList.length > 0">
  104. <el-row :gutter="35" v-for="(v, k) in state.ruleForm.extOrgIdList" :key="k">
  105. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  106. <el-form-item label="机构" :prop="`extOrgIdList[${k}].orgId`" :rules="[{ required: true, message: `机构不能为空`, trigger: 'blur' }]">
  107. <template #label>
  108. <el-button icon="ele-Delete" type="danger" circle plain size="small" @click="deleteExtOrgRow(k)" />
  109. <span class="ml5">机构</span>
  110. </template>
  111. <el-cascader :options="props.orgData" :props="cascaderProps" placeholder="机构组织" clearable filterable class="w100" v-model="state.ruleForm.extOrgIdList[k].orgId">
  112. <template #default="{ node, data }">
  113. <span>{{ data.name }}</span>
  114. <span v-if="!node.isLeaf"> ({{ data.children.length }}) </span>
  115. </template>
  116. </el-cascader>
  117. </el-form-item>
  118. </el-col>
  119. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  120. <el-form-item label="职位" :prop="`extOrgIdList[${k}].posId`" :rules="[{ required: true, message: `职位不能为空`, trigger: 'blur' }]">
  121. <el-select v-model="state.ruleForm.extOrgIdList[k].posId" placeholder="职位名称" class="w100">
  122. <el-option v-for="d in state.posData" :key="d.id" :label="d.name" :value="d.id" />
  123. </el-select>
  124. </el-form-item>
  125. </el-col>
  126. </el-row>
  127. </template>
  128. <el-empty :image-size="50" description="空数据" v-else></el-empty>
  129. </el-col>
  130. </el-row>
  131. </el-form>
  132. </el-tab-pane>
  133. <el-tab-pane label="档案信息" style="height: 550px; overflow-y: auto; overflow-x: hidden">
  134. <el-form :model="state.ruleForm" label-width="auto">
  135. <el-row :gutter="35">
  136. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  137. <el-form-item label="证件类型" prop="cardType">
  138. <g-sys-dict v-model="state.ruleForm.cardType" code="CardTypeEnum" render-as="select" placeholder="证件类型" class="w100" />
  139. </el-form-item>
  140. </el-col>
  141. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  142. <el-form-item label="证件号码">
  143. <el-input v-model="state.ruleForm.idCardNum" placeholder="证件号码" clearable />
  144. </el-form-item>
  145. </el-col>
  146. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  147. <el-form-item label="出生日期" prop="birthday">
  148. <el-date-picker v-model="state.ruleForm.birthday" type="date" placeholder="出生日期" format="YYYY-MM-DD" value-format="YYYY-MM-DD" class="w100" />
  149. </el-form-item>
  150. </el-col>
  151. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  152. <el-form-item label="性别">
  153. <g-sys-dict v-model="state.ruleForm.sex" code="GenderEnum" render-as="radio" />
  154. </el-form-item>
  155. </el-col>
  156. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb5">
  157. <el-form-item label="年龄">
  158. <el-input-number v-model="state.ruleForm.age" placeholder="年龄" class="w100" />
  159. </el-form-item>
  160. </el-col>
  161. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  162. <el-form-item label="民族">
  163. <el-input v-model="state.ruleForm.nation" placeholder="民族" clearable />
  164. </el-form-item>
  165. </el-col>
  166. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  167. <el-form-item label="地址">
  168. <el-input v-model="state.ruleForm.address" placeholder="地址" clearable type="textarea" />
  169. </el-form-item>
  170. </el-col>
  171. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  172. <el-form-item label="毕业学校">
  173. <el-input v-model="state.ruleForm.college" placeholder="毕业学校" clearable />
  174. </el-form-item>
  175. </el-col>
  176. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  177. <el-form-item label="文化程度">
  178. <g-sys-dict v-model="state.ruleForm.cultureLevel" code="CultureLevelEnum" render-as="select" placeholder="文化程度" class="w100" />
  179. </el-form-item>
  180. </el-col>
  181. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  182. <el-form-item label="政治面貌">
  183. <el-input v-model="state.ruleForm.politicalOutlook" placeholder="政治面貌" clearable />
  184. </el-form-item>
  185. </el-col>
  186. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  187. <el-form-item label="办公电话">
  188. <el-input v-model="state.ruleForm.officePhone" placeholder="办公电话" clearable />
  189. </el-form-item>
  190. </el-col>
  191. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  192. <el-form-item label="紧急联系人">
  193. <el-input v-model="state.ruleForm.emergencyContact" placeholder="紧急联系人" clearable />
  194. </el-form-item>
  195. </el-col>
  196. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  197. <el-form-item label="联系人电话">
  198. <el-input v-model="state.ruleForm.emergencyPhone" placeholder="联系人电话" clearable />
  199. </el-form-item>
  200. </el-col>
  201. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  202. <el-form-item label="联系人地址">
  203. <el-input v-model="state.ruleForm.emergencyAddress" placeholder="联系人地址" clearable type="textarea" />
  204. </el-form-item>
  205. </el-col>
  206. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  207. <el-form-item label="备注">
  208. <el-input v-model="state.ruleForm.remark" placeholder="备注" clearable type="textarea" />
  209. </el-form-item>
  210. </el-col>
  211. </el-row>
  212. </el-form>
  213. </el-tab-pane>
  214. </el-tabs>
  215. <template #footer>
  216. <span class="dialog-footer">
  217. <el-button @click="cancel">取 消</el-button>
  218. <el-button type="primary" @click="submit">确 定</el-button>
  219. </span>
  220. </template>
  221. </el-dialog>
  222. </div>
  223. </template>
  224. <script lang="ts" setup name="sysEditUser">
  225. import { onMounted, reactive, ref } from 'vue';
  226. import { storeToRefs } from 'pinia';
  227. import { useUserInfo } from '/@/stores/userInfo';
  228. import { getAPI } from '/@/utils/axios-utils';
  229. import { SysPosApi, SysRoleApi, SysUserApi } from '/@/api-services/api';
  230. import { RoleOutput, SysOrg, SysPos, UpdateUserInput } from '/@/api-services/models';
  231. const props = defineProps({
  232. title: String,
  233. orgData: Array<SysOrg>,
  234. });
  235. const emits = defineEmits(['handleQuery']);
  236. const ruleFormRef = ref();
  237. const storesUserInfo = useUserInfo();
  238. const { userInfos } = storeToRefs(storesUserInfo);
  239. const state = reactive({
  240. loading: false,
  241. isShowDialog: false,
  242. selectedTabName: '0', // 选中的 tab 页
  243. ruleForm: {} as UpdateUserInput,
  244. posData: [] as Array<SysPos>, // 职位数据
  245. roleData: [] as Array<RoleOutput>, // 角色数据
  246. });
  247. // 级联选择器配置选项
  248. const cascaderProps = { checkStrictly: true, emitPath: false, value: 'id', label: 'name', expandTrigger: 'hover' };
  249. onMounted(async () => {
  250. state.loading = true;
  251. var res = await getAPI(SysPosApi).apiSysPosListGet();
  252. state.posData = res.data.result ?? [];
  253. var res1 = await getAPI(SysRoleApi).apiSysRoleListGet();
  254. state.roleData = res1.data.result ?? [];
  255. state.loading = false;
  256. });
  257. // 打开弹窗
  258. const openDialog = async (row: any) => {
  259. ruleFormRef.value?.resetFields();
  260. state.selectedTabName = '0'; // 重置为第一个 tab 页
  261. state.ruleForm = JSON.parse(JSON.stringify(row));
  262. if (row.id != undefined) {
  263. var resRole = await getAPI(SysUserApi).apiSysUserOwnRoleListUserIdGet(row.id);
  264. state.ruleForm.roleIdList = resRole.data.result;
  265. var resExtOrg = await getAPI(SysUserApi).apiSysUserOwnExtOrgListUserIdGet(row.id);
  266. state.ruleForm.extOrgIdList = resExtOrg.data.result;
  267. } else state.ruleForm.accountType = 777; // 默认普通账号类型
  268. state.isShowDialog = true;
  269. };
  270. // 关闭弹窗
  271. const closeDialog = () => {
  272. emits('handleQuery');
  273. state.isShowDialog = false;
  274. };
  275. // 取消
  276. const cancel = () => {
  277. state.isShowDialog = false;
  278. };
  279. // 提交
  280. const submit = () => {
  281. ruleFormRef.value.validate(async (valid: boolean) => {
  282. if (!valid) return;
  283. if (state.ruleForm.id != undefined && state.ruleForm.id > 0) {
  284. await getAPI(SysUserApi).apiSysUserUpdatePost(state.ruleForm);
  285. } else {
  286. await getAPI(SysUserApi).apiSysUserAddPost(state.ruleForm);
  287. }
  288. closeDialog();
  289. });
  290. };
  291. // 增加附属机构行
  292. const addExtOrgRow = () => {
  293. if (state.ruleForm.extOrgIdList == undefined) state.ruleForm.extOrgIdList = [];
  294. state.ruleForm.extOrgIdList?.push({});
  295. };
  296. // 删除附属机构行
  297. const deleteExtOrgRow = (k: number) => {
  298. state.ruleForm.extOrgIdList?.splice(k, 1);
  299. };
  300. // 导出对象
  301. defineExpose({ openDialog });
  302. </script>