React 入门:实战案例 TodoList 底部功能实现

news2024/11/27 0:17:52

文章目录

  • 目标实现效果
  • 实现已完成和全部数量统计和显示
  • 实现全选和全不选
  • 实现清除已完成功能
  • 完整代码
    • App 组件完整代码
    • Footer 组件完整代码

通过前面的章节已经完成 TodoList 的增删改的功能,本文我们来实现底部相关功能:

  • 已完成和全部数量实时统计,在增加、勾选取消、删除时,已完成和全部数量实时更新。
  • 底部的 checkbox 实现全选和全不选,并与列表的 checkbox 联动;
  • 实现清除已完成功能。

目标实现效果

在这里插入图片描述

实现已完成和全部数量统计和显示

已完成和全部数量的统计是基于 TodoList 数据的,所以需要通过 props 把 TodoList 的数据传递给 Footer 组件。

  • 首先,从 App 组件中将 Todolist 数据通过 props 传递给 Footer 组件,代码片段如下:
// file: src/App.js

<Footer todoList={todoList} />
  • 然后,在 Footer 组件的 render() 中通过 prosps 接收到 TooList 列表数据,进行计算统计数据和渲染,代码片段如下:
// file: src/components/Footer/index.

// 以下代码在 render 方法中实现

// 从 props 获取 todoList
const { todoList } = this.props;

// 已完成数量
const doneCount = todoList.reduce((pre, todo) => pre + (todo.done ? 1 : 0), 0);

// 全部数量
const total = todoList.length;

// DOM渲染
<span>
  <span>已完成 {doneCount}</span> / 全部 {total}
</span>;

此时,界面上已经可以正确显示已完成(也就是已勾选)和全部数量(也就是 TodoList 的长度),并且更改列表中每条 Todo 的 checkbox 的勾选状态时,统计数据也会同时更新。

实现全选和全不选

底部组件还有一个 checkbox,用以控制全选和全不选,选中为全选,取消选中为全不选。
这个 checkbox 本身只是一个操作元素,不需要表示自己状态的 state,它的状态依赖于 TodoList 列表中每个 Item 的状态。而它本身勾选和不勾选,又会修改 TodoList 列表中每个 Item 的状态。

  • 首先,需要在 App.js 中定义处理全选和全不选的处理回调方法,并且通过 props 传递给 Footer 组件,代码片段如下:
// file: src/App.js

// 用于处理全选和全不选,参数 done 为 true 表示全选,否则 false 表示全不选
checkAllTodo = (done) => {
  // 获取原来的 todoList
  const { todoList } = this.state;
  // 加工数据
  const newTodoList = todoList.map((todoObj) => {
    return { ...todoObj, done };
  });
  // 更新状态
  this.setState({ todoList: newTodoList });
};

// 将上面定义的方法传递给 Footer 组件
<Footer todoList={todoList} checkAllTodo={this.checkAllTodo} />;
  • 然后,在 Footer 组件中接收全选和全不选的处理方法,并通过 checkbox 的 onChange 事件回调中调用,代码片段如下:
// file: src/components/Footer/index.jsx

// 定义 checkbox 的onChange 事件回调,用来处理全选和全不选
handleCheckAll = (event) => {
  this.props.checkAllTodo(event.target.checked);
};

// 给 checkBox 添加 onChange 事件,并动态控制其勾选状态
<input
  type="checkbox"
  checked={doneCount === total && total !== 0 ? true : false}
  onChange={this.handleCheckAll}
/>;

实现清除已完成功能

清除已完成的原理就是,把 TodoList 中已完成的数据删除掉。

  • 首先,需要在 App.js 中定义处理清除已完成的处理方法,并且通过 props 传递给 Footer 组件,代码片段如下:
// file: src/App.js

// 清除已完成
clearDone = () => {
  const { todoList } = this.state;
  // 过滤数据
  const newTodoList = todoList.filter((todoObj) => {
    return !todoObj.done;
  });
  this.setState({ todoList: newTodoList });
};

// 将上面定义的方法传递给 Footer 组件
<Footer
  todoList={todoList}
  checkAllTodo={this.checkAllTodo}
  clearDone={this.clearDone}
