react-nativereact-nativereact native

关于React组件规范化的一些建议——UI组件篇

2017-02-06  本文已影响572人  梁相辉

一个良好的代码规范,不仅能提升web性能,提高团队协作,还便于后期维护,拓展。
“代码即文档”是我们对规范的一个愿景。下面就基于ES6的React项目配置,谈几点对React组件规范的几点建议。

我们的组件分为两大类,UI组件和container组件。

UI组件

UI组件是构成前端界面的基础单元,它们不涉及业务逻辑,无生命周期函数,只负责单纯的渲染,所有数据都通过 props 传入。

一, 咱们的UI组件大致分为两种情况,无状态和有状态

import React, {PropTypes} from 'react'
import {connect} from 'react-redux'

const dataType = {
  onExpand: PropTypes.func.isRequired,
  isOpen: PropTypes.bool
}

const List = ({ onExpand, expanded = false, childred }) => 
  <form style={ expanded ? { height: 'auto' } : { height: 0 } }>
      {children}
      <button onClick={onExpand}>Expand</button>
    </form>;

List.propTypes = dataType

export default connect(List)
import React, {PureComponent, PropTypes} from 'react' 
import {connect} from 'react-redux'

@connect(...)
export default class Listing extends PureComponent {
  static propTypes = {
    model: PropTypes.object.isRequired,
    title: PropTypes.string
  }
  
  static defaultProps = {
    model: {id:0},
    title: 'minooo'
  } 
  state = {
    isOpen: false
  };
  
  onOpen = () => {
    this.setState((preState, props) => ({isOpen: !preState.isOpen}))
  }  

  render(){ 
    const {focus} = this.state;
    ...
  }
}

二, 一般在有状态组件中可能需要定义一些方法,这些方法使用箭头函数,以此避免render中的this绑定带来的性能损耗

// bad
export default class Listing extends PureComponent {
  //...
  onClick () {
    ...
  };
  render(){ 
    return(
      <div onClick={this.onClick.bind(this)}>...</div>
    )
  }
}

// good
export default class Listing extends PureComponent {
  //...
  onClick = () => {
    ...
  };
  render(){ 
    return(
      <div onClick={this.onClick}>...</div>
    )
  }
}

三, 如果闭合标签内无子节点,则写为单标签

// bad
<i className="i-left"></i>

// good
<i className="i-left" />

四, 为了让便于阅读,请给代码合理的间隔,换行,让我们的代码像诗一样干净利索,一目了然。

// bad
const Course = ({address, tel, onClick}) =>
 <div className="bg-white pad2 overflow-h">
      <a href="javascript:;" className="ui-border-r inblock pr10" onClick={onClick}>
      <i className={"i-zuobiao32 main-color font-size-16 fl mr5 "+styles.icon}></i>
      <span className={"text-overflow-1 " +styles.address}>{address}</span>
      </a>
      <a href={"tel: "+tel}><i className={"i-dianhua32 main-color font-size-16 fr "+styles.icon}></i></a
</div>;

// good
const Course = ({address, tel, onClick}) =>
  <div className="bg-white pad2 overflow-h">
    <a
      href="javascript:;"
      className="ui-border-r inblock pr10"
      onClick={onClick}
    >
      <i className={`i-zuobiao32 main-color font-size-16 fl mr5 ${styles.icon}`} />
      <span className={"text-overflow-1 " + styles.address}>{address}</span>
    </a>

    <a href={"tel: " + tel}>
      <i className={"i-dianhua32 main-color font-size-16 fr " + styles.icon} />
    </a>
  </div>;

五,尽量避免使用数组的索引作为key的属性值,推荐使用指定的ID,原因?

// bad
{todos.map((todo, index) =>
  <Todo
    {...todo}
    key={index}
  />
)}

// good
{todos.map(todo => (
  <Todo
    {...todo}
    key={todo.id}
  />
))}

六,使用 ref 回调函数,官方ref

// bad
<Foo
  ref="myRef"
/>

// good
<Foo
  ref={(ref) => { this.myRef = ref; }}
/>

七,使用箭头函数锁定局部变量

{props.items.map((item, index) =>
  <Item
    key={item.key}
    // good
    onClick={() => this.doSomethingWith(item.name, index)}
   
    // bad
    onClick={this.doSomething.bind(null, item.name, index)}
  />
)}
上一篇 下一篇

猜你喜欢

热点阅读