草庐IT

【Vue-Element UI】三级表单可选框以及显示、修改和删除表单属性

翘阳啦 2024-04-07 原文

目录

三级表单可选框

静态界面

收集数据 

获取数据

属性值内容显示 

一些知识点

(1)Vue.nextTick()

(2)$set

(3) ref

(4)一些数组方法

(5)深拷贝

(6)@blur


三级表单可选框

效果:

这里是后台管理系统的三级选择器,只有当第一级分类选中属性时,第二个才能选,以此类推

静态界面

Element ui里面的表单选择器:组件 | Element

这里用到的Form属性: 

inline属性可以让表单域变为行内的表单域    inline为true代表的是行内表单,代表一行可以放置多个表单元素

model属性为表单的数据对象

这里用到的Form-Item属性 :

label属性:标签的文本

    <el-form :inline="true" class="demo-form-inline" :model="cForm">
      <el-form-item label="一级分类">
        <el-select placeholder="请选择" value="" v-model="cForm.category1Id" @change="handle1">
          <el-option :label="c1.name" :value="c1.id" v-for="(c1,index) in list1" :key="c1.id"></el-option>
        </el-select>
      </el-form-item>
      <el-form-item label="二级分类">
        <el-select placeholder="请选择" value="" v-model="cForm.category2Id" @change="handle2">
          <el-option :label="c2.name" :value="c2.id" v-for="(c2,index) in list2" :key="c2.id"></el-option>
        </el-select>
      </el-form-item>
            <el-form-item label="三级分类">
        <el-select placeholder="请选择" value="" v-model="cForm.category3Id" @change="handle3">
          <el-option :label="c3.name" :value="c3.id" v-for="(c3,index) in list3" :key="c3.id"></el-option>
        </el-select>
      </el-form-item>
    </el-form>

收集数据 

需要调取服务器的数据,利用v-model,把三个分类的数据先在data中收集空数组:

 data() {
    return {
        //一级分类的数据
        list1:[],
        //二级分类的数据
        list2:[],
        //三级分类的数据
        list3:[],
        //收集相应一级二级三级分类的id
        cForm:{
            category1Id:'',
            category2Id:'',
            category3Id:''
        }
    };
  },

三级分类的内容都会携带id

所以利用cForm在data中收集了三级分类的不同内容的id。

获取数据

当组件挂载完毕:向服务器发请求,获取相应的数据

  mounted(){
      this.getCategory1List()
  },
  components: {},
  methods: {
    //获取一级分类数据方法
    async getCategory1List(){
        //获取一级分裂的数据 不需要携带参数
        let result = await this.$API.attr.reqCategory1List()
        if(result.code==200){
            this.list1=result.data
        }
    },
    
  }

当三级分类改变时,要响应相应的数据,当选择第一级分类的数据时,需要清除后面两级的数据,并且让它们的id置空,以此类推。

//一级分类select事件回调
    async handle1(){
        //当一级分类改变时,清除数据
        this.list2=[]
        this.list3=[]
        this.cForm.category2Id=''
        this.cForm.category3Id=''
        //解构出一级分类的id
        const {category1Id} = this.cForm
        this.$emit('getCategoryId',{categoryId:category1Id,level:1})
        //通过一级分类的id 获取二级分类的数据
        let result = await this.$API.attr.reqCategory2List(category1Id)
        //console.log(result)
        if(result.code==200){
            this.list2=result.data
        }
    },
    async handle2(){
        //当二级分类改变时,清除数据
        this.list3=[]
        this.cForm.category3Id=''
        //解构出一级分类的id
        const {category2Id} = this.cForm
        this.$emit('getCategoryId',{categoryId:category2Id,level:2})
        //通过一级分类的id 获取二级分类的数据
        let result = await this.$API.attr.reqCategory3List(category2Id)
        //console.log(result)
        if(result.code==200){
            this.list3=result.data
        }
    },
    handle3(){
        //解构出三级分类的id
        const {category3Id} = this.cForm
        this.$emit('getCategoryId',{categoryId:category3Id,level:3})
    }

属性值内容显示 

把上面的三级框封装成一个组件 

<el-card style="margin:20px 0px">
            <CategorySelect @getCategoryId="getCategoryId"/>
</el-card>

