Add comments
All checks were successful
Lint / lint (push) Successful in 12s

This commit is contained in:
Julien Valverdé
2026-03-10 03:18:45 +01:00
parent 9b3ce62d3e
commit 6917c72101
2 changed files with 233 additions and 1 deletions

View File

@@ -4,25 +4,80 @@ import * as React from "react"
import * as Component from "./Component.js"
/**
* A unique symbol representing the Async component type.
* Used as a type brand to identify Async components.
*
* @experimental
*/
export const AsyncTypeId: unique symbol = Symbol.for("@effect-fc/Async/Async")
export type AsyncTypeId = typeof AsyncTypeId
/**
* The type of the Async type ID symbol.
*/
export type AsyncTypeId = typeof AsyncTypeId
/**
* An Async component that supports suspense and promise-based async operations.
* Combines Component behavior with Async-specific options.
*
* @example
* ```ts
* const MyAsyncComponent = async(component({ ... }))
* ```
*/
export interface Async extends AsyncPrototype, AsyncOptions {}
/**
* The prototype object for Async components containing their methods and behaviors.
*/
export interface AsyncPrototype {
/**
* The Async type ID brand.
*/
readonly [AsyncTypeId]: AsyncTypeId
}
/**
* Configuration options for Async components.
*/
export interface AsyncOptions {
/**
* The default fallback React node to display while the async operation is pending.
* Used if no fallback is provided to the component when rendering.
*/
readonly defaultFallback?: React.ReactNode
}
/**
* Props for Async components, extending React.SuspenseProps without the children prop.
* The children are managed internally by the Async component.
*/
export type AsyncProps = Omit<React.SuspenseProps, "children">
/**
* The prototype object for Async components.
* Provides the `asFunctionComponent` method for converting async components to React function components.
*
* @internal Use the `async` function to create Async components instead of accessing this directly.
*/
export const AsyncPrototype: AsyncPrototype = Object.freeze({
[AsyncTypeId]: AsyncTypeId,
/**
* Converts an Async component to a React function component.
*
* @param runtimeRef - A reference to the Effect runtime for executing effects
* @returns A React function component that suspends while the async operation is executing
*
* @example
* ```ts
* const MyComponent = component({ ... })
* const AsyncMyComponent = async(MyComponent)
* const FunctionComponent = AsyncMyComponent.asFunctionComponent(runtimeRef)
* ```
*/
asFunctionComponent<P extends {}, A extends React.ReactNode, E, R>(
this: Component.Component<P, A, E, R> & Async,
runtimeRef: React.RefObject<Runtime.Runtime<Exclude<R, Scope.Scope>>>,
@@ -46,6 +101,16 @@ export const AsyncPrototype: AsyncPrototype = Object.freeze({
},
} as const)
/**
* An equivalence function for comparing AsyncProps that ignores the `fallback` property.
* Useful for memoization and re-render optimization.
*
* @param self - The first props object to compare
* @param that - The second props object to compare
* @returns `true` if the props are equivalent (excluding fallback), `false` otherwise
*
* @internal
*/
export const defaultPropsEquivalence: Equivalence.Equivalence<AsyncProps> = (
self: Record<string, unknown>,
that: Record<string, unknown>,
@@ -71,8 +136,43 @@ export const defaultPropsEquivalence: Equivalence.Equivalence<AsyncProps> = (
}
/**
* A type guard to check if a value is an Async component.
*
* @param u - The value to check
* @returns `true` if the value is an Async component, `false` otherwise
*
* @example
* ```ts
* if (isAsync(component)) {
* // component is an Async component
* }
* ```
*/
export const isAsync = (u: unknown): u is Async => Predicate.hasProperty(u, AsyncTypeId)
/**
* Converts a Component into an Async component that supports suspense and promise-based async operations.
*
* The resulting component will wrap the original component with React.Suspense,
* allowing async Effect computations to suspend and resolve properly.
*
* Note: The component cannot have a prop named "promise" as it's reserved for internal use.
*
* @param self - The component to convert to an Async component
* @returns An Async component with the same body, error, and context types as the input
*
* @example
* ```ts
* const MyComponent = component({
* body: (props) => // ...
* })
*
* const AsyncMyComponent = async(MyComponent)
* ```
*
* @throws Will produce a type error if the component has a "promise" prop
*/
export const async = <T extends Component.Component<any, any, any, any>>(
self: T & (
"promise" extends keyof Component.Component.Props<T>
@@ -96,6 +196,27 @@ export const async = <T extends Component.Component<any, any, any, any>>(
)),
)
/**
* Applies options to an Async component, returning a new Async component with the updated configuration.
*
* Supports both curried and uncurried application styles.
*
* @param self - The Async component to apply options to (in uncurried form)
* @param options - The options to apply to the component
* @returns An Async component with the applied options
*
* @example
* ```ts
* const AsyncComponent = async(myComponent)
*
* // Uncurried
* const configured = withOptions(AsyncComponent, { defaultFallback: <Loading /> })
*
* // Curried
* const configurer = withOptions({ defaultFallback: <Loading /> })
* const configured = configurer(AsyncComponent)
* ```
*/
export const withOptions: {
<T extends Component.Component<any, any, any, any> & Async>(
options: Partial<AsyncOptions>

View File

@@ -4,23 +4,80 @@ import * as React from "react"
import type * as Component from "./Component.js"
/**
* A unique symbol representing the Memoized component type.
* Used as a type brand to identify Memoized components.
*
* @experimental
*/
export const MemoizedTypeId: unique symbol = Symbol.for("@effect-fc/Memoized/Memoized")
/**
* The type of the Memoized type ID symbol.
*/
export type MemoizedTypeId = typeof MemoizedTypeId
/**
* A Memoized component that uses React.memo to optimize re-renders based on prop equality.
* Combines Component behavior with Memoized-specific options.
*
* @template P The props type of the component
*
* @example
* ```ts
* const MyComponent = component({ ... })
* const MemoizedComponent = memoized(MyComponent)
* ```
*/
export interface Memoized<P> extends MemoizedPrototype, MemoizedOptions<P> {}
/**
* The prototype object for Memoized components containing their methods and behaviors.
*/
export interface MemoizedPrototype {
/**
* The Memoized type ID brand.
*/
readonly [MemoizedTypeId]: MemoizedTypeId
}
/**
* Configuration options for Memoized components.
*
* @template P The props type of the component
*/
export interface MemoizedOptions<P> {
/**
* An optional equivalence function for comparing component props.
* If provided, this function is used by React.memo to determine if props have changed.
* Returns `true` if props are equivalent (no re-render), `false` if they differ (re-render).
*/
readonly propsEquivalence?: Equivalence.Equivalence<P>
}
/**
* The prototype object for Memoized components.
* Provides the `transformFunctionComponent` method for memoizing React function components.
*
* @internal Use the `memoized` function to create Memoized components instead of accessing this directly.
*/
export const MemoizedPrototype: MemoizedPrototype = Object.freeze({
[MemoizedTypeId]: MemoizedTypeId,
/**
* Transforms a React function component by wrapping it with React.memo.
*
* @param f - The React function component to memoize
* @returns A memoized version of the component that uses the configured propsEquivalence function
*
* @example
* ```ts
* const MemoizedComponent = memoized(MyComponent)
* const Fn = MemoizedComponent.transformFunctionComponent((props) => <div>{props.x}</div>)
* ```
*/
transformFunctionComponent<P extends {}>(
this: Memoized<P>,
f: React.FC<P>,
@@ -30,8 +87,39 @@ export const MemoizedPrototype: MemoizedPrototype = Object.freeze({
} as const)
/**
* A type guard to check if a value is a Memoized component.
*
* @param u - The value to check
* @returns `true` if the value is a Memoized component, `false` otherwise
*
* @example
* ```ts
* if (isMemoized(component)) {
* // component is a Memoized component
* }
* ```
*/
export const isMemoized = (u: unknown): u is Memoized<unknown> => Predicate.hasProperty(u, MemoizedTypeId)
/**
* Converts a Component into a Memoized component that optimizes re-renders using React.memo.
*
* The resulting component will use React.memo to skip re-renders when props haven't changed,
* based on the configured equivalence function (or the default equality check).
*
* @param self - The component to convert to a Memoized component
* @returns A Memoized component with the same body, error, and context types as the input
*
* @example
* ```ts
* const MyComponent = component({
* body: (props) => // ...
* })
*
* const MemoizedComponent = memoized(MyComponent)
* ```
*/
export const memoized = <T extends Component.Component<any, any, any, any>>(
self: T
): T & Memoized<Component.Component.Props<T>> => Object.setPrototypeOf(
@@ -42,6 +130,29 @@ export const memoized = <T extends Component.Component<any, any, any, any>>(
)),
)
/**
* Applies options to a Memoized component, returning a new Memoized component with the updated configuration.
*
* Supports both curried and uncurried application styles.
*
* @param self - The Memoized component to apply options to (in uncurried form)
* @param options - The options to apply to the component
* @returns A Memoized component with the applied options
*
* @example
* ```ts
* const MemoizedComponent = memoized(MyComponent)
*
* // Uncurried
* const configured = withOptions(MemoizedComponent, {
* propsEquivalence: (a, b) => a.id === b.id
* })
*
* // Curried
* const configurer = withOptions({ propsEquivalence: (a, b) => a.id === b.id })
* const configured = configurer(MemoizedComponent)
* ```
*/
export const withOptions: {
<T extends Component.Component<any, any, any, any> & Memoized<any>>(
options: Partial<MemoizedOptions<Component.Component.Props<T>>>