import * as Domain from "@/domain" import { TextAreaInput } from "@/lib/input/TextAreaInput" import { TextFieldInput } from "@/lib/input/TextFieldInput" import { DateTimeUtcFromZonedInput } from "@/lib/schema" import { Box, Button, Flex, IconButton } from "@radix-ui/themes" import { GetRandomValues, makeUuid4 } from "@typed/id" import { Chunk, DateTime, Effect, Match, Option, Ref, Runtime, Schema, Stream, SubscriptionRef } from "effect" import { Component, Memo } from "effect-fc" import { useMemo, useOnce, useSubscribe } from "effect-fc/hooks" import { Subscribable, SubscriptionSubRef } from "effect-fc/types" import { FaArrowDown, FaArrowUp } from "react-icons/fa" import { FaDeleteLeft } from "react-icons/fa6" import { TodosState } from "./TodosState.service" const StringTextAreaInput = TextAreaInput({ schema: Schema.String }) const OptionalDateTimeInput = TextFieldInput({ optional: true, schema: DateTimeUtcFromZonedInput }) const makeTodo = makeUuid4.pipe( Effect.map(id => Domain.Todo.Todo.make({ id, content: "", completedAt: Option.none(), })), Effect.provide(GetRandomValues.CryptoRandom), ) export type TodoProps = ( | { readonly _tag: "new" } | { readonly _tag: "edit", readonly id: string } ) export class Todo extends Component.makeUntraced("Todo")(function*(props: TodoProps) { const runtime = yield* Effect.runtime() const state = yield* TodosState const { ref, indexRef, contentRef, completedAtRef } = yield* useMemo(() => Match.value(props).pipe( Match.tag("new", () => Effect.Do.pipe( Effect.bind("ref", () => Effect.andThen(makeTodo, SubscriptionRef.make)), Effect.let("indexRef", () => Subscribable.make({ get: Effect.succeed(-1), changes: Stream.empty })), )), Match.tag("edit", ({ id }) => Effect.Do.pipe( Effect.let("ref", () => state.getElementRef(id)), Effect.let("indexRef", () => state.getIndexSubscribable(id)), )), Match.exhaustive, Effect.let("contentRef", ({ ref }) => SubscriptionSubRef.makeFromPath(ref, ["content"])), Effect.let("completedAtRef", ({ ref }) => SubscriptionSubRef.makeFromPath(ref, ["completedAt"])), // eslint-disable-next-line react-hooks/exhaustive-deps ), [props._tag, props._tag === "edit" ? props.id : undefined]) const [index, size] = yield* useSubscribe(indexRef, state.sizeSubscribable) const StringTextAreaInputFC = yield* StringTextAreaInput const OptionalDateTimeInputFC = yield* OptionalDateTimeInput return ( DateTime.now)} /> {props._tag === "new" && } {props._tag === "edit" && Runtime.runSync(runtime)(state.moveLeft(props.id))} > = size - 1} onClick={() => Runtime.runSync(runtime)(state.moveRight(props.id))} > Runtime.runSync(runtime)(state.remove(props.id))}> } ) }).pipe(Memo.memo) {}