【ArchSummit架构师峰会】探讨数据与人工智能相互驱动的关系>>> 了解详情
写点什么

用 React Hooks 做一个搜索栏

  • 2020-04-24
  • 本文字数:2539 字

    阅读完需:约 8 分钟

用React Hooks做一个搜索栏

本文最初发布于 Hackernoon 博客,经原作者授权由 InfoQ 中文站翻译并分享。


以下是我们将要构建的搜索框的动图。这是一个简单的搜索框,我们可以用它来搜索联系人列表。我们将使用函数式组件,而不是基于类的组件来实现它。



下面就开始吧。首先创建一个新的 React 应用:


npx create-react-app contacts-list 
复制代码


然后转到 contacts-list 目录。在你常用的代码编辑器中打开目录。就我而言,我使用的是 vscode,因此我要从命令行执行的操作是:


code .
复制代码


src 目录中创建一个名为 components 的新文件夹,并在其中创建一个 Numbers.js 文件。转到你的 App.js 文件并导入 Numbers.js 组件。


接下来我们需要创建一些人名,然后将这些人名作为 props 传递给 Numbers.js 组件来渲染。


import React, { useState } from "react";import { Numbers } from "./components/Numbers";export const App = () => {  const [persons] = useState([    { name: "Dayo Olorinla", number: "+234-1234-5678" },    { name: "Temi Otedola", number: "+234-9029-9229" },    { name: "Zlatan Ibile", number: "+234-1243-2345" },  ]);
return ( <div> <Numbers persons={persons} /> </div> );};export default App;
复制代码


现在在我们的 Numbers 组件中,我们将接收从 App 传递过来的 props,并使用它来显示 contacts list


参见下面的代码,其中包含每个步骤的解释说明。


