typescript编写react组件3 —— 购物车计数组件

748 查看

前言

接上一篇文章的代码片段,今天为大家提供另一组件——购物车计数,因为当初写组件时没有想到会有多复杂,所以写到后面才发现整个组件存在大的问题,开篇就不提bug在哪里了,结尾告诉大家,不过有兴趣的小伙伴们可以重构整个组件,可以联系我,我们一起来讨论。

组件结构

/*
 * ShopCarTable
 *    ProductTable
 *        ProductRow
 *            ProductCount
 *        ProductTotal
 */

代码片段

interface Product {
  id: string,
  name: string,
  price: number
}
interface ProductCount {
  pid: string,
  count: number
}

// ProductCount
interface ProductCountProps {
  onCount: (c: number) => any;
  count: number;
}
interface ProductCountState {

}
class ProductCountEditor extends React.Component<ProductCountProps, ProductCountState>{
  handleClickAdd = () => this.props.onCount(this.props.count + 1)
  handleClickRed = () => this.props.count > 0 && this.props.onCount(this.props.count - 1)
  public render() {
    return (
      <form>
        <input type="button" value="-" onClick={this.handleClickRed} />
        <input type="text" value={this.props.count + ''} readOnly />
        <input type="button" value="+" onClick={this.handleClickAdd} />
      </form>
    );
  }
}
// ProductRow
interface ProductRowProps {
  product: Product
  count: number
  onCount: (c:number)=>void
}
class ProductRow extends React.Component<ProductRowProps, {}>{
  constructor(props: any) {
    super(props);
  }
  public render() {
    const {id, name, price} = this.props.product
    const count = this.props.count
    return (
      <tr>
        <td>{id}</td>
        <td>{name}</td>
        <td>{price}¥</td>
        <td><ProductCountEditor count={count} onCount={this.props.onCount} /></td>
        <td>{price * count}¥</td>
      </tr>
    );
  }
}

// ProductTotal
interface ProductTotalProps {
  total: number
}
function ProductTotal(props: ProductTotalProps) {
  return (
    <tfoot>
      <tr>
        <td colSpan={5}>总价: {props.total}¥</td>
      </tr>
    </tfoot>
  );
}

// ProductTable
interface ProductTableProps {
  products: Product[]
}
interface ProductTableState {
  counts: ProductCount[]
}
class ProductTable extends React.Component<ProductTableProps, ProductTableState>{
  constructor(props: ProductTableProps) {
    super(props);
    let counts: ProductCount[] = props.products.map(p => ({pid: p.id, count: 0}));
    this.state = { counts };
  }
  public render() {
    const products = this.props.products;
    const counts = this.state.counts;
    let productNodes = products.map(p => {
      let c = counts.filter(c => c.pid===p.id)[0];
      return (
        <ProductRow key={p.id} product={p} count={c.count} onCount={count => {
          let countsSlice = counts.slice();
          console.log(countsSlice);
          countsSlice.splice(counts.indexOf(c), 1, {pid:c.pid, count});
          this.setState({ counts: countsSlice });
        }} />
      );
    });
    return (
      <table>
        <thead>
          <tr>
            <th>产品id</th>
            <th>产品名称</th>
            <th>产品单价</th>
            <th>产品数量</th>
            <th>产品总计</th>
          </tr>
        </thead>
        <tbody>
          {productNodes}
        </tbody>
        <ProductTotal total={this.state.counts.map(pc => {
          return this.props.products.filter(p => p.id === pc.pid)[0].price * pc.count
        }).reduce((sum,c)=>sum+c, 0)} />
      </table>
    );
  }
}

let PRODUCTS: Product[] = [
  { id: '1', name: '背包', price: 49.99 },
  { id: '2', name: '衣服', price: 34 }
];

ReactDOM.render(
  <ProductTable products={PRODUCTS} />,
  document.getElementById('example')
);

结尾

大家发现这个大问题是什么了吗?
是我的数据结构一开始就没有定义好,所以我的计数实现才那么复杂。以后数据最好改成适合自己应用的结构。。。就像这个组件中,products跟counts没有组合在一起就很不方便了。
下面是实现的效果: