更改单个项目时重新呈现整个列表

IT技术 reactjs
2021-05-11 09:11:25

我有一个产品列表,我想添加更多数据,例如价格和数量。问题是当我开始输入时我失去了输入焦点,因为整个列表都被重新渲染。我有一个简单的 Fiddle 复制那段代码:



const App = () => {

  // List of products
  const [products, setProducts] = React.useState([
  {
    title: 'My Product',
  },
  {
    title: 'My Product 2',
  }
  ]);
  
  
  // Simple debug to track changes
  React.useEffect(() => {
    console.log('PRODUCTS', products);
  }, [products]);
  
  
  const ProductForm = ({ data, index }) => {
    
    const handleProductChange = (name, value) => {
      const allProducts = [...products];
      const selectedProduct = {...allProducts[index]};
      allProducts[index] = {
        ...selectedProduct,
        [name]: value
      };
      setProducts([ ...allProducts ]);
    }
    
    return (
      <li>
        <h2>{data.title}</h2>
        <label>Price:</label>
        <input type="text" value={products[index].price} onChange={(e) => handleProductChange('price', e.target.value)} />
      </li>
    );
  }
  
  return <ul>{products.map((item, index) => <ProductForm key={item.title} index={index} data={item} />)}</ul>;
}

ReactDOM.render(
  <App />,
  document.getElementById('container')
);

https://jsfiddle.net/lucasbittar/zh1d6y37/23/

我已经尝试了一堆我在这里找到的解决方案,但没有一个真正代表我所拥有的。

谢谢!

2个回答

问题

您已经在ProductForm内部定义了toApp所以它在每个渲染周期重新创建,即它是每个渲染的全新组件。

解决方案

移动ProductForm到外部定义。现在存在handleProductChange无法访问App的功能范围products状态的问题。这里的解决方案是移handleProductChangeApp并更新函数签名以使用index. 使用data.price作为输入值,你可以提供一个备用值或该物业提供初始状态。

建议:命名输入name="price"并简单地从事件对象中使用它。

const ProductForm = ({ data, index, handleProductChange }) => {
    return (
      <li>
        <h2>{data.title}</h2>
        <label>Price:</label>
        <input
          name="price" // <-- name input field
          type="text"
          value={data.price || ''} // <-- use data.price or fallback value
          onChange={(e) => handleProductChange(e, index)} // <-- pass event and index
        />
      </li>
    );
  }

const App = () => {

    // List of products
    const [products, setProducts] = React.useState([
  {
    title: 'My Product',
  },
  {
    title: 'My Product 2',
  }
  ]);
  
  
  // Simple debug to track changes
  React.useEffect(() => {
    console.log('PRODUCTS', products);
  }, [products]);

  // Update signature to also take index
  const handleProductChange = (e, index) => {
      const { name, value } = e.target; // <-- destructure name and value
      const allProducts = [...products];
      const selectedProduct = {...allProducts[index]};
      allProducts[index] = {
        ...selectedProduct,
        [name]: value
      };
      setProducts([ ...allProducts ]);
    }
  
  return (
    <ul>
      {products.map((item, index) => (
        <ProductForm
          key={item.title}
          index={index}
          data={item}
          handleProductChange={handleProductChange} // <-- pass callback handler
        />)
      )}
    </ul>
  );
}

ReactDOM.render(
  <App />,
  document.getElementById('container')
);

工作 jsfiddle 演示

解决该问题的一种方法是使用单独的组件<ProductForm />并将所需的组件传递到props那里。

const ProductForm = ({ data, index, products, setProducts }) => {
    const handleProductChange = (name, value) => {
      const allProducts = [...products];
      const selectedProduct = {...allProducts[index]};
      allProducts[index] = {
        ...selectedProduct,
        [name]: value
      };
      setProducts([ ...allProducts ]);
    }
    
    return (
      <li>
        <h2>{data.title}</h2>
        <label>Price:</label>
        <input
          type="text"
          value={products[index].price || ''}
          onChange={(e) => handleProductChange('price', e.target.value)}
        />
      </li>
    );
}

const App = () => {
  const [products, setProducts] = React.useState([{title: 'My Product'},{title: 'My Product 2'}]);
  return (
    <ul>
      {products.map((item, index) => 
         <ProductForm
           key={item.title}
           index={index}
           data={item}
           products={products}
           setProducts={setProducts}
         />
      )}
     </ul>
  )
}

ReactDOM.render(
  <App />,
  document.getElementById('container')
);

这是有效的 JSFiddle 代码链接