代码优化——基于element-plus封装组件:表单封装

news2025/3/13 4:46:08

前言

    今天实现一个基于element-plus表单组件的二次封装,什么是二次封装?查看以下表单,传统表单组件是不是用<el-form>嵌套几个<el-form-item>即可实现,那么一个表单可不可以实现,传入一个对象给封装组件,通过对象的type值来进行表单的实现。当一个项目有多个不同的表单内容时,能够实现代码的复用性、可读性。

点击提交校验效果:

 

传统写法:

<el-form :model="form" label-width="auto" style="max-width: 600px">
    <el-form-item label="Activity name">
      <el-input v-model="form.name" />
    </el-form-item>
</el-form>

封装后写法(封装组件命名为:form-enhance):

const formConfig = [ { key: 'name', type: 'input'} ]

<form-enhance :formConfig=formConfig />

Vite+Vue3+elemplus-plus传统表单实现:

<!-- 原始表单 -->
<script setup>
import { ref } from 'vue'
const form = ref({
    name: '',
    region: '',
    date1: '',
    date2: '',
    delivery: false,
    type: [],
    resource: '',
    desc: '',
})

const onSubmit = () => {
    console.log('submit!')
}
</script>
<!-- 初始表单 -->
<template>
    <el-form ref="form" :model="form" label-width="80px">

        <el-form-item label="活动名称">
            <el-input v-model="form.name"></el-input>
        </el-form-item>

        <el-form-item label="活动区域">
            <el-select v-model="form.region" placeholder="请选择活动区域">
                <el-option label="区域一" value="shanghai"></el-option>
                <el-option label="区域二" value="beijing"></el-option>
            </el-select>
        </el-form-item>

        <el-form-item label="活动时间">
            <el-col :span="11">
                <el-date-picker type="date" placeholder="选择日期" v-model="form.date1"
                    style="width: 100%;"></el-date-picker>
            </el-col>
            <el-col class="line" :span="2">-</el-col>
            <el-col :span="11">
                <el-time-picker placeholder="选择时间" v-model="form.date2" xx` style="width: 100%;"></el-time-picker>
            </el-col>
        </el-form-item>

        <el-form-item label="即时配送">
            <el-switch v-model="form.delivery"></el-switch>
        </el-form-item>

        <el-form-item label="活动性质">
            <el-checkbox-group v-model="form.type">
                <el-checkbox label="美食/餐厅线上活动" name="type"></el-checkbox>
                <el-checkbox label="地推活动" name="type"></el-checkbox>
                <el-checkbox label="线下主题活动" name="type"></el-checkbox>
                <el-checkbox label="单纯品牌曝光" name="type"></el-checkbox>
            </el-checkbox-group>
        </el-form-item>

        <el-form-item label="特殊资源">
            <el-radio-group v-model="form.resource">
                <el-radio label="线上品牌商赞助"></el-radio>
                <el-radio label="线下场地免费"></el-radio>
            </el-radio-group>
        </el-form-item>

        <el-form-item label="活动形式">
            <el-input type="textarea" v-model="form.desc"></el-input>
        </el-form-item>

        <el-form-item>
            <el-button type="primary" @click="onSubmit">立即创建</el-button>
            <el-button>取消</el-button>
        </el-form-item>

    </el-form>
</template>

<style scoped></style>

Vite+Vue3+elemplus-plus二次封装表单实现:

FormEnhance..vue子组件

<template>
    <el-form :model="formData" :label-width="labelWidth" ref="formRef">
        <template v-for="(row, rowIndex) in formConfig" :key="rowIndex">
            <el-row :gutter="20">
                <template v-for="item in row" :key="item.key">
                    <el-col :span="item.span || 24">
                        <!-- 日期分隔符处理 -->
                        <template v-if="item.type === 'separator'">
                            <div class="form-separator">{{ item.content || '-' }}</div>
                        </template>

                        <!-- 表单项目处理 添加规则 -->
                        <el-form-item v-else :label="item.label" :prop="item.key" :rules="item.rules">

                            <!-- input输入框 -->
                            <el-input v-if="item.type === 'input'" v-model="formData[item.key]"
                                :placeholder="item.placeholder" clearable />

                            <!-- select选择器 -->
                            <el-select v-else-if="item.type === 'select'" v-model="formData[item.key]"
                                :placeholder="item.placeholder" clearable>
                                <el-option v-for="option in item.options" :key="option.value" :label="option.label"
                                    :value="option.value" />
                            </el-select>

                            <!-- 日期选择器 -->
                            <el-date-picker v-else-if="item.type === 'date'" v-model="formData[item.key]" type="date"
                                :placeholder="item.placeholder" style="width: 100%" />

                            <!-- 时间选择器 -->
                            <el-time-picker v-else-if="item.type === 'time'" v-model="formData[item.key]"
                                :placeholder="item.placeholder" style="width: 100%" />

                            <!-- 开关 -->
                            <el-switch v-else-if="item.type === 'switch'" v-model="formData[item.key]" />

                            <!-- 复选框组 -->
                            <el-checkbox-group v-else-if="item.type === 'checkbox-group'" v-model="formData[item.key]">
                                <el-checkbox v-for="option in item.options" :key="option.value" :label="option.value">
                                    {{ option.label }}
                                </el-checkbox>

                            </el-checkbox-group>

                            <!-- 单选框组 -->
                            <el-radio-group v-else-if="item.type === 'radio-group'" v-model="formData[item.key]">
                                <el-radio v-for="option in item.options" :key="option.value" :label="option.value">
                                    {{ option.label }}
                                </el-radio>
                            </el-radio-group>

                            <!-- 文本域 -->
                            <el-input v-else-if="item.type === 'textarea'" v-model="formData[item.key]" type="textarea"
                                :rows="4" />

                        </el-form-item>
                    </el-col>
                </template>
            </el-row>
        </template>
    </el-form>
</template>

<script setup>
import { ref } from 'vue'
// 接收参数
defineProps({
    formData: {
        type: Object,
        required: true
    },
    formConfig: {
        type: Array,
        required: true,
        validator: value => value.every(row => Array.isArray(row))
    },
    labelWidth: {
        type: String,
        default: '80px'
    }
})
// 提交表单的方法
const formRef = ref(null);

const submitForm = () => {
    formRef.value.validate((valid) => {
        if (valid) {
            console.log('表单提交成功:', props.formData)
        } else {
            console.error('请完善表单!')
            return false;
        }
    })
}
// 重置表单的方法
const resetForm = () => {
    // 清空校验信息
    formRef.value.clearValidate();
}
// 暴露给父组件使用
defineExpose({
    submitForm,
    resetForm
})

</script>

<style lang="scss" scoped>
.form-separator {
    text-align: center;
    line-height: 32px;
    height: 100%;
}
</style>

encapsulationForm.vue 父组件

<!-- 二次封装表单 -->
<script setup>
import { ref } from 'vue'
import FormEnhance from '../common/components/FormEnhance..vue'

const formEnhanceRef = ref(null);
// 触发子组件的submitForm方法
const handleSubmit = () => {
    if (formEnhanceRef.value && typeof formEnhanceRef.value.submitForm === 'function') {
        formEnhanceRef.value.submitForm();
    } else {
        console.error("---");
    }
}
// 重置处理函数
const handleReset = () => {
    if (formEnhanceRef.value && typeof formEnhanceRef.value.resetForm === 'function') {
        // 清除验证信息
        formEnhanceRef.value.resetForm();
        // 重置表单数据
        for (let key in resetData.value) {
            formData.value[key] = resetData.value[key];
        }


    } else {
        console.error("-----");
    }
}
// 表单数据
const formData = ref({
    name: '',//活动名称
    region: '', //活动区域
    date1: '', // 活动时间
    date2: '',  // 活动时间
    delivery: false, // 开关
    type: [], // 活动性质
    resource: '', // 活动资源
    desc: '' // 活动形式
})
// 重置表单数据
const resetData = ref({
    name: '',//活动名称
    region: '', //活动区域
    date1: '', // 活动时间
    date2: '',  // 活动时间
    delivery: false, // 开关
    type: [], // 活动性质
    resource: '', // 活动资源
    desc: '' // 活动形式
})
// 表单配置 
const formConfig = [
    // 活动名称
    [
        {
            key: 'name',
            type: 'input',
            label: '活动名称',
            span: 24,
            placeholder: '请输入活动名称',
            rules: [
                { required: true, message: '请输入活动名称', trigger: 'blur' },
            ]
        }
    ],

    // 活动区域
    [
        {
            key: 'region',
            type: 'select',
            label: '活动区域',
            span: 24,
            placeholder: '请选择活动区域',
            options: [
                { label: '区域一', value: 'shanghai' },
                { label: '区域二', value: 'beijing' }
            ],
            rules: [
                { required: true, message: '请选择活动区域', trigger: 'blur' },
            ]
        }
    ],

    // 活动时间 (日期+分隔符+时间)
    [
        {
            key: 'date1',
            type: 'date',
            label: '活动时间',
            span: 11,
            placeholder: '选择日期',
            rules: [{ required: true, message: '请选择活动时间', trigger: 'change' }]
        },
        {
            type: 'separator',//分隔符
            span: 2
        },
        {
            key: 'date2',
            type: 'time',
            span: 11,
            placeholder: '选择时间',
            rules: [{ required: true, message: '请选择活动时间', trigger: 'change' }]
        }
    ],

    // 即时配送
    [
        {
            key: 'delivery',
            type: 'switch',
            label: '即时配送',
            span: 24
        }
    ],

    // 活动性质
    [
        {
            key: 'type',
            type: 'checkbox-group',
            label: '活动性质',
            span: 24,
            options: [
                { label: '美食/餐厅线上活动', value: '美食/餐厅线上活动' },
                { label: '地推活动', value: '地推活动' },
                { label: '线下主题活动', value: '线下主题活动' },
                { label: '单纯品牌曝光', value: '单纯品牌曝光' }
            ],
            rules: [
                { required: true, message: '请选择活动性质', trigger: 'blur' },
            ]
        }
    ],

    // 特殊资源
    [
        {
            key: 'resource',
            type: 'radio-group',
            label: '特殊资源',
            span: 24,
            options: [
                { label: '线上品牌商赞助', value: '线上品牌商赞助' },
                { label: '线下场地免费', value: '线下场地免费' }
            ],
            rules: [
                { required: true, message: '请选择', trigger: 'blur' },
            ]
        }
    ],

    // 活动形式
    [
        {
            key: 'desc',
            type: 'textarea',
            label: '活动形式',
            span: 24,
            placeholder: '请输入活动形式',
            rules: [
                { required: true, message: '请输入活动形式', trigger: 'blur' },
            ]
        }
    ]
]
</script>

<template>
    <div class="form">
        <!-- 使用FormEnhance组件 -->
        <FormEnhance ref="formEnhanceRef" :form-data="formData" :form-config="formConfig" />
        <!-- 提交按钮 -->
        <el-button type="primary" @click="handleSubmit">提交</el-button>
        <!-- 重置表单 -->
        <el-button type="default" @click="handleReset">重置</el-button>
    </div>
</template>

扩展:父组件点击提交和重置时,触发子组件 submitForm, resetForm 方法

子组件:el-form记得绑定ref="formRef"

// 提交表单的方法
const formRef = ref(null);

const submitForm = () => {
    formRef.value.validate((valid) => {
        if (valid) {
            console.log('表单提交成功:', props.formData)
        } else {
            console.error('请完善表单!')
            return false;
        }
    })
}
// 重置表单的方法
const resetForm = () => {
    // 清空校验信息
    formRef.value.clearValidate();
}
// 暴露给父组件使用
defineExpose({
    submitForm,
    resetForm
})

父组件调用子组件暴露出的方法:

const formEnhanceRef = ref(null);
// 触发子组件的submitForm方法
const handleSubmit = () => {
    if (formEnhanceRef.value && typeof formEnhanceRef.value.submitForm === 'function') {
        formEnhanceRef.value.submitForm();
    } else {
        console.error("---");
    }
}
// 重置处理函数
const handleReset = () => {
    if (formEnhanceRef.value && typeof formEnhanceRef.value.resetForm === 'function') {
        // 清除验证信息
        formEnhanceRef.value.resetForm();
        // 重置表单数据
        for (let key in resetData.value) {
            formData.value[key] = resetData.value[key];
        }


    } else {
        console.error("-----");
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2314078.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C/C++中使用CopyFile、CopyFileEx原理、用法、区别及分别在哪些场景使用

文章目录 1. CopyFile原理函数原型返回值用法示例适用场景 2. CopyFileEx原理函数原型返回值用法示例适用场景 3. 核心区别4. 选择建议5. 常见问题6.区别 在Windows系统编程中&#xff0c;CopyFile和CopyFileEx是用于文件复制的两个API函数。它们的核心区别在于功能扩展性和控制…

qt 多进程使用共享内存 ,加速数据读写,进程间通信 共享内存

Summary: 项目中我们有时需要使用共享内存共享数据&#xff0c;这样&#xff0c;数据不用进程IO读写&#xff0c;加进数据加载和落地&#xff1b; 程序退出时&#xff0c;再保存到本地&#xff1b;速度提升数十倍&#xff1b; Part1:QSharedMemory Windows平台下进程间通信…

【鸿蒙开发】OpenHarmony调测工具hdc使用教程(设备开发者)

00. 目录 文章目录 00. 目录01. OpenHarmony概述02. hdc简介03. hdc获取04. option相关的命令05. 查询设备列表的命令06. 服务进程相关命令07. 网络相关的命令08. 文件相关的命令09. 应用相关的命令10. 调试相关的命令11. 常见问题12. 附录 01. OpenHarmony概述 OpenHarmony是…

【贪心算法】简介

1.贪心算法 贪心策略&#xff1a;解决问题的策略&#xff0c;局部最优----》全局最优 &#xff08;1&#xff09;把解决问题的过程分成若干步 &#xff08;2&#xff09;解决每一步的时候&#xff0c;都选择当前看起来的“最优”的算法 &#xff08;3&#xff09;“希望”得…

transformer模型介绍——大语言模型 LLMBook 学习(二)

1. transformer模型 1.1 注意力机制 **注意力机制&#xff08;Attention Mechanism&#xff09;**在人工智能中的应用&#xff0c;实际上是对人类认知系统中的注意力机制的一种模拟。它主要模仿了人类在处理信息时的选择性注意&#xff08;Selective Attention&#xff09;&a…

GStreamer —— 2.17、Windows下Qt加载GStreamer库后运行 - “播放教程 5:色彩平衡“(附:完整源码)

运行效果 介绍 亮度、对比度、色相和饱和度是常见的视频调整&#xff0c; 在 GStreamer 中统称为 Color Balance 设置。 本教程展示了&#xff1a; • 如何找出可用的色彩平衡通道 • 如何更改它们 允许访问颜色平衡设置。如果 元素支持这个接口&#xff0c;只需将其转发给应用…

串口通信ASCII码转16进制及C#串口编程完整源码下载

在工业自动化、嵌入式系统及物联网以行业中&#xff0c;串口编程非常重要。 串口编程&#xff0c;重点在于串口数据通信和数据处理。 在C#中&#xff0c;System.IO.Ports命名空间提供了SerialPort类&#xff0c;用于实现串口通信。 串口程序的开发主要包括以下几点 1.引用命…

解决vscode中出现“无法将pip项识别...“问题

问题 遇见问题如下&#xff1a; 查看pip 通过 winR &#xff0c;输入 cmd&#xff0c;进入终端&#xff0c;搜索 where pip。 发现 pip 查不出来&#xff0c;然后进入文件资源管理器&#xff0c;搜索 Scripts 文件夹&#xff0c;如果没有找到可能是电脑没有下载 python。 点击…

nacos下载及安装

下载官方最新稳定版 github下载较慢&#xff0c;推荐下面的下载链接 Nacos Server 下载 | Nacos 官网 点击下载和试用下载最新稳定版 Nacos Server 下载 | Nacos 官网 配置检查&#xff08;可选&#xff09; 默认情况下&#xff0c;Nacos 使用内置的 Derby 数据库&#x…

C++从零实现Json-Rpc框架

文章目录 一、项目介绍1. 基本原理2. 涉及到的技术栈3. 最终实现的效果 二、 第三方库的介绍与使用1. JsonCpp库Json的数据格式JsonCpp介绍封装Json工具类 2. muduo库muduo库是什么Muduo库常见接口介绍 3. C11异步操作std::future 三、框架设计1. 服务端模块划分NetworkProtoco…

rom定制系列------小米note3 原生安卓15 批量线刷 默认开启usb功能选项 插电自启等

小米Note 3搭载骁龙660处理器&#xff0c;1200万像素广角镜头、俗称大号版的小米6&#xff0c;官方最终版为12.0.1稳定版安卓9的固件。客户需要运行在安卓15的rom。根据原生官网的rom修改一些功能选项。以便客户操作需求。 定制资源说明 根据客户需求采用安卓15原生系统为底包…

大数据学习(59)-DataX执行机制

&&大数据学习&& &#x1f525;系列专栏&#xff1a; &#x1f451;哲学语录: 承认自己的无知&#xff0c;乃是开启智慧的大门 &#x1f496;如果觉得博主的文章还不错的话&#xff0c;请点赞&#x1f44d;收藏⭐️留言&#x1f4dd;支持一下博主哦&#x1f91…

YashanDB认证,YCA证书认证教程,免费证书,内含真题考试题库及答案——五分钟速成

目录 一.账号及平台注册登录流程 二.登录进行设备调试核验 三.考试&#xff08;考完获取分数&#xff09; 四.获取证书 五.题库及答案 一.账号及平台注册登录流程 1-点击这里进行账号注册&#xff08;首次学习必须先注册&#xff0c;有账号之后可以直接在2号链接登录&#…

自然语言处理中的语音识别技术:从声波到语义的智能解码

引言 语音识别&#xff08;Automatic Speech Recognition, ASR&#xff09;是自然语言处理&#xff08;NLP&#xff09;的关键分支&#xff0c;旨在将人类语音信号转化为可处理的文本信息。随着深度学习技术的突破&#xff0c;语音识别已从实验室走向日常生活&#xff0c;赋能…

010-Catch2

Catch2 一、框架简介 Catch2 是一个基于 C 的现代化单元测试框架&#xff0c;支持 TDD&#xff08;测试驱动开发&#xff09;和 BDD&#xff08;行为驱动开发&#xff09;模式。其核心优势在于&#xff1a; 单头文件设计&#xff1a;v2.x 版本仅需包含 catch.hpp 即可使用自然…

C++之序列容器(vector,list,dueqe)

1.大体对比 在软件开发的漫长历程中&#xff0c;数据结构与算法始终占据着核心地位&#xff0c;犹如大厦的基石&#xff0c;稳固支撑着整个程序的运行。在众多编程语言中&#xff0c;数据的存储与管理方式各有千秋&#xff0c;而 C 凭借其丰富且强大的工具集脱颖而出&#xff…

安卓Android与iOS设备管理对比:企业选择指南

目录 一、管理方式差异 Android Enterprise方案包含三种典型模式&#xff1a; Apple MDM方案主要提供两种模式&#xff1a; 二、安全防护能力 Android系统特点&#xff1a; 三、应用管理方案 四、设备选择建议 五、典型场景推荐 需求场景 推荐方案 六、决策建议要点…

版本控制器Git(1)

文章目录 前言一、初识Git问题引入解决方案注意事项 二、Git安装三、Git配置与基本操作Git创建Git配置用户名称和地址认识工作区、暂存区、版本库添加文件到仓库添加文件到暂存区提交暂存区内容到本地仓库 查看提交历史 四、Git 暂存区、HEAD、对象库及文件Git内部结构概览查看…

推理模型对SQL理解能力的评测:DeepSeek r1、GPT-4o、Kimi k1.5和Claude 3.7 Sonnet

引言 随着大型语言模型&#xff08;LLMs&#xff09;在技术领域的应用日益广泛&#xff0c;评估这些模型在特定技术任务上的能力变得越来越重要。本研究聚焦于四款领先的推理模型——DeepSeek r1、GPT-4o、Kimi k1.5和Claude 3.7 Sonnet在SQL理解与分析方面的能力&#xff0c;…

[动手学习深度学习]12.权重衰退

1.介绍 权重衰退是常见的处理过拟合的方法 控制模型容量方法 把模型控制的比较小&#xff0c;即里面参数比较少使参数选择范围小 约束就是正则项 每个特征的权重都大会导致模型复杂&#xff0c;从而导致过拟合。 控制权重矩阵范数可以使得减少一些特征的权重&#xff0c;甚至…