Skip to content

Commit 0f377a6

Browse files
committedJun 5, 2024
first commit
0 parents  commit 0f377a6

22 files changed

+5903
-0
lines changed
 

‎.eslintrc.cjs

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
module.exports = {
2+
root: true,
3+
env: { browser: true, es2020: true },
4+
extends: [
5+
'eslint:recommended',
6+
'plugin:react/recommended',
7+
'plugin:react/jsx-runtime',
8+
'plugin:react-hooks/recommended',
9+
],
10+
ignorePatterns: ['dist', '.eslintrc.cjs'],
11+
parserOptions: { ecmaVersion: 'latest', sourceType: 'module' },
12+
settings: { react: { version: '18.2' } },
13+
plugins: ['react-refresh'],
14+
rules: {
15+
'react/jsx-no-target-blank': 'off',
16+
'react-refresh/only-export-components': [
17+
'warn',
18+
{ allowConstantExport: true },
19+
],
20+
},
21+
}

‎.gitignore

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
# Logs
2+
logs
3+
*.log
4+
npm-debug.log*
5+
yarn-debug.log*
6+
yarn-error.log*
7+
pnpm-debug.log*
8+
lerna-debug.log*
9+
10+
node_modules
11+
dist
12+
dist-ssr
13+
*.local
14+
15+
# Editor directories and files
16+
.vscode/*
17+
!.vscode/extensions.json
18+
.idea
19+
.DS_Store
20+
*.suo
21+
*.ntvs*
22+
*.njsproj
23+
*.sln
24+
*.sw?

‎README.md

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# React + Vite
2+
3+
This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
4+
5+
Currently, two official plugins are available:
6+
7+
- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh
8+
- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh

‎index.html

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<!doctype html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="UTF-8" />
5+
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
7+
<title>Vite + React</title>
8+
</head>
9+
<body>
10+
<div id="root"></div>
11+
<script type="module" src="/src/main.jsx"></script>
12+
</body>
13+
</html>

‎package-lock.json

+4,978
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
{
2+
"name": "react-mid-project",
3+
"private": true,
4+
"version": "0.0.0",
5+
"type": "module",
6+
"scripts": {
7+
"dev": "vite",
8+
"build": "vite build",
9+
"lint": "eslint . --ext js,jsx --report-unused-disable-directives --max-warnings 0",
10+
"preview": "vite preview"
11+
},
12+
"dependencies": {
13+
"@emotion/react": "^11.11.4",
14+
"@emotion/styled": "^11.11.5",
15+
"@mui/material": "^5.15.19",
16+
"axios": "^1.7.0",
17+
"react": "^18.2.0",
18+
"react-dom": "^18.2.0",
19+
"uuid": "^9.0.1"
20+
},
21+
"devDependencies": {
22+
"@types/react": "^18.2.66",
23+
"@types/react-dom": "^18.2.22",
24+
"@vitejs/plugin-react": "^4.2.1",
25+
"eslint": "^8.57.0",
26+
"eslint-plugin-react": "^7.34.1",
27+
"eslint-plugin-react-hooks": "^4.6.0",
28+
"eslint-plugin-react-refresh": "^0.4.6",
29+
"vite": "^5.2.0"
30+
}
31+
}

‎public/vite.svg

+1
Loading

‎src/App.css

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
#root {
2+
max-width: 1280px;
3+
margin: 0 auto;
4+
padding: 2rem;
5+
/* text-align: center; */
6+
}

‎src/App.jsx

