1. Overview

Challenge & starter files: Advent of JS

Full codes: nilstarbb/advent-of-js/2-eCommerce-component

Live Demo: 02 - E-Commerce Component || Advent of JavaScript

Preview:

ecommerce-component.gif

2. Details

Users should be able to:

  • View the plates on the left side of the screen and add them to your cart on the right side.
  • When there are no plates within your cart, you should see a message that says, “Your cart is empty.”
  • When a plate is added to your cart, the Subtotal and Totals will automatically update.
  • When products are in your cart, you should be able to increase and decrease the quantity.
    • A user should not be able to mark the quantity as a negative number.
    • If the quantity goes down to 0, the user will have the option to delete or remove the product from their cart entirely.
  • Tax is based on the state of Tennessee sales tax: 0.0975

3. Start Coding

Setup templates

练习用的小项目,就不大动干戈搞全套了,还是直接套 CDN 模板。

index.html:

<!DOCTYPE html>
<html lang="en">
  <head>
    ...
    <link rel="stylesheet" href="./styles.css" />

    <script src="https://unpkg.com/react@17/umd/react.development.js"></script>
    <script src="https://unpkg.com/react-dom@17/umd/react-dom.development.js"></script>
    <script src="https://unpkg.com/@babel/standalone/babel.min.js"></script>
  </head>

  <body>
    <div class="wrapper menu" id="wrapper"></div>
    <script type="text/babel" src="./script.js"></script>
  </body>
</html>

script 里的 type="text/babel" 比较容易被忽略,如果缺了会导致 Babel 失效报错。

script.js:

const { useState, useEffect } = React;

const App = () => {
  return (
    <>
			...
    </>
  );
};

ReactDOM.render(<App />, document.getElementById("wrapper"));

Breakdown components

App 的 structure 比较复杂,遂拆分成几个 components。

其中的 MenuItemButton 可以不用特别拆出来,原样放在 MenuItem 里也可以。

const formatPrice = (price) => "$" + (price / 100).toFixed(2);

const MenuItemButton = ({ item }) => {
  if (item.count > 0) {
    return (
      <button className="in-cart">
        <img src="images/check.svg" alt="Check" />
        In Cart
      </button>
    );
  } else {
    return (
      <button className="add">
        Add to Cart
      </button>
    );
  }
};

const MenuItem = ({ item }) => {
  return (
    <li>
      <div className="plate">
        <img src={"images/" + item.image} alt={item.alt} className="plate" />
      </div>
      <div className="content">
        <p className="menu-item">{item.name}</p>
        <p className="price">{formatPrice(item.price)}</p>
        <MenuItemButton item={item} />
      </div>
    </li>
  );
};

const CartItem = ({ itemInCart }) => {
  return (
    <li>
      <div className="plate">
        <img
          src={"images/" + itemInCart.image}
          alt={itemInCart.alt}
          className="plate"
        />
        <div className="quantity">{itemInCart.count}</div>
      </div>
      <div className="content">
        <p className="menu-item">{itemInCart.name}</p>
        <p className="price">{formatPrice(itemInCart.price)}</p>
      </div>
      <div className="quantity__wrapper">
        <button className="decrease">
          <img src="images/chevron.svg" />
        </button>
        <div className="quantity">{itemInCart.count}</div>
        <button className="increase">
          <img src="images/chevron.svg" />
        </button>
      </div>
      <div className="subtotal">
        {formatPrice(itemInCart.price * itemInCart.count)}
      </div>
    </li>
  );
};

const CartSummary = ({ itemsInCart }) => {
  if (itemsInCart.length == 0) {
    return <p className="empty">Your cart is empty.</p>;
  } else {
    return (
      <ul className="cart-summary">
        {itemsInCart.map((item) => (
          <CartItem
            itemInCart={item}
            key={item.name}
          />
        ))}
      </ul>
    );
  }
};

const Cart = ({ itemsInCart }) => {
  const subtotal = itemsInCart.reduce(
    (total, item) => total + item.price * item.count,
    0
  );

  return (
    <div className="panel cart">
      <h1>Your Cart</h1>
      <CartSummary itemsInCart={itemsInCart} />

      <div className="totals">
        <div className="line-item">
          <div className="label">Subtotal:</div>
          <div className="amount price subtotal">{formatPrice(subtotal)}</div>
        </div>
        <div className="line-item">
          <div className="label">Tax:</div>
          <div className="amount price tax">
            {formatPrice(subtotal * 0.0975)}
          </div>
        </div>
        <div className="line-item total">
          <div className="label">Total:</div>
          <div className="amount price total">
            {formatPrice(subtotal * 1.0975)}
          </div>
        </div>
      </div>
    </div>
  );
};

const App = () => {
  const [items, setItems] = useState(menuItems);
  const itemsInCart = items.filter((item) => item.count > 0);

  return (
    <>
      <div className="panel">
        <h1>To Go Menu</h1>
        <ul className="menu">
          {items.map((item) => (
            <MenuItem
              key={item.name}
              item={item}
            />
          ))}
        </ul>
      </div>
      <Cart itemsInCart={itemsInCart} />
    </>
  );
};

Update the state of an array of objects

这个项目的功能很简单,需要实时更新的数据全部放在 items 里,用一个 useState 处理就可以了。购物车里的 itemsInCart 可以直接用 items 筛选获得。

const App = () => {
  const [items, setItems] = useState(menuItems);
  const itemsInCart = items.filter((item) => item.count > 0);
	...
};

itemsInCart 会根据 itemcount 实时更新,当 count == 0 时会自动从右侧购物车里消失,因此不用特地为 remove item 编写逻辑,只需要为左侧菜单添加 addItemToCart、右侧购物车添加 increaseItem/decreaseItem,再把它们传递给 child components,绑定 onClick 即可。

const App = () => {
  const [items, setItems] = useState(menuItems);
  const itemsInCart = items.filter((item) => item.count > 0);

  const addItemToCart = (item) => {
    const updatedItem = { ...item, count: 1 };
    setItems(items.map((i) => (i.name !== item.name ? i : updatedItem)));
  };

  const increaseItem = (item) => {
    const updatedItem = { ...item, count: item.count + 1 };
    setItems(items.map((i) => (i.name !== item.name ? i : updatedItem)));
  };

  const decreaseItem = (item) => {
    const updatedItem = { ...item, count: item.count - 1 };
    setItems(items.map((i) => (i.name !== item.name ? i : updatedItem)));
  };

  return (
    <>
      <div className="panel">
        <h1>To Go Menu</h1>
        <ul className="menu">
          {items.map((item) => (
            <MenuItem
              key={item.name}
              item={item}
              addItemToCart={addItemToCart}
            />
          ))}
        </ul>
      </div>

      <Cart
        itemsInCart={itemsInCart}
        increaseItem={increaseItem}
        decreaseItem={decreaseItem}
      />
    </>
  );
};

这里使用 map 函数更新 itemsstate。更新 array of objects 的办法不止这一种,不过根据 Redux 官方手册里 Immutable Update Patterns 一章所写,用 map 函数似乎已经是最优解了。