使用React和Material-UI构建TODO应用的前端UI

ops/2025/2/5 21:39:59/

使用React和Material-UI构建TODO应用的前端UI

        • 引言
        • 环境准备
        • 代码解析
          • 1. 导入必要的模块
          • 2. 创建React组件
          • 3. 定义函数
            • 3.1 获取TODO列表
            • 3.2 创建TODO项
            • 3.3 更新TODO项
            • 3.4 删除TODO项
            • 3.5 处理编辑点击事件
            • 3.6 关闭编辑对话框
            • 3.7 保存编辑内容
          • 4. 使用Effect钩子
          • 5. 渲染组件
        • 功能实现
        • 优化建议
        • 总结

引言

在现代Web开发中,TODO列表应用是一个经典的示例,用于展示如何使用前端技术构建一个简单的任务管理工具。本文将详细介绍如何使用React框架和Material-UI库来构建一个TODO列表应用,并解释代码的各个部分,帮助读者理解其工作原理。

后端API请参考,使用Express.js和SQLite3构建简单TODO应用的后端API

环境准备

在开始之前,请确保你已经安装了以下工具和库:

  1. Node.js:确保你已经安装了Node.js,可以从Node.js官网下载并安装。
  2. npm:Node.js的包管理工具,随Node.js一起安装。
  3. React:一个用于构建用户界面的JavaScript库,可以通过npm安装。
  4. Material-UI:一个基于Material Design的React组件库,同样可以通过npm安装。
  5. axios:一个基于Promise的HTTP客户端,用于处理API请求。

安装所需的依赖:

npm install react @mui/material @emotion/react @emotion/styled axios
代码解析

让我们逐步分析代码,理解每个部分的功能。

1. 导入必要的模块
typescript">import { useState, useEffect } from 'react';
import axios from 'axios';
import {TextField,Button,Checkbox,List,ListItem,ListItemText,IconButton,Dialog,DialogTitle,DialogContent,DialogActions,FormControlLabel
} from '@mui/material';
import { Delete, Edit } from '@mui/icons-material';
import { Todo } from '../types/todo';
  • React:导入useStateuseEffect钩子,用于状态管理和副作用处理。
  • axios:用于发送HTTP请求。
  • Material-UI:导入各种组件,如文本框、按钮、列表、对话框等。
  • Icons:导入删除和编辑图标。
  • Todo类型:定义TODO项的类型,确保数据的类型安全。
2. 创建React组件
typescript">export default function TodoList() {const [todos, setTodos] = useState<Todo[]>([]);const [searchTerm, setSearchTerm] = useState('');const [newTodo, setNewTodo] = useState({ title: '', description: '' });const [editingTodo, setEditingTodo] = useState<Todo | null>(null);const [editForm, setEditForm] = useState({ title: '', description: '', completed: false });
  • todos:存储TODO列表的状态。
  • searchTerm:存储搜索词的状态。
  • newTodo:存储新TODO项的状态。
  • editingTodo:存储正在编辑的TODO项的状态。
  • editForm:存储编辑表单的状态。
3. 定义函数
3.1 获取TODO列表
typescript">const fetchTodos = async () => {try {const response = await axios.get(`http://localhost:3002/api/todos?q=${searchTerm}`);setTodos(response.data);} catch (error) {uiltin">console.error('Error fetching todos:', error);}
};
  • 功能:从后端获取TODO列表,支持搜索功能。
  • 实现:使用axios.get发送GET请求,根据searchTerm进行模糊搜索。
3.2 创建TODO项
typescript">const createTodo = async () => {if (!newTodo.title.trim()) return;try {await axios.post('http://localhost:3002/api/todos', newTodo);setNewTodo({ title: '', description: '' });fetchTodos();} catch (error) {uiltin">console.error('Error creating todo:', error);}
};
  • 功能:创建一个新的TODO项。
  • 实现:检查标题是否为空,使用axios.post发送POST请求,创建成功后清空表单并刷新列表。
3.3 更新TODO项
typescript">const updateTodo = async (todo: Todo, isToggleComplete = false) => {try {const updatedTodo = isToggleComplete? { ...todo, completed: !todo.completed }: { ...todo, title: editForm.title, description: editForm.description, completed: editForm.completed };await axios.put(`http://localhost:3002/api/todos/${todo.id}`, updatedTodo);setEditingTodo(null);fetchTodos();} catch (error) {uiltin">console.error('Error updating todo:', error);}
};
  • 功能:更新TODO项,支持标记完成和编辑内容。
  • 实现:根据isToggleComplete决定是更新完成状态还是编辑内容,使用axios.put发送PUT请求。