/>;
  • 然后,在 Footer 组件中接收清除已完成的处理方法,并在 button 的 onClick 事件回调中调用,代码片段如下:
// file: src/components/Footer/index.jsx

// 定义清除所有已完成的回调方法
handleClearAllDone = () => {
  this.props.clearDone();
};

// 给 button 添加 onClick 事件
<button onClick={this.handleClearAllDone} className="btn btn-danger">
  清除已完成任务
</button>;

至此,完成了清除已完成功能。

完整代码

App 组件完整代码

// file: src/App.js

// 创建“外壳”组件

import React, { Component } from "react";
import Header from "./components/Header";
import List from "./components/List";
import Footer from "./components/Footer";
import "./App.css";

export default class App extends Component {
  // 总结:状态在哪里,操作状态的方法就在哪里。

  // 初始化状态
  state = {
    todoList: [
      { id: 1, name: "参加晨会", done: true },
      { id: 2, name: "A功能开发", done: true },
      { id: 3, name: "B功能开发", done: false },
    ],
  };

  /**
   * addTodo 用于添加一条 Todo 记录,接收的参数是 Todo 对象
   */
  addTodo = (todoObj) => {
    // 获取原 TodoList
    const { todoList } = this.state;
    // 追加一条 Todo
    const newTodoList = [todoObj, ...todoList];
    // 更新状态
    this.setState({ todoList: newTodoList });
  };

  // 用于更新一个 Todo 对象
  updateTodo = (id, done) => {
    // 获取状态的中 todoList
    const { todoList } = this.state;
    // 匹配处理数据
    const newTodoList = todoList.map((todoObj) => {
      if (todoObj.id === id) return { ...todoObj, done };
      else return todoObj;
    });
    // 更新状态
    this.setState({ todoList: newTodoList });
  };

  // 用于删除一条 Todo
  deleteTodo = (id) => {
    // 获取原来的 todoList
    const { todoList } = this.state;
    // 删除指定 id 的 todo 对象
    const newTodoList = todoList.filter((todoObj) => {
      return todoObj.id !== id;
    });
    // 更新状态
    this.setState({ todoList: newTodoList });
  };

  // 用于处理全选和全不选,参数 done 为 true 表示全选,否则 false 表示全不选
  checkAllTodo = (done) => {
    // 获取原来的 todoList
    const { todoList } = this.state;
    // 加工数据
    const newTodoList = todoList.map((todoObj) => {
      return { ...todoObj, done };
    });
    // 更新状态
    this.setState({ todoList: newTodoList });
  };

  // 清除已完成
  clearDone = () => {
    const { todoList } = this.state;
    // 过滤数据
    const newTodoList = todoList.filter((todoObj) => {
      return !todoObj.done;
    });
    this.setState({ todoList: newTodoList });
  };

  render() {
    const { todoList } = this.state;
    return (
      <div className="todo-container">
        <div className="todo-wrap">
          <Header addTodo={this.addTodo} />
          <List
            todoList={todoList}
            updateTodo={this.updateTodo}
            deleteTodo={this.deleteTodo}
          />
          <Footer
            todoList={todoList}
            checkAllTodo={this.checkAllTodo}
            clearDone={this.clearDone}
          />
        </div>
      </div>
    );
  }
}

Footer 组件完整代码

// file: src/components/Footer/index.jsx

import React, { Component } from "react";
import "./index.css";

export default class Footer extends Component {
  // 全选和全不选的回调
  handleCheckAll = (event) => {
    this.props.checkAllTodo(event.target.checked);
  };

  // 定义清除所有已完成的回调方法
  handleClearAllDone = () => {
    this.props.clearDone();
  };