+154
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,154 @@
1+
import React, { useState, useEffect } from "react"
2+
import { getUsersData, getUserTodos, getUserPosts } from "./utils"
3+
import './App.css'
4+
import Users from './Components/Users'
5+
import TodosPosts from "./Components/TodosPosts";
6+
import User from "./Components/User";
7+
import AddUser from "./Components/AddUser";
8+
import LoadingSpinner from "./Components/LoadingSpinner";
9+
10+
function App() {
11+
const [users, setUsers] = useState([]);
12+
const [search, setSearch] = useState("")
13+
const [showTP, setShowTP] = useState(false)
14+
const [showAddUser, setShowAddUser] = useState(false)
15+
const [chosenUserId, setChosenUserId] = useState(0)
16+
const [userTodos, setUserTodos] = useState([])
17+
const [userPosts, setUserPosts] = useState([])
18+
19+
useEffect(() => {
20+
const fetchData = async () => {
21+
const users_info = await getUsersData()
22+
setUsers(users_info)
23+
}
24+
fetchData()
25+
}, []);
26+
27+
// filter the users according to username and email and pass it on to Users comp
28+
const filteredUsers = users.filter(user => {
29+
const username = user.name.toLocaleLowerCase()
30+
const useremail = user.email.toLocaleLowerCase()
31+
const searchLowerCase = search.toLocaleLowerCase()
32+
return username.includes(searchLowerCase) || useremail.includes(searchLowerCase)
33+
})
34+
35+
// function to handle update user's info from User comp
36+
// only replace the updated user info instead the old info
37+
const handleUpdate = (updated_user) => {
38+
console.log(updated_user)
39+
setUsers(users.map(user => {
40+
return user.id === updated_user.id ? updated_user : user
41+
}))
42+
}
43+
44+
const handleDelete = (id) => {
45+
setUsers(users.filter(user => user.id !== id))
46+
if (chosenUserId === id && showTP)
47+
setShowTP(false)
48+
}
49+
50+
const showFillAddUser = () => {
51+
setShowTP(false)
52+
setShowAddUser(true)
53+
}
54+
55+
const hideFillAddUser = () => {
56+
setShowAddUser(false)
57+
setChosenUserId(0)
58+
}
59+
60+
// function to transfer to User comp to handle the showing of todos and posts
61+
const showTodosPosts = async (id) => {
62+
const chosenUser = users.find(user => user.id === id)
63+
setUserTodos(chosenUser.todosUser)
64+
setUserPosts(chosenUser.postsUser)
65+
66+
if (chosenUserId === id && showTP) {
67+
// if its the same user that we click on, we want to close his section of todos and posts
68+
// and setTheChosenUserId to 0 to indicate no user todos/posts chosen
69+
setShowTP(false)
70+
setChosenUserId(0)
71+
}
72+
else { // if its another user or the same as before (but with his section closed), we want to open his section of todos and posts
73+
setShowTP(true)
74+
setShowAddUser(false)
75+
// this is the id of the current user that his todos and posts are presented
76+
setChosenUserId(id)
77+
}
78+
}
79+
80+
// function to update user's todo to "completed: true" and update all the users
81+
const updateUsers = (user_id, todo_id) => {
82+
// console.log(user_id, todo_id)
83+
setUsers(users.map(user => {
84+
if (user.id === user_id) {
85+
const updated_todos = user.todosUser.todos.map(todo => {
86+
if (todo.id !== todo_id)
87+
return todo
88+
else
89+
return {...todo, completed: true}
90+
})
91+
return {...user, todosUser: {userId: user.id, todos: updated_todos}}
92+
}
93+
return user
94+
}))
95+
}
96+
97+
// function to recieve data object that could be either be todo or post
98+
// the function distinguishes between todo or post using `isTodo` property
99+
const addTodoOrPost = (data) => {
100+
setUsers(users.map(user => {
101+
if (user.id === data.userId) {
102+
if (data.isTodo) { // data is new Todo
103+
const {isTodo, ...newTodo} = data
104+
user.todosUser.todos.push(newTodo)
105+
} else { // data is new Post
106+
const {isTodo, ...newPost} = data
107+
user.postsUser.posts.push(newPost)
108+
}
109+
}
110+
return user
111+
}))
112+
}
113+
114+
const addNewUser = (new_user) => {
115+
const new_users = users
116+
new_users.push(new_user)
117+
setUsers(new_users)
118+
}
119+
120+
// update the todos and posts that are presented in the todos-list when users state is updated
121+
useEffect(() => {
122+
const chosenUser = users.find(user => user.id === chosenUserId)
123+
setUserTodos(chosenUser?.todosUser)
124+
setUserPosts(chosenUser?.postsUser)
125+
}, [users]);
126+
127+
return (
128+
<div>
129+
{/* {console.log(users)} */}
130+
{ users.length != 0 ? <div className="full_app">
131+
<div className="searchAndUsers">
132+
<div className="searchInput">
133+
<label>Search </label>
134+
<input onChange={e => setSearch(e.target.value)} type="text"></input>
135+
<button onClick={() => showFillAddUser()}>Add</button> <br/>
136+
</div>
137+
<div className="scrollBarUsers">
138+
{
139+
filteredUsers.map(user => {
140+
return <User key={user.id} user_info={user} handleUpdate={handleUpdate} handleDelete={handleDelete} showTodosPosts={showTodosPosts} chosenUserId={chosenUserId} isNewUserOpen={showAddUser}/>
141+
})
142+
}
143+
</div>
144+
</div>
145+
<div>
146+
{ showTP && <TodosPosts userTodos={userTodos} userPosts={userPosts} updateUsers={updateUsers} addTodoOrPost={addTodoOrPost}/> }
147+
{ showAddUser && <AddUser hideFillAddUser={hideFillAddUser} addNewUser={addNewUser}/> }
148+
</div>
149+
</div> : <LoadingSpinner />}
150+
</div>
151+
)
152+
}
153+
154+
export default App