3.4 删除TODO项
typescript">const deleteTodo = async (id: uiltin">number) => {try {await axios.delete(`http://localhost:3002/api/todos/${id}`);fetchTodos();} catch (error) {uiltin">console.error('Error deleting todo:', error);}
};
  • 功能:删除指定的TODO项。
  • 实现:使用axios.delete发送DELETE请求,删除成功后刷新列表。
3.5 处理编辑点击事件
typescript">const handleEditClick = (todo: Todo) => {setEditingTodo(todo);setEditForm({title: todo.title,description: todo.description || '',completed: todo.completed});
};
  • 功能:打开编辑对话框,填充TODO项的详细信息。
  • 实现:设置editingTodoeditForm状态,显示编辑表单。
3.6 关闭编辑对话框
typescript">const handleClose = () => {setEditingTodo(null);setEditForm({ title: '', description: '', completed: false });
};
  • 功能:关闭编辑对话框,重置表单。
  • 实现:清除editingTodoeditForm状态。
3.7 保存编辑内容
typescript">const handleSave = () => {if (editingTodo && editForm.title.trim()) {updateTodo(editingTodo);}
};
  • 功能:保存编辑的内容。
  • 实现:检查标题是否为空,调用updateTodo更新TODO项。
4. 使用Effect钩子
typescript">useEffect(() => {const debounceSearch = setTimeout(() => {fetchTodos();}, 300);return () => clearTimeout(debounceSearch);
}, [searchTerm]);
  • 功能:实现搜索的防抖动效果,防止频繁请求。
  • 实现:使用setTimeout延迟300毫秒后执行fetchTodos,并在组件销毁时清除定时器。
5. 渲染组件
typescript">return (<div style={{ maxWidth: 600, margin: '0 auto', padding: '20px' }}><TextFieldfullWidthlabel="Search Todos"variant="outlined"value={searchTerm}onChange={(e) => setSearchTerm(e.target.value)}margin="normal"/><List>{todos.map((todo) => (<ListItemkey={todo.id}secondaryAction={(<><IconButton onClick={() => handleEditClick(todo)}><Edit /></IconButton><IconButton onClick={() => deleteTodo(todo.id)}><Delete /></IconButton></>)}style={{display: todo.completed ? 'none' : 'flex',opacity: todo.completed ? 0.7 : 1}}><Checkboxchecked={Boolean(todo.completed)}onChange={() => updateTodo(todo, true)}/><ListItemTextprimary={todo.title}secondary={todo.description}style={{textDecoration: todo.completed ? 'line-through' : 'none',}}/></ListItem>))}</List><div style={{ display: 'flex', gap: 10, marginBottom: 20 }}><TextFieldfullWidthlabel="New Todo Title"value={newTodo.title}onChange={(e) => setNewTodo({ ...newTodo, title: e.target.value })}/><TextFieldfullWidthlabel="Description"value={newTodo.description}onChange={(e) => setNewTodo({ ...newTodo, description: e.target.value })}/><Buttonvariant="contained"color="primary"onClick={createTodo}>Add</Button></div><Dialog open={Boolean(editingTodo)} onClose={handleClose}><DialogTitle>Edit Todo</DialogTitle><DialogContent><TextFieldautoFocusmargin="dense"label="Title"fullWidthvalue={editForm.title}onChange={(e) => setEditForm({ ...editForm, title: e.target.value })}/><TextFieldmargin="dense"label="Description"fullWidthmultilinerows={3}value={editForm.description}onChange={(e) => setEditForm({ ...editForm, description: e.target.value })}/><FormControlLabelcontrol={(<Checkboxchecked={editForm.completed}onChange={(e) => setEditForm({ ...editForm, completed: e.target.checked })}/>)}label="Completed"/></DialogContent><DialogActions><Button onClick={handleClose} color="primary">Cancel</Button><ButtononClick={handleSave}color="primary"disabled={!editForm.title.trim()}>Save</Button></DialogActions></Dialog></div>
);
  • 搜索框:允许用户输入搜索词,实时搜索TODO列表。
  • TODO列表:显示所有TODO项,每个项包含标题、描述、编辑和删除按钮,以及完成状态Checkbox。
  • 添加TODO表单:允许用户输入新TODO的标题和描述,点击“Add”按钮创建。
  • 编辑对话框:当用户点击编辑按钮时,显示编辑表单,允许修改TODO的标题、描述和完成状态。
