0.1.3 #4
@@ -5,8 +5,14 @@ import { useInput, useOptionalInput } from "effect-fc/hooks"
|
|||||||
import * as React from "react"
|
import * as React from "react"
|
||||||
|
|
||||||
|
|
||||||
export type TextFieldInputProps<A, R> = Omit<useInput.Options<A, R>, "schema" | "equivalence"> & Omit<TextField.RootProps, "ref">
|
export type TextFieldInputProps<A, R> = (
|
||||||
export type TextFieldOptionalInputProps<A, R> = Omit<useOptionalInput.Options<A, R>, "schema" | "equivalence"> & Omit<TextField.RootProps, "ref">
|
& Omit<useInput.Options<A, R>, "schema" | "equivalence">
|
||||||
|
& Omit<TextField.RootProps, "ref">
|
||||||
|
)
|
||||||
|
export type TextFieldOptionalInputProps<A, R> = (
|
||||||
|
& Omit<useOptionalInput.Options<A, R>, "schema" | "equivalence">
|
||||||
|
& Omit<TextField.RootProps, "ref" | "defaultValue">
|
||||||
|
)
|
||||||
|
|
||||||
export const TextFieldInput = <A, R, O extends boolean = false>(options: {
|
export const TextFieldInput = <A, R, O extends boolean = false>(options: {
|
||||||
readonly optional?: O
|
readonly optional?: O
|
||||||
@@ -50,7 +56,7 @@ export const TextFieldInput = <A, R, O extends boolean = false>(options: {
|
|||||||
value={input.value}
|
value={input.value}
|
||||||
onChange={e => input.setValue(e.target.value)}
|
onChange={e => input.setValue(e.target.value)}
|
||||||
disabled={input.optional ? !input.enabled : undefined}
|
disabled={input.optional ? !input.enabled : undefined}
|
||||||
{...Struct.omit(props as TextFieldOptionalInputProps<A, R> | TextFieldInputProps<A, R>, "ref")}
|
{...Struct.omit(props as TextFieldOptionalInputProps<A, R> | TextFieldInputProps<A, R>, "ref", "defaultValue")}
|
||||||
/>
|
/>
|
||||||
</Flex>
|
</Flex>
|
||||||
|
|
||||||
|
|||||||
@@ -1,32 +1,38 @@
|
|||||||
import { DateTime, identity, ParseResult, Schema } from "effect"
|
import { DateTime, Effect, Option, ParseResult, Schema } from "effect"
|
||||||
|
|
||||||
|
|
||||||
export const DateTimeUtcFromZoned = Schema.transformOrFail(
|
export class DateTimeUtcFromZoned extends Schema.transformOrFail(
|
||||||
Schema.DateTimeZonedFromSelf,
|
Schema.DateTimeZonedFromSelf,
|
||||||
Schema.DateTimeUtcFromSelf,
|
Schema.DateTimeUtcFromSelf,
|
||||||
{
|
{
|
||||||
strict: true,
|
strict: true,
|
||||||
encode: DateTime.setZoneCurrent,
|
encode: DateTime.setZoneCurrent,
|
||||||
decode: zoned => ParseResult.succeed(DateTime.toUtc(zoned)),
|
decode: i => ParseResult.succeed(DateTime.toUtc(i)),
|
||||||
},
|
},
|
||||||
)
|
) {}
|
||||||
|
|
||||||
export const DateTimeZonedFromUtc = Schema.transformOrFail(
|
export class DateTimeZonedFromUtc extends Schema.transformOrFail(
|
||||||
Schema.DateTimeUtcFromSelf,
|
Schema.DateTimeUtcFromSelf,
|
||||||
Schema.DateTimeZonedFromSelf,
|
Schema.DateTimeZonedFromSelf,
|
||||||
{
|
{
|
||||||
strict: true,
|
strict: true,
|
||||||
encode: zoned => ParseResult.succeed(DateTime.toUtc(zoned)),
|
encode: a => ParseResult.succeed(DateTime.toUtc(a)),
|
||||||
decode: DateTime.setZoneCurrent,
|
decode: DateTime.setZoneCurrent,
|
||||||
},
|
},
|
||||||
)
|
) {}
|
||||||
|
|
||||||
export const DateTimeUtcFromZonedInput = Schema.transform(
|
export class DateTimeUtcFromZonedInput extends Schema.transformOrFail(
|
||||||
Schema.String,
|
Schema.String,
|
||||||
Schema.compose(Schema.DateTimeZoned, DateTimeUtcFromZoned),
|
DateTimeUtcFromZoned,
|
||||||
{
|
{
|
||||||
strict: true,
|
strict: true,
|
||||||
encode: v => v.slice(0, 16),
|
encode: a => ParseResult.succeed(DateTime.formatIsoZoned(a).slice(0, 16)),
|
||||||
decode: identity,
|
decode: (i, _, ast) => Effect.flatMap(
|
||||||
|
DateTime.CurrentTimeZone,
|
||||||
|
timeZone => Option.match(DateTime.makeZoned(i, { timeZone, adjustForTimeZone: true }), {
|
||||||
|
onSome: ParseResult.succeed,
|
||||||
|
onNone: () => ParseResult.fail(new ParseResult.Type(ast, i, `Unable to decode ${JSON.stringify(i)} into a DateTime.Zoned`)),
|
||||||
|
}),
|
||||||
|
),
|
||||||
},
|
},
|
||||||
)
|
) {}
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ import { TextAreaInput } from "@/lib/TextAreaInput"
|
|||||||
import { TextFieldInput } from "@/lib/TextFieldInput"
|
import { TextFieldInput } from "@/lib/TextFieldInput"
|
||||||
import { Box, Button, Flex, IconButton } from "@radix-ui/themes"
|
import { Box, Button, Flex, IconButton } from "@radix-ui/themes"
|
||||||
import { GetRandomValues, makeUuid4 } from "@typed/id"
|
import { GetRandomValues, makeUuid4 } from "@typed/id"
|
||||||
import { Chunk, Effect, Match, Option, Ref, Runtime, Schema, SubscriptionRef } from "effect"
|
import { Chunk, DateTime, Effect, Match, Option, Ref, Runtime, Schema, SubscriptionRef } from "effect"
|
||||||
import { Component, Memo } from "effect-fc"
|
import { Component, Memo } from "effect-fc"
|
||||||
import { Hooks } from "effect-fc/hooks"
|
import { Hooks, useOnce } from "effect-fc/hooks"
|
||||||
import { SubscriptionSubRef } from "effect-fc/types"
|
import { SubscriptionSubRef } from "effect-fc/types"
|
||||||
import { FaArrowDown, FaArrowUp } from "react-icons/fa"
|
import { FaArrowDown, FaArrowUp } from "react-icons/fa"
|
||||||
import { FaDeleteLeft } from "react-icons/fa6"
|
import { FaDeleteLeft } from "react-icons/fa6"
|
||||||
@@ -60,8 +60,9 @@ export class Todo extends Component.makeUntraced(function* Todo(props: TodoProps
|
|||||||
|
|
||||||
<Flex direction="row" justify="center" align="center" gap="2">
|
<Flex direction="row" justify="center" align="center" gap="2">
|
||||||
<OptionalDateInputFC
|
<OptionalDateInputFC
|
||||||
ref={completedAtRef}
|
|
||||||
type="datetime-local"
|
type="datetime-local"
|
||||||
|
ref={completedAtRef}
|
||||||
|
defaultValue={yield* useOnce(() => DateTime.now)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{props._tag === "new" &&
|
{props._tag === "new" &&
|
||||||
|
|||||||
Reference in New Issue
Block a user