‎src/Components/AddPost.jsx

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
import React, { useEffect, useState } from "react"
2+
import { v4 as uuid } from 'uuid'
3+
4+
export default function AddPost({userId, returnPostsList, addTodoOrPost}) {
5+
const [title, setTitle] = useState("");
6+
const [body, setBody] = useState("");
7+
const [newPost, setNewPost] = useState({isTodo: false, userId, id: uuid(), title, body})
8+
9+
useEffect(() => {
10+
setNewPost({...newPost, title, body})
11+
}, [title, body]);
12+
13+
const handleNewPost = () => {
14+
if (title === "" || body === "")
15+
alert("Please fill the title and the body for the new post")
16+
else {
17+
addTodoOrPost(newPost)
18+
returnPostsList()
19+
}
20+
}
21+
22+
return (
23+
<div className="addPost-div">
24+
<div className="addPost-row">
25+
<label>Title: </label>
26+
<input onChange={(e) => setTitle(e.target.value)} type="text"></input> <br/>
27+
</div>
28+
<div className="addPost-row">
29+
<label>Body: </label>
30+
<input onChange={(e) => setBody(e.target.value)} type="text"></input> <br/>
31+
</div>
32+
<div className="addPost-button-group">
33+
<button onClick={() => returnPostsList()}>Cancel</button>
34+
<button onClick={handleNewPost}>Add</button>
35+
</div>
36+
</div>
37+
)
38+
};

‎src/Components/AddTodo.jsx

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import React, { useEffect, useState } from "react"
2+
import { v4 as uuid } from 'uuid'
3+
4+
5+
export default function AddTodo({userId, returnTodosList, addTodoOrPost}) {
6+
const [title, setTitle] = useState("")
7+
const [newTodo, setNewTodo] = useState({isTodo: true, userId, id: uuid(), title, completed: false})
8+
9+
useEffect(() => {
10+
setNewTodo({...newTodo, title})
11+
}, [title]);
12+
13+
const handleNewTodo = () => {
14+
if (title === "")
15+
alert("Please fill the title for the new todo")
16+
else {
17+
addTodoOrPost(newTodo)
18+
returnTodosList()
19+
}
20+
}
21+
22+
return (
23+
<div className="addTodo-div">
24+
<div className="addTodo-row">
25+
<label>Title: </label>
26+
<input onChange={(e) => setTitle(e.target.value)} type="text"></input> <br/>
27+
</div>
28+
<div className="addTodo-button-group">
29+
<button onClick={() => returnTodosList()}>Cancel</button>
30+
<button onClick={handleNewTodo}>Add</button>
31+
</div>
32+
</div>
33+
)
34+
};

