From a16380250d0966ed28fed1296086f1668178f0b2 Mon Sep 17 00:00:00 2001 From: daradanilova Date: Fri, 15 Nov 2024 14:20:18 +0200 Subject: [PATCH 1/2] add task solution --- src/App.tsx | 264 +++++++++++++++++-- src/api/todos.ts | 22 ++ src/components/Footer/Footer.tsx | 57 ++++ src/components/Header/Header.tsx | 73 +++++ src/components/Notification/Notification.tsx | 31 +++ src/components/TodoItem/TodoItem.tsx | 126 +++++++++ src/components/TodoList/TodoList.tsx | 46 ++++ src/components/index.ts | 4 + src/types/Todo.ts | 6 + src/utils/fetchClient.ts | 46 ++++ 10 files changed, 660 insertions(+), 15 deletions(-) create mode 100644 src/api/todos.ts create mode 100644 src/components/Footer/Footer.tsx create mode 100644 src/components/Header/Header.tsx create mode 100644 src/components/Notification/Notification.tsx create mode 100644 src/components/TodoItem/TodoItem.tsx create mode 100644 src/components/TodoList/TodoList.tsx create mode 100644 src/components/index.ts create mode 100644 src/types/Todo.ts create mode 100644 src/utils/fetchClient.ts diff --git a/src/App.tsx b/src/App.tsx index 81e011f432..b72ad41d5d 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -1,26 +1,260 @@ -/* eslint-disable max-len */ -/* eslint-disable jsx-a11y/control-has-associated-label */ -import React from 'react'; +import React, { + useCallback, + useEffect, + useMemo, + useRef, + useState, +} from 'react'; import { UserWarning } from './UserWarning'; +import { Header } from './components/Header/Header'; +import { TodoList } from './components/TodoList/TodoList'; +import { Footer } from './components/Footer/Footer'; +import { Todo } from './types/Todo'; +import { Notification } from './components/Notification/Notification'; +import * as todoService from './api/todos'; +import { USER_ID, getTodos } from './api/todos'; -const USER_ID = 0; +export enum Filter { + All = 'all', + Active = 'active', + Completed = 'completed', +} export const App: React.FC = () => { + const [todosFromServer, setTodosFromServer] = useState([]); + const [filter, setFilter] = useState(Filter.All); + const [tempTodo, setTempTodo] = useState(null); + const [loadingTodoIds, setLoadingTodoIds] = useState([]); + const [notification, setNotification] = useState({ + isHidden: true, + message: '', + }); + + const inputRef = useRef(null); + + const showNotification = (message: string) => { + setNotification({ isHidden: false, message }); + setTimeout(() => setNotification({ isHidden: true, message: '' }), 3000); + }; + + const filterTodos = useCallback((todos: Todo[], filterBy: Filter): Todo[] => { + switch (filterBy) { + case Filter.Completed: + return todos.filter(todo => todo.completed); + + case Filter.Active: + return todos.filter(todo => !todo.completed); + + default: + return todos; + } + }, []); + + const visibleTodos = useMemo( + () => filterTodos(todosFromServer, filter), + [filterTodos, todosFromServer, filter], + ); + + const activeTodosCount = useMemo( + () => todosFromServer.filter(todo => !todo.completed).length, + [todosFromServer], + ); + + const allTodosCompleted = useMemo( + () => activeTodosCount === 0, + [activeTodosCount], + ); + + const hasCompletedTodos = useMemo( + () => todosFromServer.some(todo => todo.completed), + [todosFromServer], + ); + + const handleAddTodo = (title: string) => { + setNotification({ isHidden: true, message: '' }); + + const trimmedTitle = title.trim(); + + if (!trimmedTitle.length) { + showNotification('Title should not be empty'); + + return Promise.reject('Title is empty'); + } + + setTempTodo({ + title: trimmedTitle, + userId: USER_ID, + completed: false, + id: 0, + }); + + return todoService + .createTodo({ title: trimmedTitle, userId: USER_ID, completed: false }) + .then(newTodo => { + setTodosFromServer(currentTodos => [...currentTodos, newTodo]); + }) + .catch(error => { + showNotification('Unable to add a todo'); + throw new Error(error); + }) + .finally(() => { + setTempTodo(null); + }); + }; + + const handleDeleteTodo = (todoId: number) => { + setLoadingTodoIds([todoId]); + + return todoService + .deleteTodo(todoId) + .then(() => { + setTodosFromServer(curr => curr.filter(todo => todo.id !== todoId)); + }) + .catch(error => { + showNotification('Unable to delete a todo'); + throw new Error(error); + }) + .finally(() => { + setLoadingTodoIds([]); + if (inputRef.current) { + inputRef.current.focus(); + } + }); + }; + + const handleClearCompletedTodos = () => { + const completedTodoIds = todosFromServer + .filter(todo => todo.completed) + .map(todo => todo.id); + + setLoadingTodoIds(completedTodoIds); + Promise.all( + completedTodoIds.map(id => + todoService + .deleteTodo(id) + .then(() => { + setTodosFromServer(curr => curr.filter(todo => todo.id !== id)); + }) + .catch(error => { + showNotification('Unable to delete a todo'); + throw new Error(error); + }) + .finally(() => { + setLoadingTodoIds([]); + if (inputRef.current) { + inputRef.current.focus(); + } + }), + ), + ); + }; + + const handleUpdateTodo = (updatedTodo: Todo) => { + setLoadingTodoIds([updatedTodo.id]); + + return todoService + .updateTodo(updatedTodo) + .then(receivedTodo => { + setTodosFromServer(curr => + curr.map(todo => (todo.id === receivedTodo.id ? receivedTodo : todo)), + ); + }) + .catch(error => { + showNotification('Unable to update a todo'); + throw new Error(error); + }) + .finally(() => { + setLoadingTodoIds([]); + }); + }; + + const handleToggleAllTodoStatus = () => { + let todosToChange = []; + + if (allTodosCompleted) { + todosToChange = [...todosFromServer]; + } else { + todosToChange = todosFromServer.filter(todo => !todo.completed); + } + + const todoToChangeIds = todosToChange.map(todo => todo.id); + + setLoadingTodoIds(todoToChangeIds); + Promise.all( + todosToChange.map(todoToChange => { + const { id, completed, title, userId } = todoToChange; + + todoService + .updateTodo({ id, completed: !completed, title, userId }) + .then(receivedTodo => { + setTodosFromServer(curr => + curr.map(todo => + todo.id === receivedTodo.id ? receivedTodo : todo, + ), + ); + }) + .catch(error => { + showNotification('Unable to update a todo'); + throw new Error(error); + }) + .finally(() => { + setLoadingTodoIds([]); + }); + }), + ); + }; + + useEffect(() => { + getTodos() + .then(setTodosFromServer) + .catch(() => { + showNotification('Unable to load todos'); + }); + }, []); + if (!USER_ID) { return ; } return ( -
-

- Copy all you need from the prev task: -
- - React Todo App - Add and Delete - -

- -

Styles are already copied

-
+
+

todos

+ +
+
+ + {!!todosFromServer.length && ( + + )} + + {!!todosFromServer.length && ( +
+ )} +
+ + setNotification({ ...notification, isHidden: true })} + /> +
); }; diff --git a/src/api/todos.ts b/src/api/todos.ts new file mode 100644 index 0000000000..6970a96b03 --- /dev/null +++ b/src/api/todos.ts @@ -0,0 +1,22 @@ +import { Todo } from '../types/Todo'; +import { client } from '../utils/fetchClient'; + +export const USER_ID = 1831; + +export const getTodos = () => { + return client.get(`/todos?userId=${USER_ID}`); +}; + +export function createTodo({ title, userId, completed }: Omit) { + return client.post(`/todos`, { title, userId, completed }); +} + +export function deleteTodo(todoId: number) { + return client.delete(`/todos/${todoId}`); +} + +export function updateTodo(data: Todo): Promise { + const { id } = data; + + return client.patch(`/todos/${id}`, data); +} diff --git a/src/components/Footer/Footer.tsx b/src/components/Footer/Footer.tsx new file mode 100644 index 0000000000..62805d7670 --- /dev/null +++ b/src/components/Footer/Footer.tsx @@ -0,0 +1,57 @@ +import classNames from 'classnames'; +import { Filter } from '../../App'; + +type Props = { + activeTodosCount: number; + currFilter: Filter; + hasCompletedTodos: boolean; + onFilter: (newFilter: Filter) => void; + onClearCompletedTodos: () => void; +}; + +export const Footer: React.FC = ({ + activeTodosCount, + currFilter, + hasCompletedTodos, + onFilter, + onClearCompletedTodos, +}) => { + return ( +
+ + {`${activeTodosCount} items left`} + + + + + +
+ ); +}; diff --git a/src/components/Header/Header.tsx b/src/components/Header/Header.tsx new file mode 100644 index 0000000000..5f1b2d7eca --- /dev/null +++ b/src/components/Header/Header.tsx @@ -0,0 +1,73 @@ +import classNames from 'classnames'; +import { useEffect } from 'react'; + +type Props = { + inputRef: React.RefObject; + hasTodos: boolean; + allCompletedTodos: boolean; + onAddTodo: (value: string) => Promise; + onToggleAll: () => void; +}; + +export const Header: React.FC = ({ + inputRef, + hasTodos, + allCompletedTodos, + onAddTodo, + onToggleAll, +}) => { + useEffect(() => { + if (inputRef.current) { + inputRef.current.focus(); + } + }, [inputRef]); + + const handleSubmit = (event: React.FormEvent) => { + event.preventDefault(); + + const inputElement = inputRef.current; + + if (inputElement) { + inputElement.disabled = true; + + onAddTodo(inputElement.value) + .then(() => { + if (inputElement) { + inputElement.value = ''; + } + }) + .catch(() => {}) + .finally(() => { + if (inputElement) { + inputElement.disabled = false; + inputElement.focus(); + } + }); + } + }; + + return ( +
+ {hasTodos && ( +
+ ); +}; diff --git a/src/components/Notification/Notification.tsx b/src/components/Notification/Notification.tsx new file mode 100644 index 0000000000..23910284ef --- /dev/null +++ b/src/components/Notification/Notification.tsx @@ -0,0 +1,31 @@ +import classNames from 'classnames'; + +type Props = { + message: string; + isHidden: boolean; + onClose: (value: boolean) => void; +}; + +export const Notification: React.FC = ({ + message, + isHidden, + onClose, +}) => { + return ( +
+
+ ); +}; diff --git a/src/components/TodoItem/TodoItem.tsx b/src/components/TodoItem/TodoItem.tsx new file mode 100644 index 0000000000..6a4203d98c --- /dev/null +++ b/src/components/TodoItem/TodoItem.tsx @@ -0,0 +1,126 @@ +import React, { useRef, useState } from 'react'; +import classNames from 'classnames'; +import { Todo } from '../../types/Todo'; + +type Props = { + todo: Todo; + isLoading?: boolean; + onDeleteTodo?: (value: number) => Promise; + onUpdateTodo?: (updatedTodo: Todo) => Promise; +}; + +export const TodoItem: React.FC = ({ + todo, + onDeleteTodo = () => {}, + isLoading = false, + onUpdateTodo = () => {}, +}) => { + const { completed, id, title: todoTitle, userId } = todo; + + const [isEditing, setIsEditing] = useState(false); + const [editedTitle, setEditedTitle] = useState(todoTitle); + const inputEdit = useRef(null); + + const handleEditSubmit = (event: React.FormEvent) => { + event.preventDefault(); + + const trimmedTitle = editedTitle.trim(); + + if (trimmedTitle === todoTitle) { + setIsEditing(false); + + return; + } + + if (!trimmedTitle) { + onDeleteTodo(id) + ?.then(() => setIsEditing(false)) + .catch(() => {}); + + return; + } + + onUpdateTodo({ id, userId, completed, title: trimmedTitle }) + ?.then(() => setIsEditing(false)) + .catch(() => {}); + }; + + const handleEditCancel = (event: React.KeyboardEvent) => { + if (event.key === 'Escape') { + setIsEditing(false); + } + }; + + return ( +
+ {/* eslint-disable-next-line jsx-a11y/label-has-associated-control */} + + + {isEditing ? ( +
+ setEditedTitle(event.target.value)} + onBlur={handleEditSubmit} + autoFocus + /> +
+ ) : ( + <> + { + setIsEditing(true); + setEditedTitle(todoTitle); + }} + > + {todoTitle} + + + + )} + +
+
+
+
+
+ ); +}; diff --git a/src/components/TodoList/TodoList.tsx b/src/components/TodoList/TodoList.tsx new file mode 100644 index 0000000000..3cbe7652d0 --- /dev/null +++ b/src/components/TodoList/TodoList.tsx @@ -0,0 +1,46 @@ +import React from 'react'; +import { CSSTransition, TransitionGroup } from 'react-transition-group'; +import { Todo } from '../../types/Todo'; +import { TodoItem } from '../TodoItem/TodoItem'; + +type Props = { + todos: Todo[]; + tempTodo: Todo | null; + loading: number[]; + onDeleteTodo: (value: number) => Promise; + onUpdateTodo: (updatedTodo: Todo) => Promise; +}; + +export const TodoList: React.FC = ({ + todos, + tempTodo, + loading, + onDeleteTodo, + onUpdateTodo, +}) => { + return ( +
+ +
+ {todos.map(todo => ( + + + + ))} + + {tempTodo && ( + + + + )} +
+
+
+ ); +}; diff --git a/src/components/index.ts b/src/components/index.ts new file mode 100644 index 0000000000..44e88357e7 --- /dev/null +++ b/src/components/index.ts @@ -0,0 +1,4 @@ +// export * from './Error/Error'; +// export * from './Footer/Footer'; +// export * from './TodoList/TodoList'; +// export * from './TodoItem/TodoItem'; diff --git a/src/types/Todo.ts b/src/types/Todo.ts new file mode 100644 index 0000000000..3f52a5fdde --- /dev/null +++ b/src/types/Todo.ts @@ -0,0 +1,6 @@ +export interface Todo { + id: number; + userId: number; + title: string; + completed: boolean; +} diff --git a/src/utils/fetchClient.ts b/src/utils/fetchClient.ts new file mode 100644 index 0000000000..708ac4c17b --- /dev/null +++ b/src/utils/fetchClient.ts @@ -0,0 +1,46 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ +const BASE_URL = 'https://mate.academy/students-api'; + +// returns a promise resolved after a given delay +function wait(delay: number) { + return new Promise(resolve => { + setTimeout(resolve, delay); + }); +} + +// To have autocompletion and avoid mistypes +type RequestMethod = 'GET' | 'POST' | 'PATCH' | 'DELETE'; + +function request( + url: string, + method: RequestMethod = 'GET', + data: any = null, // we can send any data to the server +): Promise { + const options: RequestInit = { method }; + + if (data) { + // We add body and Content-Type only for the requests with data + options.body = JSON.stringify(data); + options.headers = { + 'Content-Type': 'application/json; charset=UTF-8', + }; + } + + // DON'T change the delay it is required for tests + return wait(100) + .then(() => fetch(BASE_URL + url, options)) + .then(response => { + if (!response.ok) { + throw new Error(); + } + + return response.json(); + }); +} + +export const client = { + get: (url: string) => request(url), + post: (url: string, data: any) => request(url, 'POST', data), + patch: (url: string, data: any) => request(url, 'PATCH', data), + delete: (url: string) => request(url, 'DELETE'), +}; From e3109afe9e5455e86f1cd6b51ad49029b2b8d02a Mon Sep 17 00:00:00 2001 From: daradanilova Date: Mon, 18 Nov 2024 10:03:57 +0200 Subject: [PATCH 2/2] add task solution --- src/components/TodoItem/TodoItem.tsx | 6 +++--- src/components/TodoList/TodoList.tsx | 7 +++---- src/components/index.ts | 4 ---- 3 files changed, 6 insertions(+), 11 deletions(-) delete mode 100644 src/components/index.ts diff --git a/src/components/TodoItem/TodoItem.tsx b/src/components/TodoItem/TodoItem.tsx index 6a4203d98c..fbbd18f8e4 100644 --- a/src/components/TodoItem/TodoItem.tsx +++ b/src/components/TodoItem/TodoItem.tsx @@ -11,8 +11,8 @@ type Props = { export const TodoItem: React.FC = ({ todo, - onDeleteTodo = () => {}, isLoading = false, + onDeleteTodo = () => {}, onUpdateTodo = () => {}, }) => { const { completed, id, title: todoTitle, userId } = todo; @@ -114,8 +114,8 @@ export const TodoItem: React.FC = ({
diff --git a/src/components/TodoList/TodoList.tsx b/src/components/TodoList/TodoList.tsx index 3cbe7652d0..c7caa0efa3 100644 --- a/src/components/TodoList/TodoList.tsx +++ b/src/components/TodoList/TodoList.tsx @@ -1,4 +1,4 @@ -import React from 'react'; +// import React from 'react'; import { CSSTransition, TransitionGroup } from 'react-transition-group'; import { Todo } from '../../types/Todo'; import { TodoItem } from '../TodoItem/TodoItem'; @@ -26,17 +26,16 @@ export const TodoList: React.FC = ({ ))} {tempTodo && ( - + )}
diff --git a/src/components/index.ts b/src/components/index.ts deleted file mode 100644 index 44e88357e7..0000000000 --- a/src/components/index.ts +++ /dev/null @@ -1,4 +0,0 @@ -// export * from './Error/Error'; -// export * from './Footer/Footer'; -// export * from './TodoList/TodoList'; -// export * from './TodoItem/TodoItem';