  render() {
    const { todoList } = this.props;
    // 已完成数量
    const doneCount = todoList.reduce(
      (pre, todo) => pre + (todo.done ? 1 : 0),
      0
    );
    // 全部
    const total = todoList.length;
    return (
      <div className="todo-footer">
        <div>
          <label>
            <input
              type="checkbox"
              checked={doneCount === total && total !== 0 ? true : false}
              onChange={this.handleCheckAll}
            />
          </label>
          <span>
            <span>已完成 {doneCount}</span> / 全部 {total}
          </span>
        </div>
        <div>
          <button onClick={this.handleClearAllDone} className="btn btn-danger">
            清除已完成任务
          </button>
        </div>
      </div>
    );
  }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/92304.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Linux】文件系统/inode/软硬链接

目录 一.了解磁盘 1.磁盘的概念 2.磁盘的物理结构 3.磁盘的逻辑结构 4.磁盘区域的划分 二.linux文件系统 三.inode 四.软硬链接 1.软链接 2.硬链接 一.了解磁盘 前言: 一般情况下, 系统中存在大量的未被打开的文件, 这些文件全部存储在磁盘上, 也简称磁盘级文件 在li…

功能测试进阶自动化测试,一个女测试工的坎坷之路

绝大多数测试工程师都是从功能测试做起的&#xff0c;工作忙忙碌碌&#xff0c;每天在各种业务需求学习和点点中度过&#xff0c;过了好多年发现自己还只是一个功能测试工程师。 随着移动互联网的发展&#xff0c;从业人员能力的整体进步&#xff0c;软件测试需要具备的能力要…

从ADAS到高阶自动驾驶,车联网芯片如何赋能产业升级?

2022年&#xff0c;L2及L2ADAS前装搭载率正在快速提升&#xff0c;中国智能网联汽车产业进入了由ADAS加速向高阶自动驾驶迈进的关键期。 根据《高工智能汽车研究院》数据显示&#xff0c;今年1-9月前装标配搭载L2级辅助驾驶搭载量为395.19万辆&#xff0c;同比增长69.53%&…

Java开发必备技能:RocketMQ

官方文档可见&#xff1a;https://github.com/hiwei-zhang/rocketmq/tree/develop/docs/cn RocketMQ集群架构 RocketMQ由以下这几个组件组成 NameServer : 提供轻量级的Broker路由服务。Broker&#xff1a;实际处理消息存储、转发等服务的核心组件。Producer&#xff1a;消息…

69. 利用 ALV 实现增删改查系列之一:让 ALV 报表进入可编辑状态

在 CSDN 和我的知识星球里有朋友向我提出同样的问题,询问如何在 ALV 里实现增删改查操作。 虽然需求只有一句话,但是这个需求背后涉及到的知识点不少,因此笔者会通过几篇文章的篇幅,来介绍这个需求的详细实现步骤。 本文先解决第一个障碍,如何让 ALV 报表进入可编辑状态?…

[ MySQL ] 使用 MySQL Workbentch 进行MySQL数据库备份 / 还原(Part 3:备份.sql文件方式)

本文主要讲解如何用 MySQL Workbentch 进行MySQL备份和恢复数据库&#xff08;利用.sql文件的方式&#xff09;。 关于MySQL数据库备份&#xff0c;本博客内主要相关文章有&#xff1a; [ MySQL ] 使用Navicat进行MySQL数据库备份 / 还原&#xff08;Part 1&#xff1a;备份.…

Unity发布Android平台错误记录

Unity发布Android平台错误记录 &#xff08;1&#xff09;Cannot parse project property android.enableR8‘’ of type ‘class java.lang.String’ as boolean. Expected ‘true’ or ‘false’. 注释掉android.enableR8MINIFY_WITH_R_EIGHT打包就可以了 &#xff08;2&…

电线电缆企业应用APS计划排产软件的效益

电线电缆行业是我国经济第二大的配套产业&#xff0c;仅次于汽车产业&#xff0c;年市场规模超万 亿&#xff0c;按产品可分为电力线缆、电气装备用线缆、通信线缆以及绕组线等。电线电缆用以传输电(磁)能&#xff0c;信息和实现电磁能转换的线材产品。广义的电线电缆亦简称为电…

基础算法系列之基础(二)[大数问题]

文章目录前言大数相加大数相减大数乘法除法总结前言 OK&#xff0c;继续预热哈&#xff0c;没办法还得补作业&#xff0c;要G了&#xff0c;明天看看有没有时间加更一篇。 那么今天的话还是来说一下这个精度的一些问题&#xff0c;也就是大数之类的一些问题啥的。这个当然咱们…

mysql优化

索引&#xff1a; 排好序的数据结构 从磁盘上拿一条记录要和磁盘做一次IO操作&#xff0c;&#xff0c;磁盘的IO性能不高 索引数据结构&#xff1a; 二叉树(binary search tree) &#xff1a; 单边增长的数据没有帮助红黑树(red black tree) &#xff1a; 二叉平衡树&#x…

YOLOv1-YOLOv7全系列解析汇总

导读 目标检测Yolo算法是非常经典且应用广泛的算法&#xff0c;而在Yolo中&#xff0c;又分成了输入端、网络推理、输出层&#xff0c;每个部分都可以延伸出很多的优化方式&#xff0c;本文主要从Yolov1~v7各个版本的Backbone&#xff0c;Neck&#xff0c;Head&#xff0c;Tri…

JSPM基于SSM的乐聘网人才招聘系统

开发工具(eclipse/idea/vscode等)&#xff1a;idea 数据库(sqlite/mysql/sqlserver等)&#xff1a;mysql 功能模块(请用文字描述&#xff0c;至少200字)&#xff1a;主要功能模块 &#xff08;核心功能完整&#xff0c;不局限于下面我想的功能&#xff0c;下面的功能可以改&…

LeetCode刷题总结文档

前言 本文的刷题顺序依照代码随想录进行&#xff0c;因此题目板块的划分也和代码随想录一致。每个版块我会按照以下内容进行组织&#xff1a; 该类型题目的特征时间复杂度值得一讲的相关题目知识 正文 数组 二分查找 特征&#xff1a;有序数组中找特定组合的快速查询方法…

如何查询域名是否备案,域名备案批量查询方法

所有备案的域名都是需要通过工信部审核&#xff0c;录入到系统才可正式使用&#xff0c;所以我们在工信部的网站查询域名是否由备案信息就可以辨别出此域名是否有备案了。但是&#xff0c;大多数站长少则几十个域名&#xff0c;几千几万域名也是有的&#xff0c;如果还是在工信…

kong网关使用记录

kong 是一个api网关&#xff0c;类似nginx的网关&#xff0c;一般用于api服务的管理 kong 可以从consul中通过dns获取服务路由&#xff0c;可以自动实现负载均衡 kong kong最新版没有了conteos的安装文档&#xff0c;现在用Ubuntu来安装kong 1 下载kong curl -Lo kong-enter…

ERROR: Failed building wheel for box2d-py 解决方法

当我们在一个全新的Python环境中构建一个AI/ML项目时&#xff0c;往往会在pip install -r requirements.txt阶段遇到这样一个错误&#xff1a; Building wheel for box2d-py (setup.py) ... error .... swig.exe -python -c -IBox2D -small -O -includeall -ignoremissing -w2…

Java中常见的注解

一、注解 二、三种JDK内置的基本注解 1、Override 表示该方法是重写的父类方法。 为什么要写Override&#xff1f; 其实方法前不写Override是ok的&#xff0c;并且仍然是重写了父类方法。However&#xff01;如果方法前有Override注解&#xff0c;编译器就会去检查该方法是否真…

ToB SaaS公司如何低成本高效获客

今年SaaS行业有些冷&#xff0c;如何过冬成了热门话题。原来天天对外讲帮客户降本增效&#xff0c;现在自个儿降本增效。调整组织架构&#xff0c;精简人员&#xff0c;砍预算&#xff0c;大抵都要来一遍。隔三岔五听说某家公司裁员&#xff0c;某家公司工资减半。市场预算自然…

CSS基础总结(一)

文章目录 一、CSS概述 &#xff08;1&#xff09;简介 &#xff08;2&#xff09;语法规范 &#xff08;3&#xff09;代码风格 二、CSS基础选择器 &#xff08;1&#xff09;标签选择器 &#xff08;2&#xff09;类选择器 &#xff08;3&#xff09;多类名选择器 &am…

STM32F103xx随记

关于STM32F103xx单片机的一些零碎知识&#xff0c;随时整理下来&#xff0c;方便查阅。 STM32F103xx随记STM32 & STM8 命名STM32F103xx引脚定义表引脚信息芯片手册地址大佬那里抄来的图STM32F103xxADC笔记STM32 & STM8 命名 STM32F103xx引脚定义表 今天在给新的模块添…