This commit is contained in:
@@ -4,25 +4,80 @@ import * as React from "react"
|
|||||||
import * as Component from "./Component.js"
|
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 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 {}
|
export interface Async extends AsyncPrototype, AsyncOptions {}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The prototype object for Async components containing their methods and behaviors.
|
||||||
|
*/
|
||||||
export interface AsyncPrototype {
|
export interface AsyncPrototype {
|
||||||
|
/**
|
||||||
|
* The Async type ID brand.
|
||||||
|
*/
|
||||||
readonly [AsyncTypeId]: AsyncTypeId
|
readonly [AsyncTypeId]: AsyncTypeId
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration options for Async components.
|
||||||
|
*/
|
||||||
export interface AsyncOptions {
|
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
|
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">
|
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({
|
export const AsyncPrototype: AsyncPrototype = Object.freeze({
|
||||||
[AsyncTypeId]: AsyncTypeId,
|
[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>(
|
asFunctionComponent<P extends {}, A extends React.ReactNode, E, R>(
|
||||||
this: Component.Component<P, A, E, R> & Async,
|
this: Component.Component<P, A, E, R> & Async,
|
||||||
runtimeRef: React.RefObject<Runtime.Runtime<Exclude<R, Scope.Scope>>>,
|
runtimeRef: React.RefObject<Runtime.Runtime<Exclude<R, Scope.Scope>>>,
|
||||||
@@ -46,6 +101,16 @@ export const AsyncPrototype: AsyncPrototype = Object.freeze({
|
|||||||
},
|
},
|
||||||
} as const)
|
} 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> = (
|
export const defaultPropsEquivalence: Equivalence.Equivalence<AsyncProps> = (
|
||||||
self: Record<string, unknown>,
|
self: Record<string, unknown>,
|
||||||
that: 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)
|
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>>(
|
export const async = <T extends Component.Component<any, any, any, any>>(
|
||||||
self: T & (
|
self: T & (
|
||||||
"promise" extends keyof Component.Component.Props<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: {
|
export const withOptions: {
|
||||||
<T extends Component.Component<any, any, any, any> & Async>(
|
<T extends Component.Component<any, any, any, any> & Async>(
|
||||||
options: Partial<AsyncOptions>
|
options: Partial<AsyncOptions>
|
||||||
|
|||||||
@@ -4,23 +4,80 @@ import * as React from "react"
|
|||||||
import type * as Component from "./Component.js"
|
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")
|
export const MemoizedTypeId: unique symbol = Symbol.for("@effect-fc/Memoized/Memoized")
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The type of the Memoized type ID symbol.
|
||||||
|
*/
|
||||||
export type MemoizedTypeId = typeof MemoizedTypeId
|
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> {}
|
export interface Memoized<P> extends MemoizedPrototype, MemoizedOptions<P> {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The prototype object for Memoized components containing their methods and behaviors.
|
||||||
|
*/
|
||||||
export interface MemoizedPrototype {
|
export interface MemoizedPrototype {
|
||||||
|
/**
|
||||||
|
* The Memoized type ID brand.
|
||||||
|
*/
|
||||||
readonly [MemoizedTypeId]: MemoizedTypeId
|
readonly [MemoizedTypeId]: MemoizedTypeId
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration options for Memoized components.
|
||||||
|
*
|
||||||
|
* @template P The props type of the component
|
||||||
|
*/
|
||||||
export interface MemoizedOptions<P> {
|
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>
|
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({
|
export const MemoizedPrototype: MemoizedPrototype = Object.freeze({
|
||||||
[MemoizedTypeId]: MemoizedTypeId,
|
[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 {}>(
|
transformFunctionComponent<P extends {}>(
|
||||||
this: Memoized<P>,
|
this: Memoized<P>,
|
||||||
f: React.FC<P>,
|
f: React.FC<P>,
|
||||||
@@ -30,8 +87,39 @@ export const MemoizedPrototype: MemoizedPrototype = Object.freeze({
|
|||||||
} as const)
|
} 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)
|
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>>(
|
export const memoized = <T extends Component.Component<any, any, any, any>>(
|
||||||
self: T
|
self: T
|
||||||
): T & Memoized<Component.Component.Props<T>> => Object.setPrototypeOf(
|
): 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: {
|
export const withOptions: {
|
||||||
<T extends Component.Component<any, any, any, any> & Memoized<any>>(
|
<T extends Component.Component<any, any, any, any> & Memoized<any>>(
|
||||||
options: Partial<MemoizedOptions<Component.Component.Props<T>>>
|
options: Partial<MemoizedOptions<Component.Component.Props<T>>>
|
||||||
|
|||||||
Reference in New Issue
Block a user