jjzjj

Vue配置化生成表单(个人笔记)

kevision 2023-03-28 原文

背景:为了避免编写重复的html, css, js代码(虽然一般情况下都是复制粘贴的),提高开发效率(其实是不想写)。配置化生成表单只需要我们写js对象就可以了。

知识点:Vue自带的component动态组件,is属性可以根据我们传入的参数解析成不同的组件。传入我们自定义的组件就会渲染成相应组件,传入el-input就会渲染成<el-input></el-input>组件。

封装schema-form组件

<template>
    <div>
        <el-form :model="form" ref="ruleForm" :rules="rules" :label-width="labelWidth" :class="direction==='vertical'?'normal_form':'flex_form'">
            <el-form-item v-for="item in options" :key="item.key" :prop="item.key" :label="item.label" :label-width="item.labelWidth || labelWidth" class="form_item">
                <!--提供一个插槽,提高扩展性-->
                <slot v-if="item.component === 'slot'" :name="item.slotName"></slot>
                <template v-else>
                    <component :is="item.component" v-model="form[item.key]" v-bind="item.props" v-on="item.listeners" :style="{width: item.width || '200px'}">
                        <template v-if="item.component === 'el-select'">
                            <el-option v-for="(subitem,index) in item.data" :key="index" :label="subitem.label" :value="subitem.value"></el-option>
                        </template>
                        <template v-else-if="item.component === 'el-checkbox-group'">
                            <el-checkbox
                                v-for="(subitem,index) in item.data"
                                :key="index"
                                :label="subitem.value"
                            >
                                {{ subitem.label }}
                            </el-checkbox>
                        </template>
                        <template v-else-if="item.component === 'el-radio-group'">
                            <el-radio
                                v-for="(subitem,index) in item.data"
                                :key="index"
                                :label="subitem.value"
                            >
                                {{ subitem.label }}
                            </el-radio>
                        </template>
                    </component>
                </template>
            </el-form-item>
        </el-form>
    </div>
</template>

el-select,el-checkbox-group,el-radio-group组件需要做特殊处理。

<script>
export default{
    props: {
        form: { // 表单绑定数据
            type: Object,
            default: {}
        },
        options: { // 表单项配置数据
            type: Array,
            default: []
        },
        rules: { // 表单验证规则
            type: Object,
            default: {}
        },
        labelWidth: { // label宽度
            type: String,
            default: '100px'
        },
        direction: { // 表单方向
            type: String,
            default: 'vertical'
        },
    },
    methods: {
        validate(callback) {
            this.$refs.ruleForm.validate(valid => {
                callback(valid)
            })
        },
        resetFields() {
            this.$refs.ruleForm.resetFields()
        }
    },
}
</script>
<style lang='scss' scoped>
    .flex_form {
        display: flex;
        flex-flow: wrap;
    }
</style>

使用组件

<template>
    <div>
        <schema-form ref="schemaForm" :form="form" :options="options" :rules="rules" labelWidth="120px" direction="vertical">
            <div slot="search">
                <el-button type="primary" @click="handleSubmit">搜索</el-button>
                <el-button @click="handleReset">重置</el-button>
            </div>
        </schema-form>
    </div>
</template>
<script>
import schemaForm from './components/schemaForm';
export default{
    components: { schemaForm },
    data() {
        return {
            form: {
                activity: [],
                age: '18'
            },
            options: [],
            rules: {
                name: [
                    { required: true, message: '请输入活动名称', trigger: 'blur' },
                ],
                age: [
                    { required: true, message: '请输入年龄', trigger: 'blur' },
                ],
            }
        }
    },
    created() {
        this.renderForm()
    },
    methods: {
        renderForm() {
            this.options = [
                {
                    component: 'el-input',
                    key: 'name',
                    label: '名字',
                    labelWidth: '120px',
                    props: {
                        placeholder: '请输入名字',
                        clearable: true
                    }
                },
                {
                    component: 'el-select',
                    key: 'age',
                    label: '年龄',
                    labelWidth: '120px',
                    props: {
                        placeholder: '请输入年龄',
                        clearable: true
                    },
                    data: [ // el-select, el-checkbox-group,el-radio-group 选项数据
                        {
                            label: '18',
                            value: '18'
                        },
                        {
                            label: '38',
                            value: '38'
                        }
                    ]
                },
                {
                    component: 'el-date-picker',
                    key: 'time',
                    label: '时间',
                    labelWidth: '120px',
                    width: '300px',
                    props: { // $attrs传给子组件的属性
                        placeholder: '请选择时间',
                        clearable: true,
                        type: 'daterange',
                        'range-separator': "至",
                        'start-placeholder': "开始日期",
                        'end-placeholder': "结束日期",
                        'value-format': 'yyyy-MM-dd HH:mm:ss'
                    },
                    listeners: { // $listeners传给子组件的事件
                        change: (e) => {
                            console.log("eeeeee", e)
                        }
                    }
                },
                {
                    component: 'slot',
                    slotName: 'search',
                }
            ]
        },
        handleSubmit() {
            this.$refs.schemaForm.validate(valid => {
                if(valid) {

                }
            })
        },
        handleReset() {
            this.$refs.schemaForm.resetFields()
        },
    },
}
</script>

options

