react-chain-of-responsibility 0.3.1-main.cc0a3ac → 0.4.0-main.2a72139
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +13 -1
- package/dist/{chunk-RAUACXXD.mjs → chunk-U4UF7NZV.mjs} +24 -35
- package/dist/chunk-U4UF7NZV.mjs.map +1 -0
- package/dist/index-Dm_KqEiI.d.mts +66 -0
- package/dist/index-Dm_KqEiI.d.ts +66 -0
- package/dist/react-chain-of-responsibility.d.mts +3 -72
- package/dist/react-chain-of-responsibility.d.ts +3 -72
- package/dist/react-chain-of-responsibility.fluentUI.d.mts +6 -2
- package/dist/react-chain-of-responsibility.fluentUI.d.ts +6 -2
- package/dist/react-chain-of-responsibility.fluentUI.js +29 -36
- package/dist/react-chain-of-responsibility.fluentUI.js.map +1 -1
- package/dist/react-chain-of-responsibility.fluentUI.mjs +8 -4
- package/dist/react-chain-of-responsibility.fluentUI.mjs.map +1 -1
- package/dist/react-chain-of-responsibility.js +24 -35
- package/dist/react-chain-of-responsibility.js.map +1 -1
- package/dist/react-chain-of-responsibility.mjs +3 -3
- package/dist/react-chain-of-responsibility.preview.d.mts +80 -0
- package/dist/react-chain-of-responsibility.preview.d.ts +80 -0
- package/dist/react-chain-of-responsibility.preview.js +201 -0
- package/dist/react-chain-of-responsibility.preview.js.map +1 -0
- package/dist/react-chain-of-responsibility.preview.mjs +171 -0
- package/dist/react-chain-of-responsibility.preview.mjs.map +1 -0
- package/package.json +53 -16
- package/preview.js +1 -0
- package/dist/chunk-RAUACXXD.mjs.map +0 -1
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import { ComponentType, PropsWithChildren, ReactElement } from 'react';
|
|
2
|
+
|
|
3
|
+
declare global {
|
|
4
|
+
interface ArrayConstructor {
|
|
5
|
+
isArray(arg: any): arg is readonly any[];
|
|
6
|
+
}
|
|
7
|
+
}
|
|
8
|
+
type BaseProps = object;
|
|
9
|
+
type CreateChainOfResponsibilityOptions = {
|
|
10
|
+
/**
|
|
11
|
+
* Allows one component to pass different set of props to its downstream component. Default is false.
|
|
12
|
+
*
|
|
13
|
+
* It is recommended to keep this settings as default to prevent newly added component from unexpectedly changing behavior of downstream components.
|
|
14
|
+
*/
|
|
15
|
+
readonly allowOverrideProps?: boolean | undefined;
|
|
16
|
+
/**
|
|
17
|
+
* Allows a middleware to pass another request object when calling its next middleware. Default is false.
|
|
18
|
+
*
|
|
19
|
+
* It is recommended to keep this settings as default ot prevent newly added middleware from unexpectedly changing behavior of downstream middleware.
|
|
20
|
+
*
|
|
21
|
+
* To prevent upstream middleware from modifying the request, the request object should be set to be immutable through `Object.freeze`.
|
|
22
|
+
*/
|
|
23
|
+
readonly passModifiedRequest?: boolean | undefined;
|
|
24
|
+
};
|
|
25
|
+
type ChainOfResponsibility<Request, Props extends BaseProps, Init> = {
|
|
26
|
+
readonly Provider: ComponentType<ProviderProps<Request, Props, Init>> & InferenceHelper<Request, Props, Init>;
|
|
27
|
+
readonly Proxy: ComponentType<ProxyProps<Request, Props>>;
|
|
28
|
+
readonly reactComponent: ReactComponentHandlerResult<Props>;
|
|
29
|
+
readonly useBuildRenderCallback: () => UseBuildRenderCallback<Request, Props>;
|
|
30
|
+
};
|
|
31
|
+
declare const DO_NOT_CREATE_THIS_OBJECT_YOURSELF: unique symbol;
|
|
32
|
+
type ComponentRenderer<Props> = (props: Props) => ReactElement | null;
|
|
33
|
+
interface ComponentHandlerResult<Props extends BaseProps> {
|
|
34
|
+
readonly [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: undefined;
|
|
35
|
+
readonly render: (overridingProps?: Partial<Props> | undefined) => ReactElement | null;
|
|
36
|
+
}
|
|
37
|
+
type ComponentHandler<Request, Props extends BaseProps> = (request: Request) => ComponentHandlerResult<Props> | undefined;
|
|
38
|
+
type ComponentEnhancer<Request, Props extends BaseProps> = (next: ComponentHandler<Request, Props>) => ComponentHandler<Request, Props>;
|
|
39
|
+
type ComponentMiddleware<Request, Props extends BaseProps, Init = undefined> = (init: Init) => ComponentEnhancer<Request, Props>;
|
|
40
|
+
type ReactComponentHandlerResult<Props extends object> = <P extends Props>(component: ComponentType<P>, bindProps?: (Partial<Props> & Omit<P, keyof Props>) | ((props: Props) => Partial<Props> & Omit<P, keyof Props>) | undefined) => ComponentHandlerResult<Props>;
|
|
41
|
+
type UseBuildRenderCallbackOptions<Props> = {
|
|
42
|
+
readonly fallbackComponent?: ComponentType<Props> | undefined;
|
|
43
|
+
};
|
|
44
|
+
interface UseBuildRenderCallback<Request, Props extends BaseProps> {
|
|
45
|
+
(request: Request, options?: undefined | UseBuildRenderCallbackOptions<Props>): ComponentRenderer<Props> | undefined;
|
|
46
|
+
}
|
|
47
|
+
type ProviderProps<Request, Props extends BaseProps, Init> = PropsWithChildren<{
|
|
48
|
+
readonly middleware: readonly ComponentMiddleware<Request, Props, Init>[];
|
|
49
|
+
}> & (Init extends never | void ? {
|
|
50
|
+
readonly init?: undefined;
|
|
51
|
+
} : Init extends undefined | void ? {
|
|
52
|
+
readonly init?: Init;
|
|
53
|
+
} : {
|
|
54
|
+
readonly init: Init;
|
|
55
|
+
});
|
|
56
|
+
type ProxyProps<Request, Props extends BaseProps> = Props & {
|
|
57
|
+
readonly fallbackComponent?: ComponentType<Props> | undefined;
|
|
58
|
+
readonly request: Request;
|
|
59
|
+
};
|
|
60
|
+
type InferenceHelper<Request, Props extends BaseProps, Init> = {
|
|
61
|
+
readonly '~types': {
|
|
62
|
+
readonly init: Init;
|
|
63
|
+
readonly middleware: ComponentMiddleware<Request, Props, Init>;
|
|
64
|
+
readonly props: Props;
|
|
65
|
+
readonly proxyProps: ProxyProps<Request, Props>;
|
|
66
|
+
readonly providerProps: ProviderProps<Request, Props, Init>;
|
|
67
|
+
readonly request: Request;
|
|
68
|
+
};
|
|
69
|
+
};
|
|
70
|
+
type InferInit<T extends InferenceHelper<any, any, any>> = T['~types']['init'];
|
|
71
|
+
type InferMiddleware<T extends InferenceHelper<any, any, any>> = T['~types']['middleware'];
|
|
72
|
+
type InferProps<T extends InferenceHelper<any, any, any>> = T['~types']['props'];
|
|
73
|
+
type InferProxyProps<T extends InferenceHelper<any, any, any>> = T['~types']['proxyProps'];
|
|
74
|
+
type InferProviderProps<T extends InferenceHelper<any, any, any>> = T['~types']['providerProps'];
|
|
75
|
+
type InferRequest<T extends InferenceHelper<any, any, any>> = T['~types']['request'];
|
|
76
|
+
declare function createChainOfResponsibility<Request = void, Props extends BaseProps = {
|
|
77
|
+
readonly children?: never;
|
|
78
|
+
}, Init = void>(options?: CreateChainOfResponsibilityOptions): ChainOfResponsibility<Request, Props, Init>;
|
|
79
|
+
|
|
80
|
+
export { type ChainOfResponsibility, type ComponentEnhancer, type ComponentHandler, type ComponentHandlerResult, type ComponentMiddleware, type ComponentRenderer, type CreateChainOfResponsibilityOptions, type InferInit, type InferMiddleware, type InferProps, type InferProviderProps, type InferProxyProps, type InferRequest, type InferenceHelper, type ProviderProps, type ProxyProps, type ReactComponentHandlerResult, type UseBuildRenderCallback, type UseBuildRenderCallbackOptions, createChainOfResponsibility };
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __export = (target, all) => {
|
|
9
|
+
for (var name in all)
|
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
+
};
|
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (let key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
+
}
|
|
18
|
+
return to;
|
|
19
|
+
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
+
mod
|
|
27
|
+
));
|
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
+
|
|
30
|
+
// src/index.preview.ts
|
|
31
|
+
var index_preview_exports = {};
|
|
32
|
+
__export(index_preview_exports, {
|
|
33
|
+
createChainOfResponsibility: () => createChainOfResponsibilityAsRenderCallback_default
|
|
34
|
+
});
|
|
35
|
+
module.exports = __toCommonJS(index_preview_exports);
|
|
36
|
+
|
|
37
|
+
// src/preview/createChainOfResponsibilityAsRenderCallback.tsx
|
|
38
|
+
var import_handler_chain = require("handler-chain");
|
|
39
|
+
var import_react = __toESM(require("react"));
|
|
40
|
+
var import_valibot = require("valibot");
|
|
41
|
+
|
|
42
|
+
// src/preview/private/arePropsEqual.ts
|
|
43
|
+
function arePropsEqual(x, y) {
|
|
44
|
+
if (Object.is(x, y)) {
|
|
45
|
+
return true;
|
|
46
|
+
}
|
|
47
|
+
const mapOfX = new Map(Object.entries(x));
|
|
48
|
+
const mapOfY = new Map(Object.entries(y));
|
|
49
|
+
if (mapOfX.size !== mapOfY.size) {
|
|
50
|
+
return false;
|
|
51
|
+
}
|
|
52
|
+
const keys = /* @__PURE__ */ new Set([...mapOfX.keys(), ...mapOfY.keys()]);
|
|
53
|
+
for (const key of keys) {
|
|
54
|
+
if (!Object.is(mapOfX.get(key), mapOfY.get(key))) {
|
|
55
|
+
return false;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
return true;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// src/preview/createChainOfResponsibilityAsRenderCallback.tsx
|
|
62
|
+
var DO_NOT_CREATE_THIS_OBJECT_YOURSELF = Symbol();
|
|
63
|
+
var componentHandlerResultSchema = (0, import_valibot.custom)(
|
|
64
|
+
(value) => (0, import_valibot.safeParse)((0, import_valibot.object)({ render: (0, import_valibot.function_)() }), value).success && !!value && typeof value === "object" && DO_NOT_CREATE_THIS_OBJECT_YOURSELF in value,
|
|
65
|
+
"react-chain-of-responsibility: middleware must return value constructed by reactComponent()"
|
|
66
|
+
);
|
|
67
|
+
function createComponentHandlerResult(render) {
|
|
68
|
+
return Object.freeze({ [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: void 0, render });
|
|
69
|
+
}
|
|
70
|
+
function createChainOfResponsibility(options = {}) {
|
|
71
|
+
options = Object.freeze({ ...options });
|
|
72
|
+
const BuildContext = (0, import_react.createContext)(
|
|
73
|
+
Object.freeze({ enhancer: (next) => (request) => next(request) })
|
|
74
|
+
);
|
|
75
|
+
const RenderContext = (0, import_react.createContext)(
|
|
76
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
77
|
+
new Proxy({}, {
|
|
78
|
+
get() {
|
|
79
|
+
throw new Error(
|
|
80
|
+
"react-chain-of-responsibility: this hook cannot be used outside of <Proxy> and useBuildRenderCallback()"
|
|
81
|
+
);
|
|
82
|
+
}
|
|
83
|
+
})
|
|
84
|
+
);
|
|
85
|
+
function reactComponent(component, bindProps) {
|
|
86
|
+
return createComponentHandlerResult((overridingProps) => /* @__PURE__ */ import_react.default.createElement(
|
|
87
|
+
ComponentWithProps,
|
|
88
|
+
{
|
|
89
|
+
bindProps,
|
|
90
|
+
component,
|
|
91
|
+
overridingProps
|
|
92
|
+
}
|
|
93
|
+
));
|
|
94
|
+
}
|
|
95
|
+
const ComponentWithProps = (0, import_react.memo)(function ComponentWithProps2({
|
|
96
|
+
bindProps,
|
|
97
|
+
component: Component,
|
|
98
|
+
overridingProps
|
|
99
|
+
}) {
|
|
100
|
+
const { allowOverrideProps } = options;
|
|
101
|
+
const { originalProps: renderCallbackProps } = (0, import_react.useContext)(RenderContext);
|
|
102
|
+
if (overridingProps && !arePropsEqual(overridingProps, renderCallbackProps) && !allowOverrideProps) {
|
|
103
|
+
console.warn('react-chain-of-responsibility: "allowOverrideProps" must be set to true to override props');
|
|
104
|
+
}
|
|
105
|
+
const props = Object.freeze(
|
|
106
|
+
allowOverrideProps ? { ...renderCallbackProps, ...overridingProps } : { ...renderCallbackProps }
|
|
107
|
+
);
|
|
108
|
+
return /* @__PURE__ */ import_react.default.createElement(Component, { ...props, ...typeof bindProps === "function" ? bindProps(props) : bindProps });
|
|
109
|
+
});
|
|
110
|
+
const useBuildRenderCallback = () => {
|
|
111
|
+
const { enhancer } = (0, import_react.useContext)(BuildContext);
|
|
112
|
+
return (0, import_react.useCallback)(
|
|
113
|
+
(request, buildOptions = {}) => {
|
|
114
|
+
const result = (
|
|
115
|
+
// Put the "fallbackComponent" as the last one in the chain.
|
|
116
|
+
enhancer(() => {
|
|
117
|
+
const { fallbackComponent } = buildOptions;
|
|
118
|
+
if (!fallbackComponent) {
|
|
119
|
+
console.warn(
|
|
120
|
+
'react-chain-of-responsibility: the request has fall through all middleware, set "fallbackComponent" as a catchall',
|
|
121
|
+
request
|
|
122
|
+
);
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
return reactComponent(fallbackComponent);
|
|
126
|
+
})(request)
|
|
127
|
+
);
|
|
128
|
+
return result && ((originalProps) => (
|
|
129
|
+
// This is render function, we cannot call any hooks here.
|
|
130
|
+
/* @__PURE__ */ import_react.default.createElement(BuildRenderCallback, { originalProps, render: result.render })
|
|
131
|
+
));
|
|
132
|
+
},
|
|
133
|
+
[enhancer]
|
|
134
|
+
);
|
|
135
|
+
};
|
|
136
|
+
function BuildRenderCallback({ originalProps, render }) {
|
|
137
|
+
const context = (0, import_react.useMemo)(() => Object.freeze({ originalProps }), [originalProps]);
|
|
138
|
+
return /* @__PURE__ */ import_react.default.createElement(RenderContext.Provider, { value: context }, render());
|
|
139
|
+
}
|
|
140
|
+
function ChainOfResponsibilityProvider({ children, init, middleware }) {
|
|
141
|
+
if (!Array.isArray(middleware) || middleware.some((middleware2) => typeof middleware2 !== "function")) {
|
|
142
|
+
throw new Error('react-chain-of-responsibility: "middleware" prop must be an array of functions');
|
|
143
|
+
}
|
|
144
|
+
const fortifiedMiddleware = (0, import_react.useMemo)(
|
|
145
|
+
() => Object.freeze(
|
|
146
|
+
middleware.map((fn) => (init2) => {
|
|
147
|
+
const enhancer2 = fn(init2);
|
|
148
|
+
return (next) => (originalRequest) => {
|
|
149
|
+
let hasReturned;
|
|
150
|
+
const returnValue = enhancer2((nextRequest) => {
|
|
151
|
+
if (hasReturned) {
|
|
152
|
+
throw new Error(
|
|
153
|
+
"react-chain-of-responsibility: next() cannot be called after the function had returned synchronously"
|
|
154
|
+
);
|
|
155
|
+
}
|
|
156
|
+
!options.passModifiedRequest && !Object.is(nextRequest, originalRequest) && console.warn(
|
|
157
|
+
'react-chain-of-responsibility: next() must be called with the original request, otherwise, set "options.passModifiedRequest" to true to pass a different request object downstream'
|
|
158
|
+
);
|
|
159
|
+
return next(options.passModifiedRequest ? nextRequest : originalRequest);
|
|
160
|
+
})(originalRequest);
|
|
161
|
+
hasReturned = true;
|
|
162
|
+
return returnValue && (0, import_valibot.parse)(componentHandlerResultSchema, returnValue);
|
|
163
|
+
};
|
|
164
|
+
})
|
|
165
|
+
),
|
|
166
|
+
[middleware]
|
|
167
|
+
);
|
|
168
|
+
const { enhancer: parentEnhancer } = (0, import_react.useContext)(BuildContext);
|
|
169
|
+
const enhancer = (0, import_react.useMemo)(
|
|
170
|
+
() => (
|
|
171
|
+
// We are reversing because it is easier to read:
|
|
172
|
+
// - With reverse, [a, b, c] will become a(b(c(fn)))
|
|
173
|
+
// - Without reverse, [a, b, c] will become c(b(a(fn)))
|
|
174
|
+
(0, import_handler_chain.applyMiddleware)(
|
|
175
|
+
...[...fortifiedMiddleware, ...[() => parentEnhancer]]
|
|
176
|
+
)(init)
|
|
177
|
+
),
|
|
178
|
+
[init, fortifiedMiddleware, parentEnhancer]
|
|
179
|
+
);
|
|
180
|
+
const contextValue = (0, import_react.useMemo)(() => Object.freeze({ enhancer }), [enhancer]);
|
|
181
|
+
return /* @__PURE__ */ import_react.default.createElement(BuildContext.Provider, { value: contextValue }, children);
|
|
182
|
+
}
|
|
183
|
+
function ChainOfResponsibilityProxy({ fallbackComponent, request, ...props }) {
|
|
184
|
+
const result = useBuildRenderCallback()(request, { fallbackComponent })?.(props);
|
|
185
|
+
return result ? /* @__PURE__ */ import_react.default.createElement(import_react.Fragment, null, result) : null;
|
|
186
|
+
}
|
|
187
|
+
const MemoizedChainOfResponsibilityProvider = (0, import_react.memo)(ChainOfResponsibilityProvider);
|
|
188
|
+
return Object.freeze({
|
|
189
|
+
Provider: MemoizedChainOfResponsibilityProvider,
|
|
190
|
+
Proxy: ChainOfResponsibilityProxy,
|
|
191
|
+
reactComponent,
|
|
192
|
+
useBuildRenderCallback
|
|
193
|
+
// TODO: Consider adding back `asMiddleware`.
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
var createChainOfResponsibilityAsRenderCallback_default = createChainOfResponsibility;
|
|
197
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
198
|
+
0 && (module.exports = {
|
|
199
|
+
createChainOfResponsibility
|
|
200
|
+
});
|
|
201
|
+
//# sourceMappingURL=react-chain-of-responsibility.preview.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/index.preview.ts","../src/preview/createChainOfResponsibilityAsRenderCallback.tsx","../src/preview/private/arePropsEqual.ts"],"sourcesContent":["export {\n default as createChainOfResponsibility,\n type ChainOfResponsibility,\n type ComponentEnhancer,\n type ComponentHandler,\n type ComponentHandlerResult,\n type ComponentMiddleware,\n type ComponentRenderer,\n type CreateChainOfResponsibilityOptions,\n type InferenceHelper,\n type InferInit,\n type InferMiddleware,\n type InferProps,\n type InferProviderProps,\n type InferProxyProps,\n type InferRequest,\n type ProviderProps,\n type ProxyProps,\n type ReactComponentHandlerResult,\n type UseBuildRenderCallback,\n type UseBuildRenderCallbackOptions\n} from './preview/createChainOfResponsibilityAsRenderCallback.tsx';\n","import { applyMiddleware } from 'handler-chain';\nimport React, {\n createContext,\n Fragment,\n memo,\n useCallback,\n useContext,\n useMemo,\n type ComponentType,\n type PropsWithChildren,\n type ReactElement,\n type ReactNode\n} from 'react';\nimport { custom, function_, object, parse, safeParse } from 'valibot';\n\nimport arePropsEqual from './private/arePropsEqual.ts';\n\n// TODO: Related to https://github.com/microsoft/TypeScript/issues/17002.\n// typescript@5.2.2 has a bug, Array.isArray() is a type predicate but only works with mutable array, not readonly array.\ndeclare global {\n interface ArrayConstructor {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n isArray(arg: any): arg is readonly any[];\n }\n}\n\ntype BaseProps = object;\n\ntype CreateChainOfResponsibilityOptions = {\n /**\n * Allows one component to pass different set of props to its downstream component. Default is false.\n *\n * It is recommended to keep this settings as default to prevent newly added component from unexpectedly changing behavior of downstream components.\n */\n readonly allowOverrideProps?: boolean | undefined;\n\n /**\n * Allows a middleware to pass another request object when calling its next middleware. Default is false.\n *\n * It is recommended to keep this settings as default ot prevent newly added middleware from unexpectedly changing behavior of downstream middleware.\n *\n * To prevent upstream middleware from modifying the request, the request object should be set to be immutable through `Object.freeze`.\n */\n readonly passModifiedRequest?: boolean | undefined;\n};\n\ntype ChainOfResponsibility<Request, Props extends BaseProps, Init> = {\n readonly Provider: ComponentType<ProviderProps<Request, Props, Init>> & InferenceHelper<Request, Props, Init>;\n readonly Proxy: ComponentType<ProxyProps<Request, Props>>;\n readonly reactComponent: ReactComponentHandlerResult<Props>;\n readonly useBuildRenderCallback: () => UseBuildRenderCallback<Request, Props>;\n};\n\n// TODO: Maybe this one should be local.\n// Verify that reactComponent() from an instance of CoR should throw error when used in another instance of CoR.\nconst DO_NOT_CREATE_THIS_OBJECT_YOURSELF = Symbol();\n\ntype ComponentRenderer<Props> = (props: Props) => ReactElement | null;\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst componentHandlerResultSchema = custom<ComponentHandlerResult<any>>(\n value =>\n safeParse(object({ render: function_() }), value).success &&\n !!value &&\n typeof value === 'object' &&\n DO_NOT_CREATE_THIS_OBJECT_YOURSELF in value,\n 'react-chain-of-responsibility: middleware must return value constructed by reactComponent()'\n);\n\ninterface ComponentHandlerResult<Props extends BaseProps> {\n readonly [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: undefined;\n readonly render: (overridingProps?: Partial<Props> | undefined) => ReactElement | null;\n}\n\ntype ComponentHandler<Request, Props extends BaseProps> = (\n request: Request\n) => ComponentHandlerResult<Props> | undefined;\n\ntype ComponentEnhancer<Request, Props extends BaseProps> = (\n next: ComponentHandler<Request, Props>\n) => ComponentHandler<Request, Props>;\n\ntype ComponentMiddleware<Request, Props extends BaseProps, Init = undefined> = (\n init: Init\n) => ComponentEnhancer<Request, Props>;\n\ntype ReactComponentHandlerResult<Props extends object> = <P extends Props>(\n component: ComponentType<P>,\n bindProps?:\n | (Partial<Props> & Omit<P, keyof Props>)\n | ((props: Props) => Partial<Props> & Omit<P, keyof Props>)\n | undefined\n) => ComponentHandlerResult<Props>;\n\ntype UseBuildRenderCallbackOptions<Props> = {\n readonly fallbackComponent?: ComponentType<Props> | undefined;\n};\n\ninterface UseBuildRenderCallback<Request, Props extends BaseProps> {\n (request: Request, options?: undefined | UseBuildRenderCallbackOptions<Props>): ComponentRenderer<Props> | undefined;\n}\n\ntype BuildContextType<Request, Props extends BaseProps> = {\n readonly enhancer: ComponentEnhancer<Request, Props>;\n};\n\ntype RenderContextType<Props> = {\n readonly originalProps: Props;\n};\n\ntype ProviderProps<Request, Props extends BaseProps, Init> = PropsWithChildren<{\n readonly middleware: readonly ComponentMiddleware<Request, Props, Init>[];\n}> &\n (Init extends never | void\n ? { readonly init?: undefined }\n : Init extends undefined | void\n ? { readonly init?: Init }\n : { readonly init: Init });\n\ntype ProxyProps<Request, Props extends BaseProps> = Props & {\n readonly fallbackComponent?: ComponentType<Props> | undefined;\n readonly request: Request;\n};\n\ntype InferenceHelper<Request, Props extends BaseProps, Init> = {\n readonly '~types': {\n readonly init: Init;\n readonly middleware: ComponentMiddleware<Request, Props, Init>;\n readonly props: Props;\n readonly proxyProps: ProxyProps<Request, Props>;\n readonly providerProps: ProviderProps<Request, Props, Init>;\n readonly request: Request;\n };\n};\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferInit<T extends InferenceHelper<any, any, any>> = T['~types']['init'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferMiddleware<T extends InferenceHelper<any, any, any>> = T['~types']['middleware'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProps<T extends InferenceHelper<any, any, any>> = T['~types']['props'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProxyProps<T extends InferenceHelper<any, any, any>> = T['~types']['proxyProps'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProviderProps<T extends InferenceHelper<any, any, any>> = T['~types']['providerProps'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferRequest<T extends InferenceHelper<any, any, any>> = T['~types']['request'];\n\nfunction createComponentHandlerResult<Props extends BaseProps>(\n render: (overridingProps?: Partial<Props> | undefined) => ReactElement | null\n): ComponentHandlerResult<Props> {\n return Object.freeze({ [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: undefined, render });\n}\n\nfunction createChainOfResponsibility<\n Request = void,\n Props extends BaseProps = { readonly children?: never },\n Init = void\n>(options: CreateChainOfResponsibilityOptions = {}): ChainOfResponsibility<Request, Props, Init> {\n // Freeze options to prevent accidental change.\n options = Object.freeze({ ...options });\n\n const BuildContext = createContext<BuildContextType<Request, Props>>(\n Object.freeze({ enhancer: next => request => next(request) })\n );\n\n const RenderContext = createContext<RenderContextType<Props>>(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n new Proxy({} as any, {\n get() {\n // The following is assertion, there is no way to hit this line.\n /* istanbul ignore next */\n throw new Error(\n 'react-chain-of-responsibility: this hook cannot be used outside of <Proxy> and useBuildRenderCallback()'\n );\n }\n })\n );\n\n function reactComponent<P extends Props>(\n component: ComponentType<P>,\n // For `bindProps` of type function, do not do side-effect in it, it may not be always called in all scenarios.\n bindProps?:\n | (Partial<Props> & Omit<P, keyof Props>)\n | ((props: Props) => Partial<Props> & Omit<P, keyof Props>)\n | undefined\n ): ComponentHandlerResult<Props> {\n return createComponentHandlerResult((overridingProps?: Partial<Props> | undefined) => (\n <ComponentWithProps\n bindProps={bindProps}\n component={component as ComponentType<Props>}\n overridingProps={overridingProps}\n />\n ));\n }\n\n const ComponentWithProps = memo(function ComponentWithProps({\n bindProps,\n component: Component,\n overridingProps\n }: {\n readonly bindProps?: Partial<Props> | ((props: Props) => Partial<Props>) | undefined;\n readonly component: ComponentType<Props>;\n readonly overridingProps?: Partial<Props> | undefined;\n }) {\n const { allowOverrideProps } = options;\n const { originalProps: renderCallbackProps } = useContext(RenderContext);\n\n if (overridingProps && !arePropsEqual(overridingProps, renderCallbackProps) && !allowOverrideProps) {\n console.warn('react-chain-of-responsibility: \"allowOverrideProps\" must be set to true to override props');\n }\n\n const props = Object.freeze(\n allowOverrideProps ? { ...renderCallbackProps, ...overridingProps } : { ...renderCallbackProps }\n );\n\n return <Component {...props} {...(typeof bindProps === 'function' ? bindProps(props) : bindProps)} />;\n });\n\n const useBuildRenderCallback: () => UseBuildRenderCallback<Request, Props> = () => {\n const { enhancer } = useContext(BuildContext);\n\n return useCallback(\n (request, buildOptions = {}) => {\n const result =\n // Put the \"fallbackComponent\" as the last one in the chain.\n enhancer(() => {\n const { fallbackComponent } = buildOptions;\n\n if (!fallbackComponent) {\n console.warn(\n 'react-chain-of-responsibility: the request has fall through all middleware, set \"fallbackComponent\" as a catchall',\n request\n );\n\n // For clarity, we are returning `undefined` instead of `() => undefined`.\n return;\n }\n\n // `fallbackComponent` do not need `overridingProps` because it is the last one in the chain, it would not have the next() function.\n return reactComponent(fallbackComponent);\n })(request);\n\n return (\n result &&\n ((originalProps: Props) => (\n // This is render function, we cannot call any hooks here.\n <BuildRenderCallback originalProps={originalProps} render={result.render} />\n ))\n );\n },\n [enhancer]\n );\n };\n\n type BuildRenderCallbackProps = {\n readonly originalProps: Props;\n readonly render: () => ReactNode;\n };\n\n // Do not memoize <BuildRenderCallback>.\n // `bindProps` may have side effect and we want to be re-rendered to capture the side-effect.\n // To prevent wasted render, web devs should memoize it themselves.\n function BuildRenderCallback({ originalProps, render }: BuildRenderCallbackProps) {\n const context = useMemo<RenderContextType<Props>>(() => Object.freeze({ originalProps }), [originalProps]);\n\n return <RenderContext.Provider value={context}>{render()}</RenderContext.Provider>;\n }\n\n function ChainOfResponsibilityProvider({ children, init, middleware }: ProviderProps<Request, Props, Init>) {\n if (!Array.isArray(middleware) || middleware.some(middleware => typeof middleware !== 'function')) {\n throw new Error('react-chain-of-responsibility: \"middleware\" prop must be an array of functions');\n }\n\n // Remap the middleware, so all inputs/outputs are validated.\n const fortifiedMiddleware = useMemo(\n () =>\n Object.freeze(\n middleware.map<ComponentMiddleware<Request, Props, Init>>(fn => (init: Init) => {\n const enhancer = fn(init);\n\n return next => originalRequest => {\n // False positive: although we did not re-assign the variable from true, it was initialized as undefined.\n // eslint-disable-next-line prefer-const\n let hasReturned: boolean;\n\n const returnValue = enhancer(nextRequest => {\n if (hasReturned) {\n throw new Error(\n 'react-chain-of-responsibility: next() cannot be called after the function had returned synchronously'\n );\n }\n\n // We do not allow passing void/undefined to next() because it would be confusing whether to keep the original request or pass an undefined.\n !options.passModifiedRequest &&\n !Object.is(nextRequest, originalRequest) &&\n console.warn(\n 'react-chain-of-responsibility: next() must be called with the original request, otherwise, set \"options.passModifiedRequest\" to true to pass a different request object downstream'\n );\n\n return next(options.passModifiedRequest ? nextRequest : originalRequest);\n })(originalRequest);\n\n hasReturned = true;\n\n // Make sure the return value is built using our helper function for forward-compatibility reason.\n return returnValue && parse(componentHandlerResultSchema, returnValue);\n };\n })\n ),\n [middleware]\n );\n\n const { enhancer: parentEnhancer } = useContext(BuildContext);\n\n const enhancer = useMemo<ComponentEnhancer<Request, Props>>(\n () =>\n // We are reversing because it is easier to read:\n // - With reverse, [a, b, c] will become a(b(c(fn)))\n // - Without reverse, [a, b, c] will become c(b(a(fn)))\n applyMiddleware<ComponentHandlerResult<Props> | undefined, Request, Init>(\n ...[...fortifiedMiddleware, ...[() => parentEnhancer]]\n )(init as Init),\n [init, fortifiedMiddleware, parentEnhancer]\n );\n\n const contextValue = useMemo<BuildContextType<Request, Props>>(() => Object.freeze({ enhancer }), [enhancer]);\n\n return <BuildContext.Provider value={contextValue}>{children}</BuildContext.Provider>;\n }\n\n function ChainOfResponsibilityProxy({ fallbackComponent, request, ...props }: ProxyProps<Request, Props>) {\n const result = useBuildRenderCallback()(request, { fallbackComponent })?.(props as Props);\n\n return result ? <Fragment>{result}</Fragment> : null;\n }\n\n const MemoizedChainOfResponsibilityProvider =\n memo<ProviderProps<Request, Props, Init>>(ChainOfResponsibilityProvider);\n\n return Object.freeze({\n Provider: MemoizedChainOfResponsibilityProvider as typeof MemoizedChainOfResponsibilityProvider &\n InferenceHelper<Request, Props, Init>,\n Proxy: ChainOfResponsibilityProxy,\n reactComponent,\n useBuildRenderCallback\n\n // TODO: Consider adding back `asMiddleware`.\n });\n}\n\nexport default createChainOfResponsibility;\nexport {\n type ChainOfResponsibility,\n type ComponentEnhancer,\n type ComponentHandler,\n type ComponentHandlerResult,\n type ComponentMiddleware,\n type ComponentRenderer,\n type CreateChainOfResponsibilityOptions,\n type InferenceHelper,\n type InferInit,\n type InferMiddleware,\n type InferProps,\n type InferProviderProps,\n type InferProxyProps,\n type InferRequest,\n type ProviderProps,\n type ProxyProps,\n type ReactComponentHandlerResult,\n type UseBuildRenderCallback,\n type UseBuildRenderCallbackOptions\n};\n","// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport default function arePropsEqual<T extends Record<string, any>>(x: T, y: T): boolean {\n if (Object.is(x, y)) {\n return true;\n }\n\n const mapOfX = new Map(Object.entries(x));\n const mapOfY = new Map(Object.entries(y));\n\n if (mapOfX.size !== mapOfY.size) {\n return false;\n }\n\n const keys = new Set([...mapOfX.keys(), ...mapOfY.keys()]);\n\n for (const key of keys) {\n if (!Object.is(mapOfX.get(key), mapOfY.get(key))) {\n return false;\n }\n }\n\n return true;\n}\n"],"mappings":"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","names":["React","ComponentWithProps","middleware","init","enhancer"]}
|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
// src/preview/createChainOfResponsibilityAsRenderCallback.tsx
|
|
2
|
+
import { applyMiddleware } from "handler-chain";
|
|
3
|
+
import React, {
|
|
4
|
+
createContext,
|
|
5
|
+
Fragment,
|
|
6
|
+
memo,
|
|
7
|
+
useCallback,
|
|
8
|
+
useContext,
|
|
9
|
+
useMemo
|
|
10
|
+
} from "react";
|
|
11
|
+
import { custom, function_, object, parse, safeParse } from "valibot";
|
|
12
|
+
|
|
13
|
+
// src/preview/private/arePropsEqual.ts
|
|
14
|
+
function arePropsEqual(x, y) {
|
|
15
|
+
if (Object.is(x, y)) {
|
|
16
|
+
return true;
|
|
17
|
+
}
|
|
18
|
+
const mapOfX = new Map(Object.entries(x));
|
|
19
|
+
const mapOfY = new Map(Object.entries(y));
|
|
20
|
+
if (mapOfX.size !== mapOfY.size) {
|
|
21
|
+
return false;
|
|
22
|
+
}
|
|
23
|
+
const keys = /* @__PURE__ */ new Set([...mapOfX.keys(), ...mapOfY.keys()]);
|
|
24
|
+
for (const key of keys) {
|
|
25
|
+
if (!Object.is(mapOfX.get(key), mapOfY.get(key))) {
|
|
26
|
+
return false;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
return true;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// src/preview/createChainOfResponsibilityAsRenderCallback.tsx
|
|
33
|
+
var DO_NOT_CREATE_THIS_OBJECT_YOURSELF = Symbol();
|
|
34
|
+
var componentHandlerResultSchema = custom(
|
|
35
|
+
(value) => safeParse(object({ render: function_() }), value).success && !!value && typeof value === "object" && DO_NOT_CREATE_THIS_OBJECT_YOURSELF in value,
|
|
36
|
+
"react-chain-of-responsibility: middleware must return value constructed by reactComponent()"
|
|
37
|
+
);
|
|
38
|
+
function createComponentHandlerResult(render) {
|
|
39
|
+
return Object.freeze({ [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: void 0, render });
|
|
40
|
+
}
|
|
41
|
+
function createChainOfResponsibility(options = {}) {
|
|
42
|
+
options = Object.freeze({ ...options });
|
|
43
|
+
const BuildContext = createContext(
|
|
44
|
+
Object.freeze({ enhancer: (next) => (request) => next(request) })
|
|
45
|
+
);
|
|
46
|
+
const RenderContext = createContext(
|
|
47
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
48
|
+
new Proxy({}, {
|
|
49
|
+
get() {
|
|
50
|
+
throw new Error(
|
|
51
|
+
"react-chain-of-responsibility: this hook cannot be used outside of <Proxy> and useBuildRenderCallback()"
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
})
|
|
55
|
+
);
|
|
56
|
+
function reactComponent(component, bindProps) {
|
|
57
|
+
return createComponentHandlerResult((overridingProps) => /* @__PURE__ */ React.createElement(
|
|
58
|
+
ComponentWithProps,
|
|
59
|
+
{
|
|
60
|
+
bindProps,
|
|
61
|
+
component,
|
|
62
|
+
overridingProps
|
|
63
|
+
}
|
|
64
|
+
));
|
|
65
|
+
}
|
|
66
|
+
const ComponentWithProps = memo(function ComponentWithProps2({
|
|
67
|
+
bindProps,
|
|
68
|
+
component: Component,
|
|
69
|
+
overridingProps
|
|
70
|
+
}) {
|
|
71
|
+
const { allowOverrideProps } = options;
|
|
72
|
+
const { originalProps: renderCallbackProps } = useContext(RenderContext);
|
|
73
|
+
if (overridingProps && !arePropsEqual(overridingProps, renderCallbackProps) && !allowOverrideProps) {
|
|
74
|
+
console.warn('react-chain-of-responsibility: "allowOverrideProps" must be set to true to override props');
|
|
75
|
+
}
|
|
76
|
+
const props = Object.freeze(
|
|
77
|
+
allowOverrideProps ? { ...renderCallbackProps, ...overridingProps } : { ...renderCallbackProps }
|
|
78
|
+
);
|
|
79
|
+
return /* @__PURE__ */ React.createElement(Component, { ...props, ...typeof bindProps === "function" ? bindProps(props) : bindProps });
|
|
80
|
+
});
|
|
81
|
+
const useBuildRenderCallback = () => {
|
|
82
|
+
const { enhancer } = useContext(BuildContext);
|
|
83
|
+
return useCallback(
|
|
84
|
+
(request, buildOptions = {}) => {
|
|
85
|
+
const result = (
|
|
86
|
+
// Put the "fallbackComponent" as the last one in the chain.
|
|
87
|
+
enhancer(() => {
|
|
88
|
+
const { fallbackComponent } = buildOptions;
|
|
89
|
+
if (!fallbackComponent) {
|
|
90
|
+
console.warn(
|
|
91
|
+
'react-chain-of-responsibility: the request has fall through all middleware, set "fallbackComponent" as a catchall',
|
|
92
|
+
request
|
|
93
|
+
);
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
return reactComponent(fallbackComponent);
|
|
97
|
+
})(request)
|
|
98
|
+
);
|
|
99
|
+
return result && ((originalProps) => (
|
|
100
|
+
// This is render function, we cannot call any hooks here.
|
|
101
|
+
/* @__PURE__ */ React.createElement(BuildRenderCallback, { originalProps, render: result.render })
|
|
102
|
+
));
|
|
103
|
+
},
|
|
104
|
+
[enhancer]
|
|
105
|
+
);
|
|
106
|
+
};
|
|
107
|
+
function BuildRenderCallback({ originalProps, render }) {
|
|
108
|
+
const context = useMemo(() => Object.freeze({ originalProps }), [originalProps]);
|
|
109
|
+
return /* @__PURE__ */ React.createElement(RenderContext.Provider, { value: context }, render());
|
|
110
|
+
}
|
|
111
|
+
function ChainOfResponsibilityProvider({ children, init, middleware }) {
|
|
112
|
+
if (!Array.isArray(middleware) || middleware.some((middleware2) => typeof middleware2 !== "function")) {
|
|
113
|
+
throw new Error('react-chain-of-responsibility: "middleware" prop must be an array of functions');
|
|
114
|
+
}
|
|
115
|
+
const fortifiedMiddleware = useMemo(
|
|
116
|
+
() => Object.freeze(
|
|
117
|
+
middleware.map((fn) => (init2) => {
|
|
118
|
+
const enhancer2 = fn(init2);
|
|
119
|
+
return (next) => (originalRequest) => {
|
|
120
|
+
let hasReturned;
|
|
121
|
+
const returnValue = enhancer2((nextRequest) => {
|
|
122
|
+
if (hasReturned) {
|
|
123
|
+
throw new Error(
|
|
124
|
+
"react-chain-of-responsibility: next() cannot be called after the function had returned synchronously"
|
|
125
|
+
);
|
|
126
|
+
}
|
|
127
|
+
!options.passModifiedRequest && !Object.is(nextRequest, originalRequest) && console.warn(
|
|
128
|
+
'react-chain-of-responsibility: next() must be called with the original request, otherwise, set "options.passModifiedRequest" to true to pass a different request object downstream'
|
|
129
|
+
);
|
|
130
|
+
return next(options.passModifiedRequest ? nextRequest : originalRequest);
|
|
131
|
+
})(originalRequest);
|
|
132
|
+
hasReturned = true;
|
|
133
|
+
return returnValue && parse(componentHandlerResultSchema, returnValue);
|
|
134
|
+
};
|
|
135
|
+
})
|
|
136
|
+
),
|
|
137
|
+
[middleware]
|
|
138
|
+
);
|
|
139
|
+
const { enhancer: parentEnhancer } = useContext(BuildContext);
|
|
140
|
+
const enhancer = useMemo(
|
|
141
|
+
() => (
|
|
142
|
+
// We are reversing because it is easier to read:
|
|
143
|
+
// - With reverse, [a, b, c] will become a(b(c(fn)))
|
|
144
|
+
// - Without reverse, [a, b, c] will become c(b(a(fn)))
|
|
145
|
+
applyMiddleware(
|
|
146
|
+
...[...fortifiedMiddleware, ...[() => parentEnhancer]]
|
|
147
|
+
)(init)
|
|
148
|
+
),
|
|
149
|
+
[init, fortifiedMiddleware, parentEnhancer]
|
|
150
|
+
);
|
|
151
|
+
const contextValue = useMemo(() => Object.freeze({ enhancer }), [enhancer]);
|
|
152
|
+
return /* @__PURE__ */ React.createElement(BuildContext.Provider, { value: contextValue }, children);
|
|
153
|
+
}
|
|
154
|
+
function ChainOfResponsibilityProxy({ fallbackComponent, request, ...props }) {
|
|
155
|
+
const result = useBuildRenderCallback()(request, { fallbackComponent })?.(props);
|
|
156
|
+
return result ? /* @__PURE__ */ React.createElement(Fragment, null, result) : null;
|
|
157
|
+
}
|
|
158
|
+
const MemoizedChainOfResponsibilityProvider = memo(ChainOfResponsibilityProvider);
|
|
159
|
+
return Object.freeze({
|
|
160
|
+
Provider: MemoizedChainOfResponsibilityProvider,
|
|
161
|
+
Proxy: ChainOfResponsibilityProxy,
|
|
162
|
+
reactComponent,
|
|
163
|
+
useBuildRenderCallback
|
|
164
|
+
// TODO: Consider adding back `asMiddleware`.
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
var createChainOfResponsibilityAsRenderCallback_default = createChainOfResponsibility;
|
|
168
|
+
export {
|
|
169
|
+
createChainOfResponsibilityAsRenderCallback_default as createChainOfResponsibility
|
|
170
|
+
};
|
|
171
|
+
//# sourceMappingURL=react-chain-of-responsibility.preview.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/preview/createChainOfResponsibilityAsRenderCallback.tsx","../src/preview/private/arePropsEqual.ts"],"sourcesContent":["import { applyMiddleware } from 'handler-chain';\nimport React, {\n createContext,\n Fragment,\n memo,\n useCallback,\n useContext,\n useMemo,\n type ComponentType,\n type PropsWithChildren,\n type ReactElement,\n type ReactNode\n} from 'react';\nimport { custom, function_, object, parse, safeParse } from 'valibot';\n\nimport arePropsEqual from './private/arePropsEqual.ts';\n\n// TODO: Related to https://github.com/microsoft/TypeScript/issues/17002.\n// typescript@5.2.2 has a bug, Array.isArray() is a type predicate but only works with mutable array, not readonly array.\ndeclare global {\n interface ArrayConstructor {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n isArray(arg: any): arg is readonly any[];\n }\n}\n\ntype BaseProps = object;\n\ntype CreateChainOfResponsibilityOptions = {\n /**\n * Allows one component to pass different set of props to its downstream component. Default is false.\n *\n * It is recommended to keep this settings as default to prevent newly added component from unexpectedly changing behavior of downstream components.\n */\n readonly allowOverrideProps?: boolean | undefined;\n\n /**\n * Allows a middleware to pass another request object when calling its next middleware. Default is false.\n *\n * It is recommended to keep this settings as default ot prevent newly added middleware from unexpectedly changing behavior of downstream middleware.\n *\n * To prevent upstream middleware from modifying the request, the request object should be set to be immutable through `Object.freeze`.\n */\n readonly passModifiedRequest?: boolean | undefined;\n};\n\ntype ChainOfResponsibility<Request, Props extends BaseProps, Init> = {\n readonly Provider: ComponentType<ProviderProps<Request, Props, Init>> & InferenceHelper<Request, Props, Init>;\n readonly Proxy: ComponentType<ProxyProps<Request, Props>>;\n readonly reactComponent: ReactComponentHandlerResult<Props>;\n readonly useBuildRenderCallback: () => UseBuildRenderCallback<Request, Props>;\n};\n\n// TODO: Maybe this one should be local.\n// Verify that reactComponent() from an instance of CoR should throw error when used in another instance of CoR.\nconst DO_NOT_CREATE_THIS_OBJECT_YOURSELF = Symbol();\n\ntype ComponentRenderer<Props> = (props: Props) => ReactElement | null;\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst componentHandlerResultSchema = custom<ComponentHandlerResult<any>>(\n value =>\n safeParse(object({ render: function_() }), value).success &&\n !!value &&\n typeof value === 'object' &&\n DO_NOT_CREATE_THIS_OBJECT_YOURSELF in value,\n 'react-chain-of-responsibility: middleware must return value constructed by reactComponent()'\n);\n\ninterface ComponentHandlerResult<Props extends BaseProps> {\n readonly [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: undefined;\n readonly render: (overridingProps?: Partial<Props> | undefined) => ReactElement | null;\n}\n\ntype ComponentHandler<Request, Props extends BaseProps> = (\n request: Request\n) => ComponentHandlerResult<Props> | undefined;\n\ntype ComponentEnhancer<Request, Props extends BaseProps> = (\n next: ComponentHandler<Request, Props>\n) => ComponentHandler<Request, Props>;\n\ntype ComponentMiddleware<Request, Props extends BaseProps, Init = undefined> = (\n init: Init\n) => ComponentEnhancer<Request, Props>;\n\ntype ReactComponentHandlerResult<Props extends object> = <P extends Props>(\n component: ComponentType<P>,\n bindProps?:\n | (Partial<Props> & Omit<P, keyof Props>)\n | ((props: Props) => Partial<Props> & Omit<P, keyof Props>)\n | undefined\n) => ComponentHandlerResult<Props>;\n\ntype UseBuildRenderCallbackOptions<Props> = {\n readonly fallbackComponent?: ComponentType<Props> | undefined;\n};\n\ninterface UseBuildRenderCallback<Request, Props extends BaseProps> {\n (request: Request, options?: undefined | UseBuildRenderCallbackOptions<Props>): ComponentRenderer<Props> | undefined;\n}\n\ntype BuildContextType<Request, Props extends BaseProps> = {\n readonly enhancer: ComponentEnhancer<Request, Props>;\n};\n\ntype RenderContextType<Props> = {\n readonly originalProps: Props;\n};\n\ntype ProviderProps<Request, Props extends BaseProps, Init> = PropsWithChildren<{\n readonly middleware: readonly ComponentMiddleware<Request, Props, Init>[];\n}> &\n (Init extends never | void\n ? { readonly init?: undefined }\n : Init extends undefined | void\n ? { readonly init?: Init }\n : { readonly init: Init });\n\ntype ProxyProps<Request, Props extends BaseProps> = Props & {\n readonly fallbackComponent?: ComponentType<Props> | undefined;\n readonly request: Request;\n};\n\ntype InferenceHelper<Request, Props extends BaseProps, Init> = {\n readonly '~types': {\n readonly init: Init;\n readonly middleware: ComponentMiddleware<Request, Props, Init>;\n readonly props: Props;\n readonly proxyProps: ProxyProps<Request, Props>;\n readonly providerProps: ProviderProps<Request, Props, Init>;\n readonly request: Request;\n };\n};\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferInit<T extends InferenceHelper<any, any, any>> = T['~types']['init'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferMiddleware<T extends InferenceHelper<any, any, any>> = T['~types']['middleware'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProps<T extends InferenceHelper<any, any, any>> = T['~types']['props'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProxyProps<T extends InferenceHelper<any, any, any>> = T['~types']['proxyProps'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferProviderProps<T extends InferenceHelper<any, any, any>> = T['~types']['providerProps'];\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype InferRequest<T extends InferenceHelper<any, any, any>> = T['~types']['request'];\n\nfunction createComponentHandlerResult<Props extends BaseProps>(\n render: (overridingProps?: Partial<Props> | undefined) => ReactElement | null\n): ComponentHandlerResult<Props> {\n return Object.freeze({ [DO_NOT_CREATE_THIS_OBJECT_YOURSELF]: undefined, render });\n}\n\nfunction createChainOfResponsibility<\n Request = void,\n Props extends BaseProps = { readonly children?: never },\n Init = void\n>(options: CreateChainOfResponsibilityOptions = {}): ChainOfResponsibility<Request, Props, Init> {\n // Freeze options to prevent accidental change.\n options = Object.freeze({ ...options });\n\n const BuildContext = createContext<BuildContextType<Request, Props>>(\n Object.freeze({ enhancer: next => request => next(request) })\n );\n\n const RenderContext = createContext<RenderContextType<Props>>(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n new Proxy({} as any, {\n get() {\n // The following is assertion, there is no way to hit this line.\n /* istanbul ignore next */\n throw new Error(\n 'react-chain-of-responsibility: this hook cannot be used outside of <Proxy> and useBuildRenderCallback()'\n );\n }\n })\n );\n\n function reactComponent<P extends Props>(\n component: ComponentType<P>,\n // For `bindProps` of type function, do not do side-effect in it, it may not be always called in all scenarios.\n bindProps?:\n | (Partial<Props> & Omit<P, keyof Props>)\n | ((props: Props) => Partial<Props> & Omit<P, keyof Props>)\n | undefined\n ): ComponentHandlerResult<Props> {\n return createComponentHandlerResult((overridingProps?: Partial<Props> | undefined) => (\n <ComponentWithProps\n bindProps={bindProps}\n component={component as ComponentType<Props>}\n overridingProps={overridingProps}\n />\n ));\n }\n\n const ComponentWithProps = memo(function ComponentWithProps({\n bindProps,\n component: Component,\n overridingProps\n }: {\n readonly bindProps?: Partial<Props> | ((props: Props) => Partial<Props>) | undefined;\n readonly component: ComponentType<Props>;\n readonly overridingProps?: Partial<Props> | undefined;\n }) {\n const { allowOverrideProps } = options;\n const { originalProps: renderCallbackProps } = useContext(RenderContext);\n\n if (overridingProps && !arePropsEqual(overridingProps, renderCallbackProps) && !allowOverrideProps) {\n console.warn('react-chain-of-responsibility: \"allowOverrideProps\" must be set to true to override props');\n }\n\n const props = Object.freeze(\n allowOverrideProps ? { ...renderCallbackProps, ...overridingProps } : { ...renderCallbackProps }\n );\n\n return <Component {...props} {...(typeof bindProps === 'function' ? bindProps(props) : bindProps)} />;\n });\n\n const useBuildRenderCallback: () => UseBuildRenderCallback<Request, Props> = () => {\n const { enhancer } = useContext(BuildContext);\n\n return useCallback(\n (request, buildOptions = {}) => {\n const result =\n // Put the \"fallbackComponent\" as the last one in the chain.\n enhancer(() => {\n const { fallbackComponent } = buildOptions;\n\n if (!fallbackComponent) {\n console.warn(\n 'react-chain-of-responsibility: the request has fall through all middleware, set \"fallbackComponent\" as a catchall',\n request\n );\n\n // For clarity, we are returning `undefined` instead of `() => undefined`.\n return;\n }\n\n // `fallbackComponent` do not need `overridingProps` because it is the last one in the chain, it would not have the next() function.\n return reactComponent(fallbackComponent);\n })(request);\n\n return (\n result &&\n ((originalProps: Props) => (\n // This is render function, we cannot call any hooks here.\n <BuildRenderCallback originalProps={originalProps} render={result.render} />\n ))\n );\n },\n [enhancer]\n );\n };\n\n type BuildRenderCallbackProps = {\n readonly originalProps: Props;\n readonly render: () => ReactNode;\n };\n\n // Do not memoize <BuildRenderCallback>.\n // `bindProps` may have side effect and we want to be re-rendered to capture the side-effect.\n // To prevent wasted render, web devs should memoize it themselves.\n function BuildRenderCallback({ originalProps, render }: BuildRenderCallbackProps) {\n const context = useMemo<RenderContextType<Props>>(() => Object.freeze({ originalProps }), [originalProps]);\n\n return <RenderContext.Provider value={context}>{render()}</RenderContext.Provider>;\n }\n\n function ChainOfResponsibilityProvider({ children, init, middleware }: ProviderProps<Request, Props, Init>) {\n if (!Array.isArray(middleware) || middleware.some(middleware => typeof middleware !== 'function')) {\n throw new Error('react-chain-of-responsibility: \"middleware\" prop must be an array of functions');\n }\n\n // Remap the middleware, so all inputs/outputs are validated.\n const fortifiedMiddleware = useMemo(\n () =>\n Object.freeze(\n middleware.map<ComponentMiddleware<Request, Props, Init>>(fn => (init: Init) => {\n const enhancer = fn(init);\n\n return next => originalRequest => {\n // False positive: although we did not re-assign the variable from true, it was initialized as undefined.\n // eslint-disable-next-line prefer-const\n let hasReturned: boolean;\n\n const returnValue = enhancer(nextRequest => {\n if (hasReturned) {\n throw new Error(\n 'react-chain-of-responsibility: next() cannot be called after the function had returned synchronously'\n );\n }\n\n // We do not allow passing void/undefined to next() because it would be confusing whether to keep the original request or pass an undefined.\n !options.passModifiedRequest &&\n !Object.is(nextRequest, originalRequest) &&\n console.warn(\n 'react-chain-of-responsibility: next() must be called with the original request, otherwise, set \"options.passModifiedRequest\" to true to pass a different request object downstream'\n );\n\n return next(options.passModifiedRequest ? nextRequest : originalRequest);\n })(originalRequest);\n\n hasReturned = true;\n\n // Make sure the return value is built using our helper function for forward-compatibility reason.\n return returnValue && parse(componentHandlerResultSchema, returnValue);\n };\n })\n ),\n [middleware]\n );\n\n const { enhancer: parentEnhancer } = useContext(BuildContext);\n\n const enhancer = useMemo<ComponentEnhancer<Request, Props>>(\n () =>\n // We are reversing because it is easier to read:\n // - With reverse, [a, b, c] will become a(b(c(fn)))\n // - Without reverse, [a, b, c] will become c(b(a(fn)))\n applyMiddleware<ComponentHandlerResult<Props> | undefined, Request, Init>(\n ...[...fortifiedMiddleware, ...[() => parentEnhancer]]\n )(init as Init),\n [init, fortifiedMiddleware, parentEnhancer]\n );\n\n const contextValue = useMemo<BuildContextType<Request, Props>>(() => Object.freeze({ enhancer }), [enhancer]);\n\n return <BuildContext.Provider value={contextValue}>{children}</BuildContext.Provider>;\n }\n\n function ChainOfResponsibilityProxy({ fallbackComponent, request, ...props }: ProxyProps<Request, Props>) {\n const result = useBuildRenderCallback()(request, { fallbackComponent })?.(props as Props);\n\n return result ? <Fragment>{result}</Fragment> : null;\n }\n\n const MemoizedChainOfResponsibilityProvider =\n memo<ProviderProps<Request, Props, Init>>(ChainOfResponsibilityProvider);\n\n return Object.freeze({\n Provider: MemoizedChainOfResponsibilityProvider as typeof MemoizedChainOfResponsibilityProvider &\n InferenceHelper<Request, Props, Init>,\n Proxy: ChainOfResponsibilityProxy,\n reactComponent,\n useBuildRenderCallback\n\n // TODO: Consider adding back `asMiddleware`.\n });\n}\n\nexport default createChainOfResponsibility;\nexport {\n type ChainOfResponsibility,\n type ComponentEnhancer,\n type ComponentHandler,\n type ComponentHandlerResult,\n type ComponentMiddleware,\n type ComponentRenderer,\n type CreateChainOfResponsibilityOptions,\n type InferenceHelper,\n type InferInit,\n type InferMiddleware,\n type InferProps,\n type InferProviderProps,\n type InferProxyProps,\n type InferRequest,\n type ProviderProps,\n type ProxyProps,\n type ReactComponentHandlerResult,\n type UseBuildRenderCallback,\n type UseBuildRenderCallbackOptions\n};\n","// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport default function arePropsEqual<T extends Record<string, any>>(x: T, y: T): boolean {\n if (Object.is(x, y)) {\n return true;\n }\n\n const mapOfX = new Map(Object.entries(x));\n const mapOfY = new Map(Object.entries(y));\n\n if (mapOfX.size !== mapOfY.size) {\n return false;\n }\n\n const keys = new Set([...mapOfX.keys(), ...mapOfY.keys()]);\n\n for (const key of keys) {\n if (!Object.is(mapOfX.get(key), mapOfY.get(key))) {\n return false;\n }\n }\n\n return true;\n}\n"],"mappings":"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","names":["ComponentWithProps","middleware","init","enhancer"]}
|