功能实现
  1. 添加TODO项:用户输入标题和描述后,点击“Add”按钮,发送POST请求到后端,创建新的TODO项。
  2. 编辑TODO项:用户点击编辑按钮,打开编辑对话框,修改TODO项的详细信息后,点击“Save”按钮,发送PUT请求到后端,更新TODO项。
  3. 删除TODO项:用户点击删除按钮,发送DELETE请求到后端,删除指定的TODO项。
  4. 搜索TODO项:用户输入搜索词,组件会自动搜索标题或描述中包含该词的TODO项,支持防抖动功能,减少请求次数。
  5. 标记完成:用户点击Checkbox,TODO项会被标记为完成,样式会变为灰色并添加删除线。
优化建议

尽管这段代码已经可以正常工作,但为了提升应用的性能和用户体验,可以考虑以下优化:

  1. 添加加载状态:在数据加载过程中,显示加载动画,提升用户体验。
  2. 添加错误提示:在请求失败时,显示错误提示信息,帮助用户了解问题所在。
  3. 添加成功提示:在创建、更新或删除TODO项成功后,显示成功提示信息。
  4. 优化样式:根据Material Design规范,优化组件的样式和布局,提升视觉效果。
  5. 添加响应式设计:确保应用在不同设备上都能良好显示,提升应用的适应性。
  6. 添加数据验证:在前端和后端都添加数据验证,确保输入的数据合法有效。
总结

通过本文,我们详细解析了一个使用React和Material-UI构建的TODO列表应用。从状态管理、HTTP请求到组件渲染,每个部分都进行了详细的解释。希望这篇文章能够帮助读者理解如何使用这些技术构建一个简单的TODO应用,并为进一步的学习和开发打下基础。


http://www.ppmy.cn/ops/155978.html

相关文章

golang开发技能

本文主要介绍go相关 开发技巧、调试技巧、工具使用、单元测试、基准测试、性能测试相关。 1、Go命令&#xff1a;go test工具详解 这里先大致介绍测试工具“go test”&#xff0c;go test 本身可以携带很多参数&#xff0c;熟悉这些参数可以让我们的测试过程更加方面。具体使…

第五章:元婴-React用户功能实战

文章目录 登录页面布局JWT 令牌鉴权用户功能实现用户查询页面用户更改状态用户添加页面用户添加页面表单构建用户编辑页面用户编辑表单页面登录页面布局 import React, { useEffect, useState } from react import { Button, Form, Input, message } from antd import style fr…

索引的底层数据结构、B+树的结构、为什么InnoDB使用B+树而不是B树呢

索引的底层数据结构 MySQL中常用的是Hash索引和B树索引 Hash索引&#xff1a;基于哈希表实现的&#xff0c;查找速度非常快&#xff0c;但是由于哈希表的特性&#xff0c;不支持范围查找和排序&#xff0c;在MySQL中支持的哈希索引是自适应的&#xff0c;不能手动创建 B树的…

doris:STRUCT

STRUCT<field_name:field_type [COMMENT comment_string], ... > 表示由多个 Field 组成的结构体&#xff0c;也可被理解为多个列的集合。 不能作为 Key 使用&#xff0c;目前 STRUCT 仅支持在 Duplicate 模型的表中使用。一个 Struct 中的 Field 的名字和数量固定&…

PDF 擦除工具

该软件仅仅适用于非人民币玩家&#xff0c;如果有wps会员等类似的软件的没有大用处 PDF Eraser允许用户擦除PDF文件中任何元素&#xff0c;并且支持添加文本和图像。除此之外PDF Eraser允许用户删除不必要的PDF页面&#xff0c;为了兼顾一些大型的扫描的PDF文档&#xff0c;PDF…

ubuntu 网络管理--wpa_supplicant、udhcpc

ubuntu 网络管理--wpa_supplicant 1 介绍wpa_supplicant 无线认证wpa_passphrase 配置工具 NetworkManager 网络管理udhcpc 与 dhclient对比dhclient概述主要功能 udhcpc概述主要功能 2 联系依赖关系配置文件 3 区别4 如何选择5 示例使用 wpa_supplicant 手动连接无线网络使用 …

MVC 文件夹:架构之美与实际应用

MVC 文件夹:架构之美与实际应用 引言 MVC(Model-View-Controller)是一种设计模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种架构模式不仅提高了代码的可维护性和可扩展性,而且使得开发流程更加清晰。本文将深入探讨MVC文…

HTTPS域名443端口证书到期问题排查与解决

在现代Web开发中&#xff0c;HTTPS协议广泛用于确保客户端和服务器之间的通信安全。然而&#xff0c;HTTPS依赖于SSL/TLS证书来加密通信并验证网站的身份。当证书过期时&#xff0c;客户端可能会遇到连接错误。本文将介绍如何排查和解决因证书过期引起的问题&#xff0c;尤其是…