Merge pull request #75 from cshaptx4869/patch-38

refactor: ♻️ 加强CURD表单组件、CURD使用示例改为真实接口
This commit is contained in:
Ray Hao
2024-04-29 16:34:13 +08:00
committed by GitHub
7 changed files with 132 additions and 194 deletions

View File

@@ -44,7 +44,7 @@ export function getUserForm(userId: number): AxiosPromise<UserForm> {
* *
* @param data * @param data
*/ */
export function addUser(data: any) { export function addUser(data: UserForm) {
return request({ return request({
url: "/api/v1/users", url: "/api/v1/users",
method: "post", method: "post",

View File

@@ -193,17 +193,17 @@ export interface IOperatData {
column: any; column: any;
$index: number; $index: number;
} }
export interface IContentConfig { export interface IContentConfig<T = any> {
// 页面名称(参与组成权限标识,如sys:user:xxx) // 页面名称(参与组成权限标识,如sys:user:xxx)
pageName: string; pageName: string;
// table组件属性 // table组件属性
table?: Omit<TableProps<any>, "data">; table?: Omit<TableProps<any>, "data">;
// 列表的网络请求函数(需返回promise) // 列表的网络请求函数(需返回promise)
indexAction: (data: IObject) => Promise<IObject>; indexAction: (queryParams: T) => Promise<any>;
// 删除的网络请求函数(需返回promise) // 删除的网络请求函数(需返回promise)
deleteAction?: (id: string) => Promise<any>; deleteAction?: (ids: string) => Promise<any>;
// 导出的网络请求函数(需返回promise) // 导出的网络请求函数(需返回promise)
exportAction?: (queryParams: IObject) => Promise<any>; exportAction?: (queryParams: T) => Promise<any>;
// 主键名(默认为id) // 主键名(默认为id)
pk?: string; pk?: string;
// 表格工具栏(默认支持refresh,add,delete,export,也可自定义) // 表格工具栏(默认支持refresh,add,delete,export,也可自定义)

View File

@@ -7,9 +7,10 @@
<!-- 表单 --> <!-- 表单 -->
<el-form <el-form
ref="formRef" ref="formRef"
:model="formData"
:rules="modalConfig.formRules"
label-width="80px" label-width="80px"
v-bind="modalConfig.form"
:model="formData"
:rules="formRules"
> >
<template v-for="item in modalConfig.formItems" :key="item.prop"> <template v-for="item in modalConfig.formItems" :key="item.prop">
<el-form-item :label="item.label" :prop="item.prop"> <el-form-item :label="item.label" :prop="item.prop">
@@ -65,37 +66,48 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, reactive } from "vue"; import { ref, reactive } from "vue";
import { useThrottleFn } from "@vueuse/core"; import { useThrottleFn } from "@vueuse/core";
import type { FormRules, DialogProps } from "element-plus"; import type {
DialogProps,
FormProps,
FormRules,
FormItemRule,
} from "element-plus";
// 对象类型 // 对象类型
type IObject = Record<string, any>; type IObject = Record<string, any>;
// 定义接收的属性 // 定义接收的属性
export interface IModalConfig { export interface IModalConfig<T = any> {
// dialog组件属性
dialog: Partial<Omit<DialogProps, "modelValue">>;
// 页面名称 // 页面名称
pageName?: string; pageName?: string;
// 主键名(主要用于编辑数据,默认为id)
pk?: string;
// dialog组件属性
dialog?: Partial<Omit<DialogProps, "modelValue">>;
// form组件属性
form?: Partial<Omit<FormProps, "model" | "rules">>;
// 提交的网络请求函数(需返回promise) // 提交的网络请求函数(需返回promise)
formAction: (data: IObject) => Promise<any>; formAction: (data: T) => Promise<any>;
// 提交之前处理
beforeSubmit?: (data: T) => void;
// 表单项 // 表单项
formItems: Array<{ formItems: Array<{
// 组件类型(如input,select,radio) // 组件类型(如input,select,radio,custom等默认input)
type: string; type?: string;
// 组件属性
attrs?: IObject;
// 组件可选项(适用于select,radio组件)
options?: { label: string; value: any }[];
// 插槽名(适用于组件类型为custom)
slotName?: string;
// 标签文本 // 标签文本
label: string; label: string;
// 键名 // 键名
prop: string; prop: string;
// 组件属性 // 验证规则
attrs?: IObject; rules?: FormItemRule[];
// 初始值 // 初始值
initialValue?: any; initialValue?: any;
// 可选项(适用于select,radio组件)
options?: { label: string; value: any }[];
// 插槽名(适用于组件类型为custom)
slotName?: string;
}>; }>;
// 表单验证规则
formRules: FormRules;
} }
const props = defineProps<{ const props = defineProps<{
modalConfig: IModalConfig; modalConfig: IModalConfig;
@@ -107,23 +119,32 @@ const emit = defineEmits<{
// 暴露的属性和方法 // 暴露的属性和方法
defineExpose({ setModalVisible }); defineExpose({ setModalVisible });
const pk = props.modalConfig.pk ?? "id";
const dialogVisible = ref(false); const dialogVisible = ref(false);
const formRef = ref<InstanceType<typeof ElForm>>(); const formRef = ref<InstanceType<typeof ElForm>>();
const formData = reactive<IObject>({}); const formData = reactive<IObject>({});
const formRules: FormRules = {};
// 初始化 // 初始化
function setModalVisible(initData: IObject = {}) { function setModalVisible(initData: IObject = {}) {
dialogVisible.value = true; dialogVisible.value = true;
for (const item of props.modalConfig.formItems) { for (const item of props.modalConfig.formItems) {
formData[item.prop] = initData[item.prop] ?? item.initialValue ?? ""; formData[item.prop] = initData[item.prop] ?? item.initialValue ?? "";
formRules[item.prop] = item.rules ?? [];
}
if (initData[pk]) {
formData[pk] = initData[pk];
} }
} }
// 表单提交 // 表单提交
const handleSubmit = useThrottleFn(() => { const handleSubmit = useThrottleFn(() => {
formRef.value?.validate((valid: boolean) => { formRef.value?.validate((valid: boolean) => {
if (valid) { if (valid) {
if (typeof props.modalConfig.beforeSubmit === "function") {
props.modalConfig.beforeSubmit(formData);
}
props.modalConfig.formAction(formData).then(() => { props.modalConfig.formAction(formData).then(() => {
ElMessage.success( ElMessage.success(
props.modalConfig.dialog.title props.modalConfig.dialog?.title
? `${props.modalConfig.dialog.title}成功` ? `${props.modalConfig.dialog.title}成功`
: "操作成功" : "操作成功"
); );

View File

@@ -1,6 +1,8 @@
import { addUser } from "@/api/user";
import type { UserForm } from "@/api/user/types";
import type { IModalConfig } from "@/components/PageModal/index.vue"; import type { IModalConfig } from "@/components/PageModal/index.vue";
const modalConfig: IModalConfig = { const modalConfig: IModalConfig<UserForm> = {
pageName: "sys:user", pageName: "sys:user",
dialog: { dialog: {
title: "新增用户", title: "新增用户",
@@ -8,37 +10,37 @@ const modalConfig: IModalConfig = {
appendToBody: true, appendToBody: true,
draggable: true, draggable: true,
}, },
formAction: function (data) { form: {
console.log("add", data); labelWidth: 100,
return new Promise((resolve, reject) => { },
resolve({ formAction: addUser,
code: "00000", beforeSubmit(data) {
data: null, console.log("提交之前处理", data);
msg: "新增成功",
});
});
}, },
formItems: [ formItems: [
{ {
type: "input",
label: "用户名", label: "用户名",
prop: "username", prop: "username",
rules: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
type: "input",
attrs: { attrs: {
placeholder: "请输入用户名", placeholder: "请输入用户名",
}, },
}, },
{ {
type: "input",
label: "用户昵称", label: "用户昵称",
prop: "nickname", prop: "nickname",
rules: [{ required: true, message: "用户昵称不能为空", trigger: "blur" }],
type: "input",
attrs: { attrs: {
placeholder: "请输入用户昵称", placeholder: "请输入用户昵称",
}, },
}, },
{ {
type: "tree-select",
label: "所属部门", label: "所属部门",
prop: "deptId", prop: "deptId",
rules: [{ required: true, message: "所属部门不能为空", trigger: "blur" }],
type: "tree-select",
attrs: { attrs: {
placeholder: "请选择所属部门", placeholder: "请选择所属部门",
data: [ data: [
@@ -76,9 +78,10 @@ const modalConfig: IModalConfig = {
], ],
}, },
{ {
type: "select",
label: "角色", label: "角色",
prop: "roleIds", prop: "roleIds",
rules: [{ required: true, message: "用户角色不能为空", trigger: "blur" }],
type: "select",
attrs: { attrs: {
placeholder: "请选择", placeholder: "请选择",
multiple: true, multiple: true,
@@ -100,24 +103,38 @@ const modalConfig: IModalConfig = {
type: "input", type: "input",
label: "手机号码", label: "手机号码",
prop: "mobile", prop: "mobile",
rules: [
{
pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
message: "请输入正确的手机号码",
trigger: "blur",
},
],
attrs: { attrs: {
placeholder: "请输入手机号码", placeholder: "请输入手机号码",
maxlength: 11, maxlength: 11,
}, },
}, },
{ {
type: "input",
label: "邮箱", label: "邮箱",
prop: "email", prop: "email",
rules: [
{
pattern: /\w[-\w.+]*@([A-Za-z0-9][-A-Za-z0-9]+\.)+[A-Za-z]{2,14}/,
message: "请输入正确的邮箱地址",
trigger: "blur",
},
],
type: "input",
attrs: { attrs: {
placeholder: "请输入邮箱", placeholder: "请输入邮箱",
maxlength: 50, maxlength: 50,
}, },
}, },
{ {
type: "radio",
label: "状态", label: "状态",
prop: "status", prop: "status",
type: "radio",
options: [ options: [
{ label: "正常", value: 1 }, { label: "正常", value: 1 },
{ label: "禁用", value: 0 }, { label: "禁用", value: 0 },
@@ -125,28 +142,6 @@ const modalConfig: IModalConfig = {
initialValue: 1, initialValue: 1,
}, },
], ],
formRules: {
username: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
nickname: [
{ required: true, message: "用户昵称不能为空", trigger: "blur" },
],
deptId: [{ required: true, message: "所属部门不能为空", trigger: "blur" }],
roleIds: [{ required: true, message: "用户角色不能为空", trigger: "blur" }],
email: [
{
pattern: /\w[-\w.+]*@([A-Za-z0-9][-A-Za-z0-9]+\.)+[A-Za-z]{2,14}/,
message: "请输入正确的邮箱地址",
trigger: "blur",
},
],
mobile: [
{
pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
message: "请输入正确的手机号码",
trigger: "blur",
},
],
},
}; };
export default modalConfig; export default modalConfig;

View File

@@ -1,70 +1,24 @@
import { deleteUsers, exportUser, getUserPage } from "@/api/user";
import type { UserQuery } from "@/api/user/types";
import type { IContentConfig } from "@/components/PageContent/index.vue"; import type { IContentConfig } from "@/components/PageContent/index.vue";
import { exportUser } from "@/api/user";
const contentConfig: IContentConfig = { const contentConfig: IContentConfig<UserQuery> = {
pageName: "sys:user", pageName: "sys:user",
table: { table: {
border: true, border: true,
highlightCurrentRow: true, highlightCurrentRow: true,
}, },
indexAction: function (data) { indexAction: function (params) {
console.log("index", data); if ("createAt" in params) {
return new Promise((resolve, reject) => { const createAt = params.createAt as string[];
setTimeout(() => { params.startTime = createAt[0];
resolve({ params.endTime = createAt[1];
code: "00000", delete params.createAt;
data: { }
list: [ return getUserPage(params);
{
id: 2,
username: "admin",
nickname: "系统管理员",
mobile: "17621210366",
genderLabel: "男",
avatar:
"https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif",
email: null,
status: 1,
deptName: "有来技术",
roleNames: "系统管理员",
createTime: "2019-10-10",
},
{
id: 3,
username: "test",
nickname: "测试小用户",
mobile: "17621210366",
genderLabel: "男",
avatar:
"https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif",
email: null,
status: 1,
deptName: "测试部门",
roleNames: "访问游客",
createTime: "2021-06-05",
},
],
total: 2,
},
msg: "一切ok",
});
}, 800);
});
},
deleteAction: function (id) {
console.log("delete", id);
return new Promise((resolve, reject) => {
resolve({
code: "00000",
data: null,
msg: "删除成功",
});
});
},
exportAction: function (queryParams) {
// 导出Excel文件
return exportUser(queryParams as any);
}, },
deleteAction: deleteUsers,
exportAction: exportUser,
pk: "id", pk: "id",
toolbar: [ toolbar: [
"refresh", "refresh",

View File

@@ -1,44 +1,46 @@
import { updateUser } from "@/api/user";
import type { UserForm } from "@/api/user/types";
import type { IModalConfig } from "@/components/PageModal/index.vue"; import type { IModalConfig } from "@/components/PageModal/index.vue";
const modalConfig: IModalConfig = { const modalConfig: IModalConfig<UserForm> = {
pageName: "sys:user", pageName: "sys:user",
pk: "id",
dialog: { dialog: {
title: "修改用户", title: "修改用户",
width: 800, width: 800,
appendToBody: true, appendToBody: true,
}, },
formAction: function (data) { formAction: function (data) {
console.log("edit", data); return updateUser(data.id as number, data);
return new Promise((resolve, reject) => { },
resolve({ beforeSubmit(data) {
code: "00000", console.log("提交之前处理", data);
data: null,
msg: "修改成功",
});
});
}, },
formItems: [ formItems: [
{ {
type: "input",
label: "用户名", label: "用户名",
prop: "username", prop: "username",
rules: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
type: "input",
attrs: { attrs: {
placeholder: "请输入用户名", placeholder: "请输入用户名",
readonly: true, readonly: true,
}, },
}, },
{ {
type: "input",
label: "用户昵称", label: "用户昵称",
prop: "nickname", prop: "nickname",
rules: [{ required: true, message: "用户昵称不能为空", trigger: "blur" }],
type: "input",
attrs: { attrs: {
placeholder: "请输入用户昵称", placeholder: "请输入用户昵称",
}, },
}, },
{ {
type: "tree-select",
label: "所属部门", label: "所属部门",
prop: "deptId", prop: "deptId",
rules: [{ required: true, message: "所属部门不能为空", trigger: "blur" }],
type: "tree-select",
attrs: { attrs: {
placeholder: "请选择所属部门", placeholder: "请选择所属部门",
data: [ data: [
@@ -76,9 +78,10 @@ const modalConfig: IModalConfig = {
], ],
}, },
{ {
type: "select",
label: "角色", label: "角色",
prop: "roleIds", prop: "roleIds",
rules: [{ required: true, message: "用户角色不能为空", trigger: "blur" }],
type: "select",
attrs: { attrs: {
placeholder: "请选择", placeholder: "请选择",
multiple: true, multiple: true,
@@ -100,53 +103,44 @@ const modalConfig: IModalConfig = {
type: "input", type: "input",
label: "手机号码", label: "手机号码",
prop: "mobile", prop: "mobile",
rules: [
{
pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
message: "请输入正确的手机号码",
trigger: "blur",
},
],
attrs: { attrs: {
placeholder: "请输入手机号码", placeholder: "请输入手机号码",
maxlength: 11, maxlength: 11,
}, },
}, },
{ {
type: "input",
label: "邮箱", label: "邮箱",
prop: "email", prop: "email",
rules: [
{
pattern: /\w[-\w.+]*@([A-Za-z0-9][-A-Za-z0-9]+\.)+[A-Za-z]{2,14}/,
message: "请输入正确的邮箱地址",
trigger: "blur",
},
],
type: "input",
attrs: { attrs: {
placeholder: "请输入邮箱", placeholder: "请输入邮箱",
maxlength: 50, maxlength: 50,
}, },
}, },
{ {
type: "radio",
label: "状态", label: "状态",
prop: "status", prop: "status",
type: "radio",
options: [ options: [
{ label: "正常", value: 1 }, { label: "正常", value: 1 },
{ label: "禁用", value: 0 }, { label: "禁用", value: 0 },
], ],
initialValue: 1,
}, },
], ],
formRules: {
username: [{ required: true, message: "用户名不能为空", trigger: "blur" }],
nickname: [
{ required: true, message: "用户昵称不能为空", trigger: "blur" },
],
deptId: [{ required: true, message: "所属部门不能为空", trigger: "blur" }],
roleIds: [{ required: true, message: "用户角色不能为空", trigger: "blur" }],
email: [
{
pattern: /\w[-\w.+]*@([A-Za-z0-9][-A-Za-z0-9]+\.)+[A-Za-z]{2,14}/,
message: "请输入正确的邮箱地址",
trigger: "blur",
},
],
mobile: [
{
pattern: /^1[3|4|5|6|7|8|9][0-9]\d{8}$/,
message: "请输入正确的手机号码",
trigger: "blur",
},
],
},
}; };
export default modalConfig; export default modalConfig;

View File

@@ -42,12 +42,13 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import searchConfig from "./config/search"; import { getUserForm } from "@/api/user";
import contentConfig from "./config/content"; import type { IObject, IOperatData } from "@/hooks/usePage";
import addModalConfig from "./config/add";
import editModalConfig from "./config/edit";
import usePage from "@/hooks/usePage"; import usePage from "@/hooks/usePage";
import type { IOperatData, IObject } from "@/hooks/usePage"; import addModalConfig from "./config/add";
import contentConfig from "./config/content";
import editModalConfig from "./config/edit";
import searchConfig from "./config/search";
const { const {
searchRef, searchRef,
@@ -62,37 +63,10 @@ const {
handleExportClick, handleExportClick,
} = usePage(); } = usePage();
// 编辑 // 编辑
function handleEditClick(row: IObject) { async function handleEditClick(row: IObject) {
// 模拟根据id获取数据进行填充 // 根据id获取数据进行填充
const idMap: IObject = { const response = await getUserForm(row.id);
2: { editModalRef.value?.setModalVisible(response.data);
id: 2,
username: "admin",
nickname: "系统管理员",
mobile: "17621210366",
gender: 1,
avatar:
"https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif",
email: "",
status: 1,
deptId: 1,
roleIds: [2],
},
3: {
id: 3,
username: "test",
nickname: "测试小用户",
mobile: "17621210366",
gender: 1,
avatar:
"https://oss.youlai.tech/youlai-boot/2023/05/16/811270ef31f548af9cffc026dfc3777b.gif",
email: "youlaitech@163.com",
status: 1,
deptId: 3,
roleIds: [3],
},
};
editModalRef.value?.setModalVisible(idMap[row.id]);
} }
// 其他工具栏 // 其他工具栏
function handleToolbarClick(name: string) { function handleToolbarClick(name: string) {