-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathTodo.tsx
60 lines (54 loc) · 2.18 KB
/
Todo.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
'use client';
import { TrashIcon } from '@heroicons/react/24/outline';
import { Todo, User } from '@prisma/client';
import { useDeleteTodo, useUpdateTodo } from 'lib/hooks';
import { ChangeEvent } from 'react';
import Avatar from './Avatar';
import TimeInfo from './TimeInfo';
type Props = {
value: Todo & { owner: User };
optimistic?: boolean;
};
export default function TodoComponent({ value, optimistic }: Props) {
const { mutate: updateTodo } = useUpdateTodo({ optimisticUpdate: true });
const { mutate: deleteTodo } = useDeleteTodo({ optimisticUpdate: true });
const onDelete = () => {
deleteTodo({ where: { id: value.id } });
};
const onToggleCompleted = (completed: boolean) => {
if (completed === !!value.completedAt) {
return;
}
updateTodo({
where: { id: value.id },
data: { completedAt: completed ? new Date() : null },
});
};
return (
<div className="border rounded-lg px-8 py-4 shadow-lg flex flex-col items-center w-full lg:w-[480px]">
<div className="flex justify-between w-full mb-4">
<h3
className={`text-xl line-clamp-1 flex items-center
${value.completedAt ? 'line-through text-gray-400 italic' : 'text-gray-700'}
}`}
>
{value.title}
{optimistic && <span className="loading loading-spinner loading-sm ml-1"></span>}
</h3>
<div className="flex">
<input
type="checkbox"
className="checkbox mr-2"
checked={!!value.completedAt}
onChange={(e: ChangeEvent<HTMLInputElement>) => onToggleCompleted(e.currentTarget.checked)}
/>
<TrashIcon className="w-6 h-6 text-gray-500 cursor-pointer" onClick={onDelete} />
</div>
</div>
<div className="flex justify-end w-full space-x-2">
<TimeInfo value={value} />
<Avatar user={value.owner} size={18} />
</div>
</div>
);
}