‎src/Components/AddUser.jsx

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
import React, { useEffect, useState } from "react"
2+
import { v4 as uuid } from 'uuid'
3+
4+
export default function AddUser({hideFillAddUser, addNewUser}) {
5+
const id = uuid().slice(0,6)
6+
const [name, setName] = useState("")
7+
const [email, setEmail] = useState("")
8+
const [user, setUser] = useState({
9+
id: id,
10+
name,
11+
email,
12+
street: "",
13+
city: "",
14+
zipcode: "",
15+
todosUser: {userId: id, todos: []},
16+
postsUser: {userId: id, posts: []}
17+
})
18+
19+
useEffect(() => {
20+
setUser({...user, name, email})
21+
}, [name, email]);
22+
23+
const HandleNewUser = () => {
24+
if (name === "" || email === "")
25+
alert("Please fill the Name and the Email for the new user")
26+
else {
27+
addNewUser(user)
28+
hideFillAddUser()
29+
}
30+
}
31+
32+
return (
33+
<div className="addUser-div">
34+
<div className="addUser-container">
35+
<div className="addUser-row">
36+
<label>Name: </label>
37+
<input onChange={(e) => setName(e.target.value)} type="text"></input> <br/>
38+
</div>
39+
<div className="addUser-row">
40+
<label>Email: </label>
41+
<input onChange={(e) => setEmail(e.target.value)} type="text"></input> <br/>
42+
</div>
43+
<div className="addUser-button-group">
44+
<button onClick={() => hideFillAddUser()}>Cancel</button>
45+
<button onClick={HandleNewUser}>Add</button>
46+
</div>
47+
</div>
48+
</div>
49+
)
50+
};

‎src/Components/LoadingSpinner.jsx

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import React from "react"
2+
import Backdrop from '@mui/material/Backdrop';
3+
import CircularProgress from '@mui/material/CircularProgress';
4+
5+
export default function LoadingSpinner() {
6+
return (
7+
<div>
8+
<Backdrop
9+
sx={{ color: '#fff', zIndex: (theme) => theme.zIndex.drawer + 1 }}
10+
open >
11+
<CircularProgress color="inherit" />
12+
</Backdrop>
13+
</div>
14+
)
15+
};

‎src/Components/Posts.jsx

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import React from "react"
2+
3+
export default function Posts({post}) {
4+
return (
5+
<div className="posts-div">
6+
<div className="post-row">
7+
<span className="post-label"><strong>Title: </strong>{post.title}</span>
8+
</div>
9+
<div className="post-row">
10+
<span className="post-label"><strong>Body: </strong>{post.body}</span>
11+
</div>
12+
</div>
13+
)
14+
};

‎src/Components/Todos.jsx

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import React, { useEffect, useState } from "react"
2+
3+
export default function Todos({todo, updateUsers}) {
4+
5+
const [todoInfo, setTodoInfo] = useState({})
6+
7+
useEffect(() => {
8+
setTodoInfo(todo)
9+
}, [todo]);
10+
11+
const handleClick = () => {
12+
setTodoInfo({...todoInfo, completed: true})
13+
updateUsers(todoInfo.userId, todoInfo.id)
14+
}
15+
16+
return (
17+
<div className="todos-div">
18+
{/* {console.log(todo)} */}
19+
<div className="todo-row">
20+
<span className="todo-label"><strong>Title: </strong>{todo.title}</span>
21+
</div>
22+
<div className="todo-row">
23+
<span className="todo-label"><strong>Completed: </strong>{String(todo.completed)}</span>
24+
{!todo.completed && <button onClick={() => handleClick(todo.id)}>Mark Completed</button>}
25+
</div>
26+
</div>
27+
)
28+
};

