Skip to content

Commit

Permalink
solution
Browse files Browse the repository at this point in the history
  • Loading branch information
Anya-Laban committed Sep 15, 2024
1 parent f1536f7 commit f5b443b
Show file tree
Hide file tree
Showing 13 changed files with 405 additions and 148 deletions.
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,4 +33,4 @@ Implement a simple [TODO app](https://mate-academy.github.io/react_todo-app/) th
- Implement a solution following the [React task guidelines](https://github.com/mate-academy/react_task-guideline#react-tasks-guideline).
- Use the [React TypeScript cheat sheet](https://mate-academy.github.io/fe-program/js/extra/react-typescript).
- Open another terminal and run tests with `npm test` to ensure your solution is correct.
- Replace `<your_account>` with your GitHub username in the [DEMO LINK](https://<your_account>.github.io/react_todo-app/) and add it to the PR description.
- Replace `<your_account>` with your GitHub username in the [DEMO LINK](https://Anya-Laban.github.io/react_todo-app/) and add it to the PR description.
154 changes: 11 additions & 143 deletions src/App.tsx
Original file line number Diff line number Diff line change
@@ -1,156 +1,24 @@
/* eslint-disable jsx-a11y/label-has-associated-control */
/* eslint-disable jsx-a11y/control-has-associated-label */
import React from 'react';
import React, { useContext } from 'react';
import { Header } from './Components/Header';
import { TodoList } from './Components/TodoList';
import { Footer } from './Components/Footer';
import { TodosContext } from './Components/TodosContext';

export const App: React.FC = () => {
const { todos } = useContext(TodosContext);

return (
<div className="todoapp">
<h1 className="todoapp__title">todos</h1>

<div className="todoapp__content">
<header className="todoapp__header">
{/* this button should have `active` class only if all todos are completed */}
<button
type="button"
className="todoapp__toggle-all active"
data-cy="ToggleAllButton"
/>

{/* Add a todo on form submit */}
<form>
<input
data-cy="NewTodoField"
type="text"
className="todoapp__new-todo"
placeholder="What needs to be done?"
/>
</form>
</header>

<section className="todoapp__main" data-cy="TodoList">
{/* This is a completed todo */}
<div data-cy="Todo" className="todo completed">
<label className="todo__status-label">
<input
data-cy="TodoStatus"
type="checkbox"
className="todo__status"
checked
/>
</label>

<span data-cy="TodoTitle" className="todo__title">
Completed Todo
</span>

{/* Remove button appears only on hover */}
<button type="button" className="todo__remove" data-cy="TodoDelete">
×
</button>
</div>

{/* This todo is an active todo */}
<div data-cy="Todo" className="todo">
<label className="todo__status-label">
<input
data-cy="TodoStatus"
type="checkbox"
className="todo__status"
/>
</label>

<span data-cy="TodoTitle" className="todo__title">
Not Completed Todo
</span>

<button type="button" className="todo__remove" data-cy="TodoDelete">
×
</button>
</div>

{/* This todo is being edited */}
<div data-cy="Todo" className="todo">
<label className="todo__status-label">
<input
data-cy="TodoStatus"
type="checkbox"
className="todo__status"
/>
</label>

{/* This form is shown instead of the title and remove button */}
<form>
<input
data-cy="TodoTitleField"
type="text"
className="todo__title-field"
placeholder="Empty todo will be deleted"
value="Todo is being edited now"
/>
</form>
</div>

{/* This todo is in loadind state */}
<div data-cy="Todo" className="todo">
<label className="todo__status-label">
<input
data-cy="TodoStatus"
type="checkbox"
className="todo__status"
/>
</label>

<span data-cy="TodoTitle" className="todo__title">
Todo is being saved now
</span>

<button type="button" className="todo__remove" data-cy="TodoDelete">
×
</button>
</div>
</section>

{/* Hide the footer if there are no todos */}
<footer className="todoapp__footer" data-cy="Footer">
<span className="todo-count" data-cy="TodosCounter">
3 items left
</span>

{/* Active link should have the 'selected' class */}
<nav className="filter" data-cy="Filter">
<a
href="#/"
className="filter__link selected"
data-cy="FilterLinkAll"
>
All
</a>

<a
href="#/active"
className="filter__link"
data-cy="FilterLinkActive"
>
Active
</a>
<Header />

<a
href="#/completed"
className="filter__link"
data-cy="FilterLinkCompleted"
>
Completed
</a>
</nav>
{!!todos.length && <TodoList />}

{/* this button should be disabled if there are no completed todos */}
<button
type="button"
className="todoapp__clear-completed"
data-cy="ClearCompletedButton"
>
Clear completed
</button>
</footer>
{!!todos.length && <Footer />}
</div>
</div>
);
Expand Down
52 changes: 52 additions & 0 deletions src/Components/Footer.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
import React, { useContext, useMemo } from 'react';
import { TodosContext } from './TodosContext';
import { Filter } from '../Types/Filter';
import classNames from 'classnames';

export const Footer: React.FC = () => {
const { todos, setTodos, filter, setFilter } = useContext(TodosContext);

const countActiveTodo = useMemo(() => {
return todos.reduce((count, todo) => count + Number(!todo.completed), 0);
}, [todos]);

const handleClearCompleted = () => {
const clearedTodos = [...todos].filter(todo => !todo.completed);

setTodos(clearedTodos);
};

return (
<footer className="todoapp__footer" data-cy="Footer">
<span className="todo-count" data-cy="TodosCounter">
{countActiveTodo} items left
</span>

<nav className="filter" data-cy="Filter">
{Object.keys(Filter).map(key => (
<a
key={key}
href={`#/${Filter[key as keyof typeof Filter]}`}
className={classNames('filter__link', {
selected: filter === Filter[key as keyof typeof Filter],
})}
data-cy={`FilterLink${key}`}
onClick={() => setFilter(Filter[key as keyof typeof Filter])}
>
{key}
</a>
))}
</nav>

<button
type="button"
className="todoapp__clear-completed"
data-cy="ClearCompletedButton"
onClick={handleClearCompleted}
disabled={todos.length === countActiveTodo}
>
Clear completed
</button>
</footer>
);
};
75 changes: 75 additions & 0 deletions src/Components/Header.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
import React, { useContext, useEffect, useMemo, useRef, useState } from 'react';
import { TodosContext } from './TodosContext';
import classNames from 'classnames';

export const Header: React.FC = () => {
const { todos, setTodos } = useContext(TodosContext);

const [value, setValue] = useState('');

const formField = useRef<HTMLInputElement>(null);

useEffect(() => {
if (formField.current) {
formField.current.focus();
}
}, [todos]);

const handleSubmit = (event: React.FormEvent<HTMLFormElement>) => {
event.preventDefault();

if (!!value.trim().length) {
setTodos([
...todos,
{
id: +new Date(),
title: value.trim(),
completed: false,
},
]);

setValue('');
}
};

const isEveryTodoCompleted = useMemo(
() => todos.every(todo => todo.completed),
[todos],
);

const handleChangeCompleted = () => {
const changedTodos = [...todos].map(todo => ({
...todo,
completed: isEveryTodoCompleted ? false : true,
}));

setTodos(changedTodos);
};

return (
<header className="todoapp__header">
{!!todos.length && (
<button
type="button"
className={classNames('todoapp__toggle-all', {
active: isEveryTodoCompleted,
})}
data-cy="ToggleAllButton"
onClick={handleChangeCompleted}
/>
)}

<form onSubmit={handleSubmit}>
<input
data-cy="NewTodoField"
type="text"
className="todoapp__new-todo"
placeholder="What needs to be done?"
ref={formField}
value={value}
onChange={event => setValue(event.target.value)}
/>
</form>
</header>
);
};
Loading

0 comments on commit f5b443b

Please sign in to comment.