参数 说明 默认值
component 组件名称
key 属性名称
label 标签文本
labelWidth 标签文本宽度 '100px'
width 表单元素宽度 '200px'
props 表单元素属性,根据element-ui提供属性
listeners 表单元素事件,根据element-ui提供事件
data 对el-select, el-checkbox-group,el-radio-group有效,选项数据

direction
表单方向,可选值:vertical和horizontal。

到目前写的例子还是很简单的配置,不能兼容各种场景,但是对于一般通用的场景还是值得一试的。后面可以根据业务场景进行不断的完善。(最后发现还是要写不少代码,感觉偷懒失败。)

参考文章:Vue2 Element Schema Form 配置式生成表单的实现

有关Vue配置化生成表单(个人笔记)的更多相关文章

  1. ruby-on-rails - Rails 编辑表单不显示嵌套项 - 2

    我得到了一个包含嵌套链接的表单。编辑时链接字段为空的问题。这是我的表格:Editingkategori{:action=>'update',:id=>@konkurrancer.id})do|f|%>'Trackingurl',:style=>'width:500;'%>'Editkonkurrence'%>|我的konkurrencer模型:has_one:link我的链接模型:classLink我的konkurrancer编辑操作:defedit@konkurrancer=Konkurrancer.find(params[:id])@konkurrancer.link_attrib

  2. ruby-on-rails - 独立 ruby​​ 脚本的配置文件 - 2

    我有一个在Linux服务器上运行的ruby​​脚本。它不使用rails或任何东西。它基本上是一个命令行ruby​​脚本,可以像这样传递参数:./ruby_script.rbarg1arg2如何将参数抽象到配置文件(例如yaml文件或其他文件)中?您能否举例说明如何做到这一点?提前谢谢你。 最佳答案 首先,您可以运行一个写入YAML配置文件的独立脚本:require"yaml"File.write("path_to_yaml_file",[arg1,arg2].to_yaml)然后,在您的应用中阅读它:require"yaml"arg

  3. ruby - 如何在 Rails 4 中使用表单对象之前的验证回调? - 2

    我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务Controller#创建Action:defcreate@service=Service.new@service_form=ServiceFormObject.new(@service)@service_form.validate(params[:service_form_object])and@service_form.saverespond_with(@service_form,location:admin_services_path)end在验证@ser

  4. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  5. Vscode+Cmake配置并运行opencv环境(Windows和Ubuntu大同小异) - 2

    之前在培训新生的时候,windows环境下配置opencv环境一直教的都是网上主流的vsstudio配置属性表,但是这个似乎对新生来说难度略高(虽然个人觉得完全是他们自己的问题),加之暑假之后对cmake实在是爱不释手,且这样配置确实十分简单(其实都不需要配置),故斗胆妄言vscode下配置CV之法。其实极为简单,图比较多所以很长。如果你看此文还配不好,你应该思考一下是不是自己的问题。闲话少说,直接开始。0.CMkae简介有的人到大二了都不知道cmake是什么,我不说是谁。CMake是一个开源免费并且跨平台的构建工具,可以用简单的语句来描述所有平台的编译过程。它能够根据当前所在平台输出对应的m

  6. LC滤波器设计学习笔记(一)滤波电路入门 - 2

    目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称

  7. 神州数码无线产品(AC+AP)配置 - 2

    注意:本文主要掌握DCN自研无线产品的基本配置方法和注意事项,能够进行一般的项目实施、调试与运维AP基本配置命令AP登录用户名和密码均为:adminAP默认IP地址为:192.168.1.10AP默认情况下DHCP开启AP静态地址配置:setmanagementstatic-ip192.168.10.1AP开启/关闭DHCP功能:setmanagementdhcp-statusup/downAP设置默认网关:setstatic-ip-routegeteway192.168.10.254查看AP基本信息:getsystemgetmanagementgetmanaged-apgetrouteAP配

  8. 计算机毕业设计ssm+vue基本微信小程序的小学生兴趣延时班预约小程序 - 2

    项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU

  9. hadoop安装之保姆级教程(二)之YARN的配置 - 2

    1.1.1 YARN的介绍 为克服Hadoop1.0中HDFS和MapReduce存在的各种问题⽽提出的,针对Hadoop1.0中的MapReduce在扩展性和多框架⽀持⽅⾯的不⾜,提出了全新的资源管理框架YARN. ApacheYARN(YetanotherResourceNegotiator的缩写)是Hadoop集群的资源管理系统,负责为计算程序提供服务器计算资源,相当于⼀个分布式的操作系统平台,⽽MapReduce等计算程序则相当于运⾏于操作系统之上的应⽤程序。 YARN被引⼊Hadoop2,最初是为了改善MapReduce的实现,但是因为具有⾜够的通⽤性,同样可以⽀持其他的分布式计算模

  10. Ruby 默认将 IRB 配置为 Pretty_Inspect - 2

    我是ruby​​的新手,正在配置IRB。我喜欢pretty-print(需要'pp'),但总是输入pp来漂亮地打印它似乎很麻烦。我想做的是默认情况下让它漂亮地打印出来,所以如果我有一个var,比如说,'myvar',然后键入myvar,它会自动调用pretty_inspect而不是常规检查。我从哪里开始?理想情况下,我将能够向我的.irbrc文件添加一个自动调用的方法。有什么想法吗?谢谢! 最佳答案 irb中默认pretty-print对象正是hirb被迫去做。Theseposts解释hirb如何将几乎所有内容转换为ascii表。虽

随机推荐