‎src/Components/TodosPosts.jsx

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import React, { useState } from "react"
2+
import Todos from "./Todos"
3+
import Posts from "./Posts"
4+
import AddTodo from "./AddTodo"
5+
import AddPost from "./AddPost"
6+
7+
export default function TodosPosts({userTodos, userPosts, updateUsers, addTodoOrPost}) {
8+
const [addNewTodo, setAddNewTodo] = useState(false)
9+
const [addNewPost, setAddNewPost] = useState(false)
10+
11+
const returnTodosList = () => {
12+
setAddNewTodo(false)
13+
}
14+
15+
const returnPostsList = () => {
16+
setAddNewPost(false)
17+
}
18+
19+
return (
20+
<div className="todos-posts-div">
21+
{/* Todos Section */}
22+
<div className="todos-posts-header">
23+
{/* {console.log(userTodos)} */}
24+
<label className="todos-posts-label">Todos - User {userTodos.userId}</label>
25+
{ !addNewTodo && <button className="todos-posts-button" onClick={() => setAddNewTodo(true)}>Add</button> }
26+
</div>
27+
<div className="todos-posts-container">
28+
{
29+
!addNewTodo && userTodos.todos.map((todo, index) => {
30+
return <Todos key={index} todo={todo} updateUsers={updateUsers}/>
31+
})
32+
}
33+
{addNewTodo && <AddTodo userId={userTodos.userId} returnTodosList={returnTodosList} addTodoOrPost={addTodoOrPost}/>}
34+
</div> <br/>
35+
36+
{/* Posts Section */}
37+
<div className="todos-posts-header">
38+
<label className="todos-posts-label">Posts - User {userPosts.userId}</label>
39+
{ !addNewPost && <button className="todos-posts-button" onClick={() => setAddNewPost(true)} >Add</button> }
40+
</div>
41+
<div className="todos-posts-container">
42+
{
43+
!addNewPost && userPosts.posts.map((post, index) => {
44+
return <Posts key={index} post={post}/>
45+
})
46+
}
47+
{ addNewPost && <AddPost userId={userPosts.userId} returnPostsList={returnPostsList} addTodoOrPost={addTodoOrPost}/>}
48+
</div>
49+
</div>
50+
)
51+
};

‎src/Components/User.jsx

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
import React, { useEffect, useState } from "react"
2+
3+
export default function User({user_info, handleUpdate, handleDelete, showTodosPosts, chosenUserId, isNewUserOpen}) {
4+
const [isCompleated, setIsCompleated] = useState(false)
5+
const [otherData, setOtherData] = useState(false)
6+
const [updated_user, setUpdated_user] = useState(user_info)
7+
const [isColor, setIsColor] = useState(false)
8+
9+
// to synchronize the highlighted user when pressing the "Add" new user button
10+
useEffect(() => {
11+
if (isNewUserOpen)
12+
setIsColor(false)
13+
}, [isNewUserOpen]);
14+
15+
// to switch highlighted user when other user is pressed
16+
useEffect(() => {
17+
if (chosenUserId !== user_info.id && isColor)
18+
setIsColor(false)
19+
}, [chosenUserId]);
20+
21+
// check if all todos are completed for coloring the frame
22+
useEffect(() => {
23+
const fetchTaskStatus = async () => {
24+
const todos = user_info.todosUser.todos
25+
setIsCompleated(todos.every(todo => todo.completed))
26+
}
27+
fetchTaskStatus()
28+
}, [showTodosPosts]);
29+
30+
// handle changes of user's properties
31+
const handleChange = (e) => {
32+
const {name, value} = e.target
33+
setUpdated_user({...updated_user, [name]: value})
34+
}
35+
36+
// Click on user's id for open/close his section of todos and posts
37+
const colorAndShowTP = () => {
38+
setIsColor(!isColor)
39+
showTodosPosts(user_info.id)
40+
}
41+
42+
return (
43+
// if "Add" user button was NOT pressed, check indication of selected user click
44+
// and valid id (for open or close -> orange or white)
45+
<div className={`user-container ${isCompleated ? 'completed' : ''} ${isNewUserOpen ? '' : (isColor && chosenUserId !== 0 ? 'highlighted' : '')}`}>
46+
<div className="user-row">
47+
<label>ID:</label>
48+
<span className="clickable-label" onClick={colorAndShowTP}>{user_info.id}</span>
49+
</div>
50+
<div className="user-row">
51+
<label>Name:</label>
52+
<input name="name" type="text" defaultValue={user_info.name} onChange={handleChange}></input>
53+
</div>
54+
<div className="user-row">
55+
<label>Email:</label>
56+
<input name="email" type="text" defaultValue={user_info.email} onChange={handleChange}></input>
57+
</div>
58+
59+
<div className="buttons-row">
60+
<button className="otherData-button" onMouseOver={e => setOtherData(!otherData)}>Other Data</button>
61+
</div>
62+
63+
{otherData && <div className="otherData-div">
64+
<div className="otherData-row">
65+
<label>Street:</label>
66+
<input name="street" type="text" defaultValue={user_info.street} onChange={handleChange}></input> <br/>
67+
</div>
68+
<div className="otherData-row">
69+
<label>City:</label>
70+
<input name="city" type="text" defaultValue={user_info.city} onChange={handleChange}></input> <br/>
71+
</div>
72+
<div className="otherData-row">
73+
<label>Zip Code:</label>
74+
<input name="zipcode" type="text" defaultValue={user_info.zipcode} onChange={handleChange}></input> <br/>
75+
</div>
76+
</div>
77+
}
78+
79+
<div className="button-group">
80+
<button onClick={() => handleUpdate({...updated_user, todosUser: user_info.todosUser, postsUser: user_info.postsUser})}>Update</button>
81+
<button onClick={() => handleDelete(user_info.id)}>Delete</button>
82+
</div>
83+
84+
</div>
85+
)
86+
};