上边的三级分类选择完成后,就会在下面的列表中显示数据。

当点击添加属性按钮时,就会隐藏上面的表格,出现另一个表格

这里两个el-card 利用v-show来显示和隐藏,当isShowTable为true时,显示第一个界面,反之显示另一个。

<el-card>
           <div v-show="isShowTable">
            <el-button type="primary" icon="el-icon-plus" :disabled="!category3Id" @click="addAttr" >添加属性</el-button>
            <el-table :data="attrList" style="width:100%" border>
               <el-table-column prop="prop"  type="index" label="序号" width="80" align="center">
                      
               </el-table-column>
               <el-table-column prop="attrName" label="属性名称" width="150">
                      
               </el-table-column>
               <el-table-column prop="prop" label="属性值列表" width="width">
                      <template slot-scope="{row,$index}">
                          <el-tag type="success" v-for="(attrValue,index) in row.attrValueList" :key="attrValue.id" style="margin-right:20px">{{attrValue.valueName}}</el-tag>
                      </template>
               </el-table-column>
                <el-table-column prop="prop" label="操作" width="150">
                      <template slot-scope="{row,$index}">
                          <el-button type="warning"  icon="el-icon-edit" size="mini"  @click="updateAttr(row)"></el-button>
                          <el-button type="danger"  icon="el-icon-delete" size="mini" ></el-button>
                          
                      </template>                      
               </el-table-column>
            </el-table>
           </div>
           <!-- 添加属性或者修改属性 -->
           <div v-show="!isShowTable">
             <el-form :inline="true" ref="form" label-width="80px" :model="attrInfo">
                <el-form-item label="属性名">
                   <el-input placeholder="请输入属性名" v-model="attrInfo.attrName"></el-input>
                </el-form-item>
             </el-form>
             <el-button type="primary" icon="el-icon-plus" @click="addAttrValue" :disabled="!attrInfo.attrName">添加属性值</el-button> 
             <el-button @click="isShowTable=true">取消</el-button>
             <el-table style="width:100%; margin:20px 0px" border :data="attrInfo.attrValueList">
                  <el-table-column align="center" type="index" label="序号" width="80"></el-table-column>
                  <el-table-column align="center" label="属性值名称">
                     <template slot-scope="{row,$index}">
                        <!-- 这里需要span和input来回切换 查看模式:显示span  编辑模式:显示input-->
                        <el-input v-model="row.valueName" placeholder="请输入属性值名称" size="mini" v-if="row.flag" @blur="toLook(row)" @keyup.native.enter="row.flag=false" :ref="$index"></el-input>
                        <span v-else @click="toEdit(row,$index)" style="display:block">{{row.valueName}}</span>
                     </template>
                  </el-table-column>
                  <el-table-column align="center" label="操作">
                     <template slot-scope="{row,$index}">   
                        <!-- 气泡确认框    onConfirm为点击正确按钮的事件 注意elm ui的版本 这个是2.13                 -->
                        <el-popconfirm :title="`确定删除${row.valueName}吗`" @onConfirm="deleteAttrValue">
                            <el-button  type="danger" icon="el-icon-delete" size="mini" slot="reference" ></el-button>
                        </el-popconfirm>
                     </template>                    
                  </el-table-column>
             </el-table>
             <el-button type="primary" @click="addOrUpdateAttr">保存</el-button> 
             <el-button @click="isShowTable=true">取消</el-button>
             
           </div>
        </el-card>

需要收集的数据

data() {
        return {
            category1Id:'',
            category2Id:'',
            category3Id:'',
            //存储平台属性的地方
            attrList:[],
            //这个属性控制table的显示与隐藏
            isShowTable:true,
            //收集新增属性|修改属性使用的
            attrInfo:{            
                attrName: "",  //属性名
                attrValueList: [      //属性名中的属性值,因为属性值可以是多个,因此需要的是数组

                ],
                categoryId: 0,   //三级分类的id
                categoryLevel: 0,  //因为服务器也需要区分几级id
              }
            }
        },

一些知识点

(1)Vue.nextTick()

理解:当数据更新了,在dom中渲染后,自动执行该函数。

Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新。$nextTick 是在下次 DOM 更新循环结束之后执行延迟回调,在修改数据之后使用 $nextTick,则可以在回调中获取更新后的 DOM。

