目录
效果:

这里是后台管理系统的三级选择器,只有当第一级分类选中属性时,第二个才能选,以此类推。
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
}
}
},
理解:当数据更新了,在dom中渲染后,自动执行该函数。
Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新。$nextTick 是在下次 DOM 更新循环结束之后执行延迟回调,在修改数据之后使用 $nextTick,则可以在回调中获取更新后的 DOM。
解决数据没有被双向绑定我们可以使用 vm.$set 实例方法,该方法是全局方法 Vue.set 的一个别名。理解就是把数据变成响应式。
this.$set(原数组, 索引值, 需要赋的值)
在Vue中,我们不用获取dom节点,元素绑定ref之后,直接通过this.$refs即可调用,这样可以减少获取dom节点的消耗。
ref特性就是为元素或子组件赋予一个ID引用,通过this.$refs.refName来访问元素或子组件的实例
some方法:它会迭代数组的每一个元素,直到函数返回true。
some() 方法用于检测数组中的元素是否满足指定条件(函数提供)。
some() 方法会依次执行数组的每个元素:
如果有一个元素满足条件,则表达式返回true , 剩余的元素不会再执行检测。
如果没有满足条件的元素,则返回false。
注意:some() 不会对空数组进行检测。
注意:some() 不会改变原始数组
filter方法:对数组中的每个元素给定函数,返回该函数会返回true的元素组成的数组。
这里的用到了lodash中的深拷贝,因为当修改属性值时,需要把已有的属性值再复制一份到修改页面。这里不能用浅拷贝,因为浅拷贝会使原数据跟着改变。
当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){
}
}
}
我得到了一个包含嵌套链接的表单。编辑时链接字段为空的问题。这是我的表格: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
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用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
我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为
查看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
我发现ActiveRecord::Base.transaction在复杂方法中非常有效。我想知道是否可以在如下事务中从AWSS3上传/删除文件:S3Object.transactiondo#writeintofiles#raiseanexceptionend引发异常后,每个操作都应在S3上回滚。S3Object这可能吗?? 最佳答案 虽然S3API具有批量删除功能,但它不支持事务,因为每个删除操作都可以独立于其他操作成功/失败。该API不提供任何批量上传功能(通过PUT或POST),因此每个上传操作都是通过一个独立的API调用完成的
所以我在关注Railscast,我注意到在html.erb文件中,ruby代码有一个微弱的背景高亮效果,以区别于其他代码HTML文档。我知道Ryan使用TextMate。我正在使用SublimeText3。我怎样才能达到同样的效果?谢谢! 最佳答案 为SublimeText安装ERB包。假设您安装了SublimeText包管理器*,只需点击cmd+shift+P即可获得命令菜单,然后键入installpackage并选择PackageControl:InstallPackage获取包管理器菜单。在该菜单中,键入ERB并在看到包时选择
我试图在索引页中创建一个超链接,但它没有显示,也没有给出任何错误。这是我的index.html.erb代码。ListingarticlesTitleTextssss我检查了我的路线,我认为它们也没有问题。PrefixVerbURIPatternController#Actionwelcome_indexGET/welcome/index(.:format)welcome#indexarticlesGET/articles(.:format)articles#indexPOST/articles(.:format)articles#createnew_articleGET/article
我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务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
我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c