‎src/Components/Users.jsx

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import React, { useState, useEffect } from "react"
2+
import { getUsersData } from "../utils"
3+
import User from "./User";
4+
5+
export default function Users({users, handleUpdate, handleDelete, showTodosPosts, chosenUserId}) {
6+
7+
return (
8+
<div>
9+
{/* {console.log(users)} */}
10+
{
11+
users.map((user) => {
12+
return <User key={user.id} user_info={user} handleUpdate={handleUpdate} handleDelete={handleDelete} showTodosPosts={showTodosPosts} chosenUserId={chosenUserId}/>
13+
})
14+
}
15+
</div>
16+
)
17+
};

‎src/index.css

+227
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,227 @@
1+
:root {
2+
font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;
3+
line-height: 1.5;
4+
font-weight: 400;
5+
6+
color-scheme: light dark;
7+
color: rgba(255, 255, 255, 0.87);
8+
background-color: #242424;
9+
10+
font-synthesis: none;
11+
text-rendering: optimizeLegibility;
12+
-webkit-font-smoothing: antialiased;
13+
-moz-osx-font-smoothing: grayscale;
14+
}
15+
16+
body {
17+
margin: 0;
18+
min-width: 320px;
19+
min-height: 100vh;
20+
}
21+
22+
h1 {
23+
font-size: 3.2em;
24+
line-height: 1.1;
25+
}
26+
27+
button {
28+
border-radius: 8px;
29+
border: 1px solid transparent;
30+
padding: 0.6em 1.2em;
31+
font-size: 1em;
32+
font-weight: 500;
33+
font-family: inherit;
34+
cursor: pointer;
35+
transition: border-color 0.25s;
36+
}
37+
button:hover {
38+
border-color: #646cff;
39+
}
40+
41+
.full_app {
42+
display: flex;
43+
gap: 10px;
44+
}
45+
46+
.searchAndUsers {
47+
border: 2px solid black;
48+
border-radius: 45px;
49+
padding: 15px;
50+
margin-top: 10px;
51+
}
52+
53+
.searchInput {
54+
margin-bottom: 10px;
55+
}
56+
57+
.scrollBarUsers {
58+
flex-grow: 1;
59+
max-height: 100vh;
60+
overflow-y: auto;
61+
border: 1px solid #ccc;
62+
border-radius: 15px;
63+
padding: 15px;
64+
}
65+
66+
.searchInput label {
67+
flex: 0 0 60px;
68+
margin-right: 10px;
69+
}
70+
71+
.searchInput input {
72+
margin-right: 10px;
73+
}
74+
75+
.todos-posts-button, .todo-row button, .searchInput button {
76+
padding: 5px 10px;
77+
}
78+
79+
.user-container {
80+
background-color: white;
81+
border: 4px solid red;
82+
width: 400px;
83+
margin-bottom: 10px;
84+
padding: 15px;
85+
}
86+
87+
.user-container.completed {
88+
border-color: green;
89+
}
90+
91+
.user-container.highlighted {
92+
background-color: #ffc664;
93+
}
94+
95+
.clickable-label {
96+
cursor: pointer; /* Indicates the label is clickable */
97+
color: blue; /* Change text color to blue */
98+
font-size: larger;
99+
}
100+
101+
.clickable-label:hover {
102+
color: rgb(139, 0, 0); /* Change text color on hover */
103+
}
104+
105+
.addPost-row,
106+
.addTodo-row,
107+
.otherData-row,
108+
.user-row,
109+
.addUser-row {
110+
display: flex;
111+
align-items: center;
112+
margin-bottom: 10px;
113+
font-weight: bold;
114+
}
115+
116+
.user-row label,
117+
.addTodo-row label,
118+
.addPost-row label,
119+
.addUser-row label {
120+
flex: 0 0 60px;
121+
margin-right: 10px;
122+
}
123+
124+
input {
125+
flex: 1;
126+
padding: 5px;
127+
box-sizing: border-box;
128+
background-color: transparent;
129+
}
130+
131+
.otherData-button {
132+
background-color: lightgray
133+
}
134+
135+
.otherData-div {
136+
border: 2px solid black;
137+
border-radius: 25px;
138+
padding: 15px;
139+
margin-top: 10px;
140+
font-weight: bold;
141+
}
142+
143+
.otherData-row label {
144+
flex: 0 0 80px;
145+
margin-right: 5px;
146+
}
147+
148+
.button-group {
149+
display: flex;
150+
justify-content: flex-start;
151+
gap: 10px;
152+
margin-top: 10px;
153+
margin-left: 50%;
154+
}
155+
156+
.addTodo-button-group,
157+
.addPost-button-group,
158+
.addUser-button-group {
159+
display: flex;
160+
justify-content: flex-start;
161+
gap: 10px;
162+
margin-top: 40px;
163+
margin-left: 40%;
164+
}
165+
166+
.todos-posts-div, .addUser-div {
167+
margin-top: 60px;
168+
margin-left: 50px;
169+
}
170+
171+
.todos-posts-header, .buttons-row, .post-row, .todo-row {
172+
display: flex;
173+
justify-content: space-between;
174+
align-items: center;
175+
margin-bottom: 5px;
176+
}
177+
178+
.todos-posts-container, .addUser-container {
179+
border: 2px solid black;
180+
width: 400px;
181+
margin-bottom: 20px;
182+
}
183+
184+
.todos-posts-container {
185+
flex-grow: 1;
186+
max-height: 45vh;
187+
overflow-y: auto;
188+
}
189+
190+
.addUser-container {
191+
margin: 30px;
192+
padding: 30px;
193+
}
194+
195+
.todos-posts-label {
196+
font-weight: bold;
197+
}
198+
199+
.posts-div, .todos-div {
200+
border: 2px solid pink;
201+
width: auto;
202+
margin: 10px;
203+
padding: 10px;
204+
}
205+
206+
.addTodo-div, .addPost-div {
207+
margin: 30px;
208+
padding: 30px;
209+
}
210+
211+
.post-label, .todo-label {
212+
margin-left: 10px;
213+
}
214+
215+
216+
@media (prefers-color-scheme: light) {
217+
:root {
218+
color: #0e1a24;
219+
background-color: #ffffff;
220+
}
221+
a:hover {
222+
color: #747bff;
223+
}
224+
button {
225+
background-color: rgb(233, 229, 1);
226+
}
227+
}

