sync
在vue2.4以前,父组件向子组件传值用props;子组件不能直接更改父组件传入的值,需要通过$emit触发自定义事件,通知父组件改变后的值。比较繁琐,写法如下:
//父组件 <template> <div class="parent"> <p>父组件传入子组件的值:{{name}}</p> <fieldset> <legend>子组件</legend> <child :val="name" @update="modify"> </child> </fieldset> </div> </template> <script> import Child from './Child' export default { components:{Child}, data () { return { name:'linda' } }, methods:{ modify(newVal){ this.name=newVal } } } </script> //子组件 <template> <label class="child"> 输入框: <input :value=val @input="$emit('update',$event.target.value)"/> </label> </template> <script> export default { props:['val'] } </script>
vue2.4以后的写法明显舒服许多,上面同样的功能,直接上代码
//父组件 <template> <div class="parent"> <p>父组件传入子组件的值:{{name}}</p> <fieldset> <legend>子组件</legend> <child :val.sync="name"> </child> </fieldset> </div> </template> <script> import Child from './Child' export default { components:{Child}, data () { return { name:'linda' } } } </script> //子组件 <template> <label class="child"> 输入框: <input :value=val @input="$emit('update:val',$event.target.value)"/> </label> </template> <script> export default { props:['val'] } </script>
写法上简化了一部分,很明显父组件不用再定义方法检测值变化了。其实只是对以前的$emit方式的一种缩写,.sync其实就是在父组件定义了一update:val方法,来监听子组件修改值的事件。
$attrs
想象一下,你打算封装一个自定义input组件——MyInput,需要从父组件传入type,placeholder,title等多个html元素的原生属性。此时你的MyInput组件props如下:
props:['type','placeholder','title',...]
很丑陋不是吗?$attrs专门为了解决这种问题而诞生,这个属性允许你在使用自定义组件时更像是使用原生html元素。比如:
//父组件 <my-input placeholder="请输入你的姓名" type="text" title="姓名" v-model="name"/>
my-input的使用方式就像原生的input一样。而MyInput并没有设置props,如下
<template> <div> <label>输入框:</label><input v-bind="$attrsAll" @input="$emit('input',$event.target.value)"/> </div> </template> <script> export default { inheritAttrs:false, computed: { $attrsAll() { return { value: this.$vnode.data.model.value, ...this.$attrs } } } } </script>
基础扫盲
v-model是v-bind:value和v-on:input的简写,所以在父组件你完全可以直接写 :value="name",@input="val => name = val"。查看文档
疑难
引用下vue的官方api中对$attrs的说明
$attrs包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)
比较迷惑的一点是给子组件设置:value="name"相当于给子组件设置props:['value'],所以在MyInput中直接从$attrs获取不到value,需要重新包装$attrsAll,添加value属性。所以子组件还有下面写法,我倾向于这种写法,因为它更优雅
<template> <div> <label>输入框:</label><input v-bind="$attrs" :value="value" @input="$emit('input',$event.target.value)"/> </div> </template> <script> export default { inheritAttrs:false, props:['value'] } </script>
$listener
同上面$attrs属性一样,这个属性也是为了在自定义组件中使用原生事件而产生的。比如要让前面的MyInput组件实现focus事件,直接这么写是没用的
<my-input @focus="focus" placeholder="请输入你的姓名" type="text" title="姓名" v-model="name"/>
必须要让focus事件作用于MyInput组件的input元素上,最终的MyInput源码如下:
<template> <div> <label>输入框:</label><input v-bind="$attrsAll" v-on="$listenserAll"/> </div> </template> <script> export default { inheritAttrs:false, props:['value'], computed:{ $attrsAll() { return { value: this.value, ...this.$attrs } }, $listenserAll(){ return Object.assign( {}, this.$listeners, {input:(event) => this.$emit('input',event.target.value)}) } } } </script>
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]