import Rect, { useStae } from "react";export const Numbers = props => {// word 会跟踪 filter box 内输入的任何更改  const [word, setword] = useState("");// filterdisplay 会基于 search 来显示更新的列表,其默认状态是我们的 persons 列表 prop  const [filterDispllay, setFilterDisplay] = useStae(props.persons);// handleChange 每次运行时在输入字段都会有一个更改  const handleChange = e => {// 在一个新数组中存放原始列表,将所有人名转为小写字母,因为我们不知道用户要输入什么格式;然后我们返回 OldList 作为一个对象数组,来存放这个更改的列表  let oldList = props.persons.map(person => {    return { name: person.name.toLowerCase(), number: person.number };  });  // 如果输入栏不为空,则运行以下代码;否则,setFilterDisplay 设为原始列表 prop  if (e !== "") {    let newList =[];    // setWord 一直跟踪输入的任何更改    setWord(e);    // newList 是保存符合搜索参数的 persons 的数组    newList = oldList.filter(person =>// 我们调用 includes 方法并用小写传递进'word'状态,这会检查 oldList 是否包含名字中带有'word'的人名      person.name.includes(word.toLowerCase())    );// 我们会一直检查输入并返回 newList 数组。我们调用 setFilterDisplay 来在每次输入调整后更新状态      setFilterDisplay(newList);    }  };  return (    <div>      <hl>Number</hl>      filter: <input onChange={e => handleChange(e.target.value)} />      {filterDisplay.map((person, i) => (        <div key={i}>          <li>            {person.name} &nbsp;            <span>{person.number}</span>          </li>        </div>      ))}    </div>  );};
复制代码


最后,每次更新时,我们都会从 FilterDisplay 返回更新的信息。如果你和我一样想将搜索栏分成一个单独的组件,请继续看下去。下面我们来重构这个东西!将搜索拆分成一个单独的组件后,我们就可以在应用程序的其他组件中使用同样的搜索栏了。


首先我们创建一个 Filter 组件,在我们的 components 文件夹中将其命名为 Filter.js。它需要 2 个 props,分别用于输入值和 onChange 事件。


import React from "react";export const Filter = ({ value, handleChange }) => {  return (    <div>      filter: <input value={value} onChange={handleChange} />    </div>  );};
复制代码


接下来我们需要重构 Numbers.js 组件,让它只渲染过滤过的人员列表。它将接受一个 prop,也就是 list/array。


import React from "react";export const Numbers = ({ persons }) => {  return (    <div>      <hl>Numbers</hl>      {person.map((person, i) => (        <div key={i}>          <li>            {person.name} &nbsp;            <span>{person.number}</span>          </li>        </div>      ))}    </div>  );};
复制代码


回想一下,我们所有的状态都在 App 组件内管理,并作为 props 传递给我们的组件。最后,在 App 组件中我们将一个有状态值传递给 Filter 组件中的输入字段,还将传递一个 handleChange 方法,当输入字段中发生更改时将调用这个方法。


import React, { useState } from "react";import { Filter } from "./components/Filter";import { Numbers } from "./components/Numbers";export const App = () => {  const [word, setWord] = useState("");  const [persons] = useState([    { name: "Dayo Olorinla", number: "+234-1244-5678" },    { name: "Temi Otedola", number: "+234-9029-9229" },    { name: "Zlatan Ibile", number: "+234-1243-2345" }  ]);  const [filterDisplay, setFilterDisplay] = useState([]);
const handleChange = e => { setWord(e); let oldList = persons.map(person => { return { name: person.name.toLowerCase(), number: person.number }; });
if (word !== "") { let newList = [];
newList = oldList.filter(person => person.name.includes(word.toLowerCase()) );
setFilterDisplay(newList); } else { setFilterDisplay(persons); } };
return ( <div> <Filter value={word} handleChange={e => handleChange(e.target.value)} /> <Numbers persons={word.length <1 ? persons : filterDisplay} /> </div> );};export default App;
复制代码


return 中,我们的 Numbers 组件将始终检查输入字段是否为空白。如果是的话就渲染原始的 Persons 数组,否则我们根据在输入字段中输入的内容渲染列表。就是这样,搞定!

英文原文

How to Build a Search Bar in React With React Hooks


2020-04-24 16:483504
用户头像
小智 让所有人认同的文字称不上表达

发布了 408 篇内容, 共 377.3 次阅读, 收获喜欢 1972 次。

关注

评论

发布
暂无评论
发现更多内容

拆分电商系统为微服务

大眼喵

「架构实战营」

在线火星文转换器工具

入门小站

工具

模块六:拆分电商系统为微服务

jiaoxn

「架构实战营」

代码之外:谈谈算法该怎么准备,不准备可以吗

宇宙之一粟

算法面试 代码之外 5月月更

flask框架的学习笔记【二】

恒山其若陋兮

5月月更

拆分电商系统为微服务

dan629xy

架构学习(二)

爱晒太阳的大白

5月月更

Hyperspace索引系统论文解析

漫长的白日梦

spark 数据湖 索引系统

计算机网络——物理层

工程师日月

计算机网络 5月月更

绿色环保作为经济长线主题,MOVE PROTOCOL运动APP来助力

股市老人

位运算小妙招-求二进制序列中0的个数

芒果酱

数据结构 算法 5月月更

跨平台应用开发进阶(十三) :uni-app应用异常退出时处理机制探究

No Silver Bullet

uni-app 5月月更 异常退出 处理机制

Go 依赖注入 Wire 详解 - 用户指南

baiyutang

Go 微服务 依赖注入 5月月更

用IntelliJ IDEA ULTIMATE版看Java类图

程序员欣宸

Java IDEA 5月月更

【ELT.ZIP】OpenHarmony啃论文俱乐部——人工智能短字符串压缩

ELT.ZIP

人工智能 鸿蒙 数据压缩 ELT.ZIP

拆分电商系统为微服务

踩着太阳看日出

架构训练营

在线HTML转XML工具

入门小站

工具

架构模块六

小马

「架构实战营」

九、云原生链路追踪

穿过生命散发芬芳

链路追踪 5月月更

AI简报-逆光也清晰-色彩增强算法CURL

AIWeker

人工智能 深度学习 5月月更 AI简报

架构实战营 - 模块六 - 作业

michael

架构实战营 #架构实战营 架构师实战营 「架构实战营」

STM32F103系列开发_点亮LED灯

DS小龙哥

5月月更

【ELT.ZIP】OpenHarmony啃论文俱乐部——电子设备软件更新压缩

ELT.ZIP

鸿蒙 rsync 数据压缩 ELT.ZIP

模块六作业: 拆分电商系统为微服务

凯博无线

M_6: 拆分电商系统为微服务

Jadedev

架构训练营

OKALEIDO解决NFT流动性不足难题,更有创新平台通证分配方案

股市老人

Linux环境封装静态库

Loken

音视频 5月月更

【ELT.ZIP】OpenHarmony啃论文俱乐部——多层存储分级数据压缩

ELT.ZIP

鸿蒙 数据压缩 ELT.ZIP HCompress

运动健康深入人心,MOVE PROTOCOL引领品质生活

BlockChain先知

HashMap 源码分析-基础结构

zarmnosaj

5月月更

今天要学习的技术点,Python 筛选数字,模块导入,特殊变量__all__ 实战博客

梦想橡皮擦

5月月更

用React Hooks做一个搜索栏_大前端_adebola_InfoQ精选文章