最佳编码方式
-
最佳编码方式
经过较长时间的探索与验证,目前 Taro 在微信小程序端是采用依托于小程序原生自定义组件系统来设计实现 Taro 组件化的,所以目前小程序端的组件化会受到小程序原生组件系统的限制,而同时为了实现以 React 方式编写代码的目标,Taro 本身做了一些编译时以及运行时的处理,这样也带来了一些值得注意的约束,所以有必要阐述一下 Taro 编码上的最佳实践。
组件样式说明
微信小程序的自定义组件样式默认是不能受外部样式影响的,例如在页面中引用了一个自定义组件,在页面样式中直接写自定义组件元素的样式是无法生效的。这一点,在 Taro 中也是一样,而这也是与大家认知的传统 Web 开发不太一样。
给组件设置
defaultProps
在微信小程序端的自定义组件中,只有在
properties
中指定的属性,才能从父组件传入并接收Component({ properties: { myProperty: { // 属性名 type: String, // 类型(必填),目前接受的类型包括:String, Number, Boolean, Object, Array, null(表示任意类型) value: '', // 属性初始值(可选),如果未指定则会根据类型选择一个 observer: function (newVal, oldVal, changedPath) { // 属性被改变时执行的函数(可选),也可以写成在 methods 段中定义的方法名字符串, 如:'_propertyChange' // 通常 newVal 就是新设置的数据, oldVal 是旧数据 } }, myProperty2: String // 简化的定义方式 } ... })
而在 Taro 中,对于在组件代码中使用到的来自
props
的属性,会在编译时被识别并加入到编译后的properties
中,暂时支持到了以下写法this.props.property const { property } = this.props const property = this.props.property
但是一千个人心中有一千个哈姆雷特,不同人的代码写法肯定也不尽相同,所以 Taro 的编译肯定不能覆盖到所有的写法,而同时可能会有某一属性没有使用而是直接传递给子组件的情况,这种情况是编译时无论如何也处理不到的,这时候就需要大家在编码时给组件设置
defaultProps
来解决了。组件设置的
defaultProps
会在运行时用来弥补编译时处理不到的情况,里面所有的属性都会被设置到properties
中初始化组件,正确设置defaultProps
可以避免很多异常的情况的出现。组件传递函数属性名以
on
开头在 Taro 中,父组件要往子组件传递函数,属性名必须以
on
开头// 调用 Custom 组件,传入 handleEvent 函数,属性名为 onTrigger class Parent extends Component { handleEvent () { } render () { return ( <Custom onTrigger={this.handleEvent}></Custom> ) } }
这是因为,微信小程序端组件化是不能直接传递函数类型给子组件的,在 Taro 中是借助组件的事件机制来实现这一特性,而小程序中传入事件的时候属性名写法为
bindmyevent
或者bind:myevent
<!-- 当自定义组件触发 myevent 事件时,调用 onMyEvent 方法 --> <component-tag-name bindmyevent="onMyEvent" /> <!-- 或者可以写成 --> <component-tag-name bind:myevent="onMyEvent" />
所以 Taro 中约定组件传递函数属性名以
on
开头,同时这也和内置组件的事件绑定写法保持一致了。小程序端不要在组件中打印传入的函数
前面已经提到小程序端的组件传入函数的原理,所以在小程序端不要在组件中打印传入的函数,因为拿不到结果,但是
this.props.onXxx && this.props.onXxx()
这种判断函数是否传入来进行调用的写法是完全支持的。小程序端不要将在模板中用到的数据设置为
undefined
由于小程序不支持将 data 中任何一项的 value 设为
undefined
,在 setState 的时候也请避免这么用。你可以使用 null 来替代。小程序端不要在组件中打印
this.props.children
在微信小程序端是通过
<slot />
来实现往自定义组件中传入元素的,而 Taro 利用this.props.children
在编译时实现了这一功能,this.props.children
会直接被编译成<slot />
标签,所以它在小程序端属于语法糖的存在,请不要在组件中打印它。支持 props 传入 JSX
自
1.1.9
开始支持支持 props 传入 JSX,但是元素传入 JSX 的属性名必须以
render
开头例如,子组件写法
class Dialog extends Component { render () { return ( <View className='dialog'> <View className='header'> {this.props.renderHeader} </View> <View className='body'> {this.props.children} </View> <View className='footer'> {this.props.renderFooter} </View> </View> ) } }
父组件调用子组件是传入 JSX
class App extends Component { render () { return ( <View className='container'> <Dialog renderHeader={ <View className='welcome-message'>Welcome!</View> } renderFooter={ <Button className='close'>Close</Button> } > <View className="dialog-message"> Thank you for using Taro. </View> </Dialog> </View> ) } }
组件属性传递注意
不要以
id
、class
、style
作为自定义组件的属性与内部 state 的名称,因为这些属性名在微信小程序小程序中会丢失。组件
state
与props
里字段重名的问题不要在
state
与props
上用同名的字段,因为这些被字段在微信小程序中都会挂在data
上。小程序中页面生命周期
componentWillMount
不一致问题由于微信小程序里页面在
onLoad
时才能拿到页面的路由参数,而页面 onLoad 前组件都已经attached
了。因此页面的componentWillMount
可能会与预期不太一致。例如:// 错误写法 render () { // 在 willMount 之前无法拿到路由参数 const abc = this.$router.params.abc return <Custom adc={abc} /> } // 正确写法 componentWillMount () { const abc = this.$router.params.abc this.setState({ abc }) } render () { // 增加一个兼容判断 return this.state.abc && <Custom adc={abc} /> }
对于不需要等到页面 willMount 之后取路由参数的页面则没有任何影响。
组件的
constructor
与render
提前调用很多细心的开发者应该已经注意到了,在 Taro 编译到小程序端后,组件的
constructor
与render
默认会多调用一次,表现得与 React 不太一致。这是因为,Taro 的组件编译后就是小程序的自定义组件,而小程序的自定义组件的初始化时是可以指定
data
来让组件拥有初始化数据的。开发者一般会在组件的constructor
中设置一些初始化的state
,同时也可能会在render
中处理state
与props
产生新的数据,在 Taro 中多出的这一次提前调用,就是为了收集组件的初始化数据,给自定义组件提前生成data
,以保证组件初始化时能带有数据,让组件初次渲染正常。所以,在编码时,需要在处理数据的时候做一些容错处理,这样可以避免在
constructor
与render
提前调用时出现由于没有数据导致出错的情况。JS 编码必须用单引号
在 Taro 中,JS 代码里必须书写单引号,特别是 JSX 中,如果出现双引号,可能会导致编译错误。
环境变量
process.env
的使用不要以解构的方式来获取通过
env
配置的process.env
环境变量,请直接以完整书写的方式process.env.NODE_ENV
来进行使用// 错误写法,不支持 const { NODE_ENV = 'development' } = process.env if (NODE_ENV === 'development') { ... } // 正确写法 if (process.env.NODE_ENV === 'development') { }
使用
this.$componentType
来判断当前 Taro.Component 是页面还是组件this.$componentType
可能取值分别为PAGE
和COMPONENT
,开发者可以根据此变量的取值分别采取不同逻辑。