(2)$set

解决数据没有被双向绑定我们可以使用 vm.$set 实例方法,该方法是全局方法 Vue.set 的一个别名。理解就是把数据变成响应式。

this.$set(原数组, 索引值, 需要赋的值)

(3) ref

在Vue中,我们不用获取dom节点,元素绑定ref之后,直接通过this.$refs即可调用,这样可以减少获取dom节点的消耗。 

ref特性就是为元素或子组件赋予一个ID引用,通过this.$refs.refName来访问元素或子组件的实例

(4)一些数组方法

 some方法:它会迭代数组的每一个元素,直到函数返回true。

some() 方法用于检测数组中的元素是否满足指定条件(函数提供)。
some() 方法会依次执行数组的每个元素:
如果有一个元素满足条件,则表达式返回true , 剩余的元素不会再执行检测。
如果没有满足条件的元素,则返回false。
注意:some() 不会对空数组进行检测。
注意:some() 不会改变原始数组 

 filter方法:对数组中的每个元素给定函数,返回该函数会返回true的元素组成的数组。

(5)深拷贝

这里的用到了lodash中的深拷贝,因为当修改属性值时,需要把已有的属性值再复制一份到修改页面。这里不能用浅拷贝,因为浅拷贝会使原数据跟着改变。

(6)@blur

当input框失去焦点时触发 

//按需引入lodash当中的深拷贝
import cloneDeep from "lodash/cloneDeep"

methods: {
        //自定义事件的回调
        getCategoryId({categoryId,level}){
             if(level==1){
                this.category1Id = categoryId
                this.category2Id=''
                this.category3Id=''
             }else if(level==2){
                this.category2Id = categoryId
                this.category3Id=''
             }else{
                this.category3Id = categoryId
                //发请求获取品牌属性
                this.getAttrList()
             }
        },
        //获取平台属性的数据
        //当用户确定三级分类的数据时候,可以向服务器发请求获取平台啊属性进行展示
        async getAttrList(){
            const {category1Id,category2Id,category3Id} =this
            let result =await this.$API.attr.reqAttrList(category1Id,category2Id,category3Id)
            // console.log(result)
            if(result.code==200){
                this.attrList = result.data
            }
        },
        //添加属性值
        addAttrValue(){
            //向属性值的数组内添加元素
            //attrId:相应属性的id,目前是添加属性的操作,还没有相应的属性的id,目前而言带给服务器的id为undefined
            //valueName:相应的属性值的名称
            this.attrInfo.attrValueList.push({
                attrId:this.attrInfo.id,  //对于修改某一个属性的时候,可以在已有的属性值基础上要带上id 
                valueName:'',
                flag:true  //给每一个属性值添加一个标记,用于切换查看和编辑模式,好处是每一个属性值可以控制自己的模式切换
            })
             this.$nextTick(()=>{
                this.$refs[this.attrInfo.attrValueList.length-1].focus()
            })
        },
        //点击添加属性
        addAttr(){
            //切换table的显示与隐藏
           this.isShowTable=false
           //清除数据
           this.attrInfo={
              attrName: "",  //属性名
              attrValueList: [      //属性名中的属性值,因为属性值可以是多个,因此需要的是数组

                ],
              categoryId: this.category3Id,   //三级分类的id
              categoryLevel: 0,  //因为服务器也需要区分几级id
           }
        },
        //修改属性
        updateAttr(row){
            //console.log(row)
             this.isShowTable=false
             //将选中的属性赋值给attrInfo
             //由于数据结构当中存在对象里面套数组,数组里面套对象,要用深拷贝解决此类问题
             //this.attrInfo = {...row}
             this.attrInfo = cloneDeep(row)
             //在修改某个属性的时候,将相应的属性值元素添加上flag这个标记
             this.attrInfo.attrValueList.forEach(item=>{
                   //这样书写也可以给属性值添加flag,但是会发现视图不会跟着变化(因为flag不是响应式数据)
                   //Vue无法探测普通的property,这样书写的属性并非响应式属性
                   //第一个参数:对象  第二个参数:添加新的响应式属性  第三个参数:新的属性的属性值
                   this.$set(item,'flag',false)

             })
        },
        //失去焦点的事件:切换为查看模式,展示span
        toLook(row){
            //如果属性值为空不能作为新的属性值,需要给用户提示,让他输入新的属性值
            if(row.valueName.trim()==''){
                this.$message('请输入一个正确的属性值')
                return
            }
            //新增的属性值不能与已有的属性值重复
            let isRepeat = this.attrInfo.attrValueList.some((item)=>{
                //需要将row从数组里面判断的时候去除
                //row为最新的新增的属性值,数组的最后一项
                if(row!==item){
                    return row.valueName == item.valueName
                }
            })
            if(isRepeat) return
            //row是当前用户添加的最新的属性值
            row.flag=false
        },
        //点击span的回调,变为编辑模式
        toEdit(row,index){
            row.flag=true
            //获取input节点,实现自动聚焦
            //需要注意的是  点击span时,切换为input变更为编辑模式,但是需要注意的是对于浏览器而言,页面的重绘重排是耗时间的
            //点击span时,重绘重排一个input他是耗时间的,因此我们不可能一开始就立马获取到input
            //$nextTick  当节点渲染完毕,会执行一次
            this.$nextTick(()=>{
                this.$refs[index].focus()
            })
        },
        //气泡确认框的确定按钮的回调
        deleteAttrValue(index){
            //当前删除属性值的操作是不需要发送请求的
             this.attrInfo.attrValueList.splice(index,1)

        },
        //保存按钮
        async addOrUpdateAttr(){
            //整理参数1:如果用户添加很多属性值,且属性值为空的不应该提交给服务器
            //提交给服务器数据当中不应该出现flag字段
            this.attrInfo.attrValueList = this.attrInfo.attrValueList.filter(item=>{
                //过滤掉属性值不为空的
                if(item.valueName!=""){
                    //删除掉flag属性
                    delete item.flag
                    return true
                }
            })
            try{
                //发请求
               await this.$API.attr.reqAddOrUpdateAttr(this.attrInfo)
               //展示平台属性的页面
               this.isShowTable = true
               //提示消息
               this.$message({type:'success',message:'保存成功'})
               //保存成功以后需要再次获取平台属性进行展示
               this.getAttrList()
            }catch(error){
            
            }
        }
    }

