vue-manage-system/src/components/table-edit.vue

153 lines
4.0 KiB
Vue

<template>
<el-form
ref="formRef"
:model="form"
:rules="rules"
:label-width="options.labelWidth"
>
<el-row>
<el-col :span="options.span" v-for="item in options.list">
<el-form-item :label="item.label" :prop="item.prop">
<!-- 文本框、数字框、下拉框、日期框、开关、上传 -->
<el-input
v-if="item.type === 'input'"
v-model="form[item.prop]"
:disabled="item.disabled"
:placeholder="item.placeholder"
clearable
/>
<el-input-number
v-else-if="item.type === 'number'"
v-model="form[item.prop]"
:disabled="item.disabled"
controls-position="right"
/>
<el-select
v-else-if="item.type === 'select'"
v-model="form[item.prop]"
:disabled="item.disabled"
:placeholder="item.placeholder"
clearable
>
<el-option
v-for="opt in item.opts"
:label="opt.label"
:value="opt.value"
/>
</el-select>
<el-date-picker
v-else-if="item.type === 'date'"
type="date"
v-model="form[item.prop]"
:value-format="item.format"
/>
<el-switch
v-else-if="item.type === 'switch'"
v-model="form[item.prop]"
:active-value="item.activeValue"
:inactive-value="item.inactiveValue"
:active-text="item.activeText"
:inactive-text="item.inactiveText"
/>
<el-upload
v-else-if="item.type === 'upload'"
class="avatar-uploader"
action="#"
:show-file-list="false"
:on-success="handleAvatarSuccess"
>
<img v-if="form[item.prop]" :src="form[item.prop]" class="avatar" />
<el-icon v-else class="avatar-uploader-icon">
<Plus />
</el-icon>
</el-upload>
<slot :name="item.prop" v-else> </slot>
</el-form-item>
</el-col>
</el-row>
<el-form-item>
<el-button type="primary" @click="saveEdit(formRef)"> </el-button>
</el-form-item>
</el-form>
</template>
<script lang="ts" setup>
import { FormOption } from "@/types/form-option";
import { FormInstance, FormRules, UploadProps } from "element-plus";
import { PropType, ref } from "vue";
const { options, formData, edit, update } = defineProps({
options: {
type: Object as PropType<FormOption>,
required: true,
},
formData: {
type: Object,
required: true,
},
edit: {
type: Boolean,
required: false,
},
update: {
type: Function,
required: true,
},
});
const form = ref({ ...(edit ? formData : {}) });
const rules: FormRules = options.list
.map((item) => {
if (item.required) {
return {
[item.prop]: [
{ required: true, message: `${item.label}不能为空`, trigger: "blur" },
],
};
}
return {};
})
.reduce((acc, cur) => ({ ...acc, ...cur }), {});
const formRef = ref<FormInstance>();
const saveEdit = (formEl: FormInstance | undefined) => {
if (!formEl) return;
formEl.validate((valid) => {
if (!valid) return false;
update(form.value);
});
};
const handleAvatarSuccess: UploadProps["onSuccess"] = (
response,
uploadFile
) => {
form.value.thumb = URL.createObjectURL(uploadFile.raw!);
};
</script>
<style>
.avatar-uploader .el-upload {
border: 1px dashed var(--el-border-color);
border-radius: 6px;
cursor: pointer;
position: relative;
overflow: hidden;
transition: var(--el-transition-duration-fast);
}
.avatar-uploader .el-upload:hover {
border-color: var(--el-color-primary);
}
.el-icon.avatar-uploader-icon {
font-size: 28px;
color: #8c939d;
width: 178px;
height: 178px;
text-align: center;
}
</style>