‎src/main.jsx

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import React from 'react'
2+
import ReactDOM from 'react-dom/client'
3+
import App from './App.jsx'
4+
import './index.css'
5+
6+
ReactDOM.createRoot(document.getElementById('root')).render(
7+
8+
<App />
9+
10+
)

‎src/utils.js

+90
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,90 @@
1+
import axios from 'axios';
2+
3+
const users_url = "https://jsonplaceholder.typicode.com/users"
4+
const todos_url = "https://jsonplaceholder.typicode.com/todos"
5+
const posts_url = "https://jsonplaceholder.typicode.com/posts"
6+
7+
8+
const getUsersData = async () => {
9+
const { data: users } = await axios.get(users_url)
10+
const users_ids = users.map(user => user.id)
11+
const users_info = []
12+
// const users_todos = []
13+
// const users_posts = []
14+
for (const id of users_ids) {
15+
const user = await getUserBasicData(id)
16+
users_info.push(user)
17+
// const user_todos = await getUserTodos(id)
18+
// users_todos.push(user_todos)
19+
// const user_posts = await getUserPosts(id)
20+
// users_posts.push(user_posts)
21+
}
22+
return users_info
23+
}
24+
25+
const getUserBasicData = async (id) => {
26+
const { data: user } = await axios.get(users_url+'/'+id)
27+
const { data: todos } = await axios.get(todos_url+'?userId='+id)
28+
const { data: posts } = await axios.get(posts_url+'?userId='+id)
29+
30+
// const todos_slice = todos.slice(0, 2)
31+
// const posts_slice = posts.slice(0, 2)
32+
return {
33+
id: user.id,
34+
name: user.name,
35+
email: user.email,
36+
street: user.address.street,
37+
city: user.address.city,
38+
zipcode: user.address.zipcode,
39+
todosUser: {userId: user.id, todos},
40+
postsUser: {userId: user.id, posts}
41+
}
42+
}
43+
44+
const getUserTodos = async (id) => {
45+
const { data: todos } = await axios.get(todos_url+'?userId='+id)
46+
const todos_slice = todos.slice(0, 2)
47+
return {userId: id, todos: todos_slice}
48+
}
49+
50+
const getUserPosts = async (id) => {
51+
const { data: posts } = await axios.get(posts_url+'?userId='+id)
52+
const posts_slice = posts.slice(0, 2)
53+
return {userId: id, posts: posts_slice}
54+
}
55+
56+
/*const getUserFullData = async (id) => {
57+
const { data: user } = await axios.get(users_url+'/'+id)
58+
const { data: todos } = await axios.get(todos_url+'?userId='+id)
59+
const { data: posts } = await axios.get(posts_url+'?userId='+id)
60+
61+
const todos_slice = todos.slice(0, 2)
62+
// const todos_title = todos_slice.map(todo => todo.title)
63+
const posts_slice = posts.slice(0, 2)
64+
// const posts_title = posts_slice.map(post => post.title)
65+
return {
66+
id: user.id,
67+
name: user.name,
68+
email: user.email,
69+
street: user.address.street,
70+
city: user.address.city,
71+
zipcode: user.address.zipcode,
72+
todos: todos_slice,
73+
posts: posts_slice
74+
}
75+
}*/
76+
77+
// const checkTasksStatus = async (id) => {
78+
// const { data: todos } = await axios.get(todos_url+'?userId='+id)
79+
// const todos_slice = todos.slice(0, 2)
80+
// const unfinishedTasks = todos_slice.filter(todo => todo.completed === false)
81+
// return unfinishedTasks.length === 0 ? true : false
82+
// }
83+
84+
export {
85+
getUserBasicData,
86+
getUserTodos,
87+
getUserPosts,
88+
// getUserFullData,
89+
getUsersData,
90+
}

‎vite.config.js

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { defineConfig } from 'vite'
2+
import react from '@vitejs/plugin-react'
3+
4+
// https://vitejs.dev/config/
5+
export default defineConfig({
6+
plugins: [react()],
7+
})

0 commit comments

Comments
 (0)
Please sign in to comment.