3

我正在研究将 ESLint(Airbnb 配置)与 React / Redux 一起使用的纪律。

下面的代码是一个标准类型的 React / Redux 类,我编写它是为了享受处理 Airbnb 的所有 linting 规则。

通过了解 Airbnb 的配置偏好,我已经解决了大多数 linting 错误,但是,有几个方面我目前不了解。这些是:

  1. 整个方法 renderTableHeader() 用红色标出,ESLint 告诉我:

[eslint] 类方法“renderTableHeader”应使用“this”。(class-methods-use-this) (JSX 属性) className: string 在此处输入图像描述

其他方法都没有这种掉毛问题

  1. 我将一个对象从 Redux 状态连接到 props(包含许多我需要在此类中迭代的键/对象)。ESLint 似乎不喜欢我将对象从状态连接到道具......给我的信息:

[eslint] Prop 类型object被禁止 (react/forbid-prop-types) import PropTypes

在此处输入图像描述

我的状态对象是一个包含许多引用用户对象的键的对象 - 所以我需要users成为一个对象。这是不好的做法吗?如果是我将如何连接对象以映射而不出现此 lint 消息?

在此先感谢...这是我的课程:

import React, { Component } from 'react';
import { bindActionCreators } from 'redux';
import { connect } from 'react-redux';
import PropTypes from 'prop-types';
import _ from 'lodash';
import { getUsers } from '../actions/getUsers';
import setUserMarketingPref from '../actions/setUserMarketingPref';
import { setFilter } from '../actions/setFilter';
import TableRow from '../components/TableRow';
import '../styles/styles.css';

class AppContainer extends Component {
  componentDidMount() {
    this.props.getUsers();
  }

  renderTableHeader() {
    return (
      <div className="table__header">
        <div className="table__header--name">Name</div>
        <div className="table__header--gender">Gender</div>
        <div className="table__header--region">Region</div>
      </div>
    );
  }

  renderTable() {
    const { users, filterMode } = this.props;
    const usersView = filterMode ? _.omitBy(users, user => !user.checked) : users;
    return _.map(usersView, user => (
      <TableRow
        user={user}
        key={user.id}
        setUserMarketingPref={_.debounce(() => this.props.setUserMarketingPref(user.id), 100)}
      />
    ));
  }

  renderFilters() {
    return (
      <div className="table__buttons">
        <button
          id="marketing-other"
          className="table__other"
          onClick={e => this.props.setFilter(e.target.id)}
        >Other filter
        </button>
        <button
          id="marketing-none"
          className="table__no-marketing"
          onClick={e => this.props.setFilter(e.target.id)}
        >No marketing
        </button>
      </div>
    );
  }

  render() {
    if (!_.size(this.props.users)) {
      return null;
    }
    return (
      <div className="table__container">
        {this.renderTableHeader()}
        {this.renderTable()}
        {this.renderFilters()}
      </div>
    );
  }
}

const mapDispatchToProps = dispatch => bindActionCreators(
  {
    getUsers,
    setUserMarketingPref,
    setFilter,
  },
  dispatch,
);

const mapStateToProps = state => ({
  users: state.users,
  filter: state.filter,
  filterMode: state.marketing.filterMode,
});

AppContainer.defaultProps = {
  filterMode: false,
  getUsers: null,
  setUserMarketingPref: null,
  setFilter: null,
};

AppContainer.propTypes = {
  users: PropTypes.object,
  filterMode: PropTypes.bool,
  getUsers: PropTypes.func,
  setUserMarketingPref: PropTypes.func,
  setFilter: PropTypes.func,
};

export default connect(mapStateToProps, mapDispatchToProps)(AppContainer);
4

1 回答 1

7

这两条规则解决了两个不同的问题

第一:预计类方法“renderTableHeader”将使用“this”。(class-methods-use-this) (JSX 属性) className: string

根据文档

如果一个类方法不使用这个,有时可以做成静态函数。如果您确实将该方法转换为静态函数,则调用该特定方法的类的实例也必须转换为静态调用(MyClass.callStaticMethod())

还要注意,在上面的示例中,如果将方法切换为静态方法,则(let a = new A(); a.sayHi();)必须将调用静态方法的类的实例更新为静态调用(A.sayHi();),而不是让类的实例调用该方法

如何避免出现此警告

exceptMethods选项允许您传递要忽略警告的方法名称数组。例如,您可能有来自外部库的规范,要求您将方法覆盖为常规函数(而不是静态方法),并且不在函数体内使用 this。在这种情况下,您可以添加该方法以在警告中忽略。

"class-methods-use-this": [<enabled>, { "exceptMethods": [<...exceptions>] }]

二:禁止Prop类型对象(react/forbid-prop-types)导入PropTypes

定义不模糊的 PropType 是一个很好的做法,例如any, arrayobject这并不能清楚地告诉你 prop 实际上是什么类型。

根据文档

默认情况下,此规则会阻止具有更具体的替代方案(任何、数组、对象)的模糊道具类型,但如果需要,可以禁用任何道具类型。选择默认值是因为它们有明显的替换。any 应该替换为,好吧,任何东西。array 和 object 可以分别替换为 arrayOf 和 shape。

在你的情况下说是user一个像

{
    id: 123,
    name: 'abc'
}

你可以像这样定义 PropType

AppContainer.propTypes = {
  users: PropTypes.shape({
    id: PropTypes.number,
    name: PropTypes.string
  }),,
  filterMode: PropTypes.bool,
  getUsers: PropTypes.func,
  setUserMarketingPref: PropTypes.func,
  setFilter: PropTypes.func,
};
于 2018-02-14T17:10:13.040 回答