姿势很重要,末尾有福利
vue-antd-ui开源了一段时间后,收到了一些反馈,尤其是Form组件上线后,很多用户对JSX的使用感到迷惑和不习惯,为此专门介绍下Vue JSX的使用姿势及注意事项。
Form组件的自动收集校验功能需要在JSX下使用,当然如果不需要自动收集校验,你依然可以使用template
Vue 推荐在绝大多数情况下使用template来创建你的HTML。然而在一些场景中,你真的需要 JavaScript 的完全编程的能力,就需要使用render函数,它比 template 更接近编译器。但是复杂的render函数书写异常痛苦,好在官方提供了一个Babel 插件,可以将更接近于模板语法的JSX转译成JavaScript。
使用过React的同学对JSX肯定不陌生,但是Vue的JSX写法和React的还是有一些区别。
React中父子之间传递的所有数据都是属性,即所有数据均挂载在props下(style, className, children, value, onChange等等)。
Vue则不然,仅仅属性就有三种:组件属性props,普通html属性attrs,Dom属性domProps。
接下来我们通过一个示例来详细解释他们的区别:
本文代码可在codesandbox查看运行
const ButtonCounter = {
name: "button-counter",
props: ["count"],
methods: {
onClick() {
this.$emit("change", this.count + 1);
}
},
render() {
return (
<button onClick={this.onClick}>You clicked me {this.count} times.</button>
);
}
};
export default {
name: "button-counter-container",
data() {
return {
count: 0
};
},
methods: {
onChange(val) {
this.count = val;
}
},
render() {
const { count, onChange } = this;
return (
<div>
<ButtonCounter
style={{ marginTop: "10px" }}
count={count}
type="button"
onChange={onChange}
/>
<ButtonCounter
style={{ marginTop: "10px" }}
count={count}
type="button"
domPropsInnerHTML={`hello ${this.count}.`}
onChange={onChange}
/>
</div>
);
}
};
复制代码
组件属性props:指组件声明的属性,即上述示例中声明的props: ['count']。
普通html属性attrs: __指组件未声明的属性,即上述示例中的type="button",该属性默认会直接挂载到组件根节点的上,如果不需要挂载到根节点,可声明 inheritAttrs: false。
Dom属性domProps:指的Dom属性,如上述示例中的innerHTML,它会覆盖组件内部的children, 这类属性我们一般很少使用到。
同样事件属性也分了两种:on nativeOn
那么问题来了,组件是如何区分各类属性的呢?
答:正则则则...... ?,babel-plugin-transform-vue-jsx插件会通过正则匹配的方式在编译阶段将书写在组件上属性进行“分类”。 onXXX的均被认为是事件,nativeOnXXX是原生事件,domPropsXXX是Dom属性。
class,staticClass,style,key,ref,refInFor,slot,scopedSlots这些被认为是顶级属性,至于我们属性声明的props,以及html属性attrs,不需要加前缀,插件会将其统一分类到attrs属性下,然后在运行阶段根据是否在props声明来决定属性归属(即属于props还是attrs)。
在编译阶段通过正则来区分,毕竟不是很严谨,那么根据以上分类规则会有哪些问题呢?
第一、属性分类是编译阶段进行的分类,那么对于动态属性如何划分分类?
在React中所有属性都是顶级属性,直接使用{...props}就可以了,但是在Vue中,你需要明确该属性所属的分类,如一个动态属性value和事件change,你可以使用如下方式(延展属性)传递:
const dynamicProps = {
props: {},
on: {},
}
if(haValue) dynamicProps.props.value = value
if(hasChange) dynamicProps.on.change = onChange
<Dynamic {...dynamicProps} />
复制代码
当然你可以混合着使用:
<Dynamic {...dynamicProps} style="color: red"/>
复制代码
先别高兴太早,如果你没有深入使用过Vue JSX,不建议你使用混合方式,因为Vue会对其进行属性合并,至于合并的规则官方也并没有详细的文档,文档中有一处示例,我在这再举一个例子:
const dynamicProps2 = { on: { change: onChange2 } };
<Dynamic
{...{ on: { change: onChange1 } }}
{...dynamicProps2}
onChange={onChange3}
/>
复制代码
上例中的onChange1、onChange2、onChange3都会触发,而你想要的可能仅仅是onChange3。其它属性的合并规则我就不一一列举了,总之,我不建议你使用混合方式,除非你及你的团队其他小伙伴对其规则了解的足够透彻。
注:理想情况你不应该需要动态属性,在业务开发中也比较少的使用动态属性,但如果你尝试开发一些通用性比较强的组件,就很难逃过动态属性的使用。
第二、如果声明的属性就是onXXX怎么处理?
首先我不建议你这么做,但如果真的需要,你必须明确该属性的分类,如下所示:
<div>
<Dynamic value="获取到value,然而并不能获取到onXXX" onXXX="?" />
<Dynamic {...{ props: { onXXX: "获取到onXXX,但不建议使用" } }} />
</div>
复制代码
第三、函数式组件的props如何处理?
关于函数式组件的相关概念可查看官方文档,文档中有如下一段内容:
注意:在 2.3.0 之前的版本中,如果一个函数式组件想要接受 props,则 props 选项是必须的。在 2.3.0 或以上的版本中,你可以省略 props 选项,所有组件上的属性都会被自动解析为 props。
props和attrs之间存在着很微妙的关系,在普通组件中,只要明确声明的属性会被划分到props分类中,剩下的均在attrs中。而对于函数式组件,只要省略了props选项,传参时不管是否明确分类,最终context.props获取到的都是全部属性,如果你需要获取明确的分类情况,可以在context.data下查看。 总之,在函数式组件中,推荐省略props选项。
第四、指令是否还可用?
很不幸的告诉你,大多数指令并不能在JSX中使用,对于原生指令,只有v-show是支持的。 大部分指令在JSX中可以使用表达式来替代,如:条件运算符(?:)替代v-if、array.map替代v-for 对于自定义的指令,可以使用如下方式使用:
const directives = [
{ name: 'my-dir', value: 123, modifiers: { abc: true } }
]
return <div {...{ directives }}/>
复制代码
更多Vue JSX的知识,可以查看官方文档。
总结:
说了那么多,其实只要记住一点,尽量使用明确分类的方式传递属性,而不是要babel插件帮你分类及合并属性。
最后汇报下vue-antd-ui的进度,目前组件数量48个,相较react版,还有List、TreeSelect、Metion、Carousel没有开发,再过去的一段时间里,我们更多的去完善组件单元测试,测试覆盖率83%,单测数量610个,接下来测试依然是我们的主要工作。
质量永远要比数量重要的多。
欢迎star,欢迎pr。