有关【Vue-Element UI】三级表单可选框以及显示、修改和删除表单属性的更多相关文章

  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 - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  3. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  4. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  5. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

    查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

  6. ruby - 我可以使用 aws-sdk-ruby 在 AWS S3 上使用事务性文件删除/上传吗? - 2

    我发现ActiveRecord::Base.transaction在复杂方法中非常有效。我想知道是否可以在如下事务中从AWSS3上传/删除文件:S3Object.transactiondo#writeintofiles#raiseanexceptionend引发异常后,每个操作都应在S3上回滚。S3Object这可能吗?? 最佳答案 虽然S3API具有批量删除功能,但它不支持事务,因为每个删除操作都可以独立于其他操作成功/失败。该API不提供任何批量上传功能(通过PUT或POST),因此每个上传操作都是通过一个独立的API调用完成的

  7. ruby-on-rails - 使用 Sublime Text 3 突出显示 HTML 背景语法中的 ERB? - 2

    所以我在关注Railscast,我注意到在html.erb文件中,ruby代码有一个微弱的背景高亮效果,以区别于其他代码HTML文档。我知道Ryan使用TextMate。我正在使用SublimeText3。我怎样才能达到同样的效果?谢谢! 最佳答案 为SublimeText安装ERB包。假设您安装了SublimeText包管理器*,只需点击cmd+shift+P即可获得命令菜单,然后键入installpackage并选择PackageControl:InstallPackage获取包管理器菜单。在该菜单中,键入ERB并在看到包时选择

  8. ruby-on-rails - link_to 不显示任何 rails - 2

    我试图在索引页中创建一个超链接,但它没有显示,也没有给出任何错误。这是我的index.html.erb代码。ListingarticlesTitleTextssss我检查了我的路线,我认为它们也没有问题。PrefixVerbURIPatternController#Actionwelcome_indexGET/welcome/index(.:format)welcome#indexarticlesGET/articles(.:format)articles#indexPOST/articles(.:format)articles#createnew_articleGET/article

  9. 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

  10. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

随机推荐