From 85b7e1e0d7bc574e2c9c9c00786ef7f0a6ca3aa6 Mon Sep 17 00:00:00 2001 From: Rostyslav Date: Fri, 11 Oct 2024 16:07:47 +0300 Subject: [PATCH 1/2] Solution --- src/App.tsx | 143 ++++++++++----- src/components/NewCommentForm.tsx | 281 ++++++++++++++++++++---------- src/components/PostDetails.tsx | 192 ++++++++++---------- src/components/PostsList.tsx | 130 ++++++-------- src/components/UserSelector.tsx | 70 ++++++-- src/utils/api.ts | 30 ++++ 6 files changed, 534 insertions(+), 312 deletions(-) create mode 100644 src/utils/api.ts diff --git a/src/App.tsx b/src/App.tsx index 017957182..274e4caa8 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -1,60 +1,121 @@ -import classNames from 'classnames'; +import { FC, useEffect, useState } from 'react'; +import cn from 'classnames'; -import 'bulma/css/bulma.css'; -import '@fortawesome/fontawesome-free/css/all.css'; -import './App.scss'; +import { User } from './types/User'; +import { Post } from './types/Post'; + +import { getPosts, getUsers } from './utils/api'; import { PostsList } from './components/PostsList'; import { PostDetails } from './components/PostDetails'; import { UserSelector } from './components/UserSelector'; import { Loader } from './components/Loader'; -export const App = () => ( -
-
-
-
-
-
- -
+import 'bulma/css/bulma.css'; +import '@fortawesome/fontawesome-free/css/all.css'; +import './App.scss'; -
-

No user selected

+export const App: FC = () => { + const [users, setUsers] = useState([]); + const [selectedUser, setSelectedUser] = useState(null); - + const [userPosts, setUserPosts] = useState([]); + const [selectedPost, setSelectedPost] = useState(null); -
- Something went wrong! -
+ const [isLoading, setIsLoading] = useState(false); + const [isError, setIsError] = useState(false); + + useEffect(() => { + setIsLoading(true); + setIsError(false); + + getUsers() + .then(setUsers) + .catch(() => setIsError(true)) + .finally(() => setIsLoading(false)); + }, []); -
- No posts yet + useEffect(() => { + if (!selectedUser) { + return; + } + + setIsLoading(true); + setIsError(false); + setUserPosts([]); + setSelectedPost(null); + + getPosts(selectedUser.id) + .then(setUserPosts) + .catch(() => setIsError(true)) + .finally(() => setIsLoading(false)); + }, [selectedUser]); + + return ( +
+
+
+
+
+
+
- +
+ {!selectedUser && ( +

No user selected

+ )} + + {isLoading && } + + {isError && !isLoading && ( +
+ Something went wrong! +
+ )} + + {!userPosts.length && + selectedUser && + !isLoading && + !isError && ( +
+ No posts yet +
+ )} + + {!!userPosts.length && selectedUser && ( + + )} +
-
-
-
- +
+
+ {selectedPost && } +
-
-
-); +
+ ); +}; diff --git a/src/components/NewCommentForm.tsx b/src/components/NewCommentForm.tsx index 73a8a0b45..f19f1235b 100644 --- a/src/components/NewCommentForm.tsx +++ b/src/components/NewCommentForm.tsx @@ -1,103 +1,206 @@ -import React from 'react'; +import { + ChangeEvent, + Dispatch, + FC, + FormEvent, + SetStateAction, + useState, +} from 'react'; +import cn from 'classnames'; + +import { Comment } from '../types/Comment'; +import { addComment } from '../utils/api'; + +const initialFormData = { name: '', email: '', body: '' }; +const initialFromErrors = { name: false, email: false, body: false }; + +interface Props { + postId: number; + updateComments: Dispatch>; +} + +export const NewCommentForm: FC = ({ postId, updateComments }) => { + const [formData, setFormData] = useState(initialFormData); + const [formErrors, setFormErrors] = useState(initialFromErrors); + + const [isLoading, setIsLoading] = useState(false); + const [isError, setIsError] = useState(false); + + const handleChangeField = ( + event: ChangeEvent, + ) => { + const { name, value } = event.target; + + setFormData(prevData => ({ ...prevData, [name]: value.trimStart() })); + }; + + const handleResetForm = () => { + setFormData(initialFormData); + setFormErrors(initialFromErrors); + }; + + const handleFormSubmit = (event: FormEvent) => { + event.preventDefault(); + + const { name, email, body } = formData; + const trimmedData = { + name: name.trim(), + email: email.trim(), + body: body.trim(), + }; + + setFormErrors({ + name: !trimmedData.name, + email: !trimmedData.email, + body: !trimmedData.body, + }); + + if (!trimmedData.name && !trimmedData.email && !trimmedData.body) { + return; + } + + setIsLoading(true); + + addComment({ + postId, + ...trimmedData, + }) + .then(comment => { + updateComments(prevComments => [...prevComments, comment]); + setFormData(prevState => ({ ...prevState, body: '' })); + }) + .catch(() => setIsError(true)) + .finally(() => setIsLoading(false)); + }; -export const NewCommentForm: React.FC = () => { return ( -
-
- - -
- - - - - - - - - + <> + {isError && ( +
+ Something went wrong!
+ )} + + +
+ + +
+ + + + + + + {formErrors.name && ( + + + + )} +
-

- Name is required -

-
- -
- - -
- - - - - - - - - + {formErrors.name && ( +

+ Name is required +

+ )}
-

- Email is required -

-
- -
- - -
-