c3f21958d2
The todoapp example contains a Netlify plan which uses the latest dagger additions: do & Client API. We are thinking of merging the examples repository into this one to make working with this easier. This is a step in that direction. We are not using the yarn package so that we can revert https://github.com/dagger/dagger/pull/1673 without breaking this implementation. The GitHub Action is WIP, we will continue with that tomorrow: https://github.com/dagger/dagger-for-github/issues/24 Signed-off-by: Gerhard Lazu <gerhard@lazu.co.uk>
121 lines
2.8 KiB
JavaScript
121 lines
2.8 KiB
JavaScript
import React, { useState, useRef, useEffect } from "react";
|
|
import Form from "./components/Form";
|
|
import FilterButton from "./components/FilterButton";
|
|
import Todo from "./components/Todo";
|
|
import { nanoid } from "nanoid";
|
|
|
|
|
|
function usePrevious(value) {
|
|
const ref = useRef();
|
|
useEffect(() => {
|
|
ref.current = value;
|
|
});
|
|
return ref.current;
|
|
}
|
|
|
|
const FILTER_MAP = {
|
|
All: () => true,
|
|
Active: task => !task.completed,
|
|
Completed: task => task.completed
|
|
};
|
|
|
|
const FILTER_NAMES = Object.keys(FILTER_MAP);
|
|
|
|
function App(props) {
|
|
const [tasks, setTasks] = useState(props.tasks);
|
|
const [filter, setFilter] = useState('All');
|
|
|
|
function toggleTaskCompleted(id) {
|
|
const updatedTasks = tasks.map(task => {
|
|
// if this task has the same ID as the edited task
|
|
if (id === task.id) {
|
|
// use object spread to make a new obkect
|
|
// whose `completed` prop has been inverted
|
|
return {...task, completed: !task.completed}
|
|
}
|
|
return task;
|
|
});
|
|
setTasks(updatedTasks);
|
|
}
|
|
|
|
|
|
function deleteTask(id) {
|
|
const remainingTasks = tasks.filter(task => id !== task.id);
|
|
setTasks(remainingTasks);
|
|
}
|
|
|
|
|
|
function editTask(id, newName) {
|
|
const editedTaskList = tasks.map(task => {
|
|
// if this task has the same ID as the edited task
|
|
if (id === task.id) {
|
|
//
|
|
return {...task, name: newName}
|
|
}
|
|
return task;
|
|
});
|
|
setTasks(editedTaskList);
|
|
}
|
|
|
|
const taskList = tasks
|
|
.filter(FILTER_MAP[filter])
|
|
.map(task => (
|
|
<Todo
|
|
id={task.id}
|
|
name={task.name}
|
|
completed={task.completed}
|
|
key={task.id}
|
|
toggleTaskCompleted={toggleTaskCompleted}
|
|
deleteTask={deleteTask}
|
|
editTask={editTask}
|
|
/>
|
|
));
|
|
|
|
const filterList = FILTER_NAMES.map(name => (
|
|
<FilterButton
|
|
key={name}
|
|
name={name}
|
|
isPressed={name === filter}
|
|
setFilter={setFilter}
|
|
/>
|
|
));
|
|
|
|
function addTask(name) {
|
|
const newTask = { id: "todo-" + nanoid(), name: name, completed: false };
|
|
setTasks([...tasks, newTask]);
|
|
}
|
|
|
|
|
|
const tasksNoun = taskList.length !== 1 ? 'tasks' : 'task';
|
|
const headingText = `${taskList.length} ${tasksNoun} remaining`;
|
|
|
|
const listHeadingRef = useRef(null);
|
|
const prevTaskLength = usePrevious(tasks.length);
|
|
|
|
useEffect(() => {
|
|
if (tasks.length - prevTaskLength === -1) {
|
|
listHeadingRef.current.focus();
|
|
}
|
|
}, [tasks.length, prevTaskLength]);
|
|
|
|
return (
|
|
<div className="todoapp stack-large">
|
|
<Form addTask={addTask} />
|
|
<div className="filters btn-group stack-exception">
|
|
{filterList}
|
|
</div>
|
|
<h2 id="list-heading" tabIndex="-1" ref={listHeadingRef}>
|
|
{headingText}
|
|
</h2>
|
|
<ul
|
|
className="todo-list stack-large stack-exception"
|
|
aria-labelledby="list-heading"
|
|
>
|
|
{taskList}
|
|
</ul>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default App;
|