react-with-hoc
Version:
Work with React and HOCs (Higher-Order Components)
121 lines (120 loc) • 15.1 kB
JavaScript
import React, { Fragment, createElement, isValidElement } from "react";
import { newHoc } from "../utils/newHoc";
const isReactNode = value => isValidElement(value) || typeof value !== "object" || value === null;
/**
* Like Array.map but for React components.
* Concatenates components for each item in array, range or object
*
* @experimental It needs to discuss its name
*
* @example
* const Component = withForEach(3)(Example)
* <Component {...props} />
* // is equivalent to
* <>
* <Example key={0} {...props} i={0} />
* <Example key={1} {...props} i={1} />
* <Example key={2} {...props} i={2} />
* </>
*
* @example
* const Component = withForEach(["a", "b"])(Example)
* <Component {...props} />
* // is equivalent to
* <>
* <Example key={0} {...props} i={0}>a</Example>
* <Example key={1} {...props} i={1}>b</Example>
* </>
*
* @example
* const Component = withForEach({
* a: 100,
* b: 200,
* })(Example)
* <Component {...props} />
* // is equivalent to
* <>
* <Example key="a" {...props} i="a">100</Example>
* <Example key="b" {...props} i="b">200</Example>
* </>
*/
export const withForEach = newHoc(function withForEach(Component, target, {
indexName = "i",
key = props => props[indexName],
valueName = "children"
} = {}) {
function componentWithKey(newProps) {
if (valueName === "children") {
const {
children,
...rest
} = newProps;
return /*#__PURE__*/React.createElement(Component, {
key: key(newProps),
...rest
}, children);
}
return /*#__PURE__*/React.createElement(Component, {
key: key(newProps),
...newProps
});
}
return function WithForEach(props) {
const children = (() => {
if (typeof target === "number") {
return Array.from({
length: target
}).map((_v, i) => componentWithKey({
...props,
[indexName]: i
}));
}
if (Array.isArray(target)) {
if (target.length === 0) {
return [];
}
if (isReactNode(target[0])) {
return target.map((v, i) => componentWithKey({
...props,
[indexName]: i,
[valueName]: v
}));
}
return target.map((v, i) => componentWithKey({
...v,
...props,
[indexName]: i
}));
}
if (typeof target === "object" && target !== null) {
if (Object.keys(target).length === 0) {
return [];
}
if (isReactNode(Object.values(target)[0])) {
return Object.entries(target).map(([index, value]) => {
return componentWithKey({
...props,
[indexName]: index,
[valueName]: value
});
});
}
return Object.entries(target).map(([index, value]) => {
return componentWithKey({
...value,
...props,
[indexName]: index
});
});
}
if (process.env.NODE_ENV !== "production") {
const never = target;
never;
}
throw new Error("withForEach: should be used with number, Array or object");
})();
return createElement(Fragment, null, ...children);
};
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","Fragment","createElement","isValidElement","newHoc","isReactNode","value","withForEach","Component","target","indexName","key","props","valueName","componentWithKey","newProps","children","rest","WithForEach","Array","from","length","map","_v","i","isArray","v","Object","keys","values","entries","index","process","env","NODE_ENV","never","Error"],"sources":["../../../src/hocs/withForEach.tsx"],"sourcesContent":["import React, {\n  ComponentType,\n  Fragment,\n  FunctionComponent,\n  Key,\n  ReactNode,\n  createElement,\n  isValidElement,\n} from \"react\";\nimport { SetOptionalFn } from \"../types/Fn\";\nimport { Hoc } from \"../types/Hoc\";\nimport { newHoc } from \"../utils/newHoc\";\n\ninterface Options<IndexName extends string, ValueName extends string> {\n  /**\n   * @default \"i\"\n   *\n   * @example\n   * //                                            ↓ \"someName\" here as well\n   * const Component = withForEach(3, {indexName: \"someName\"})(Example)\n   * <Component {...props} />\n   * // is equivalent to\n   * <>\n   *   //                          ↓ \"someName\" here as well\n   *   <Example key={0} {...props} someName={0} />\n   *   <Example key={1} {...props} someName={1} />\n   *   <Example key={2} {...props} someName={2} />\n   * </>\n   */\n  indexName?: IndexName;\n  /**\n   * @default \"children\"\n   *\n   * @example\n   * const Component = withForEach(\n   *   [\"a\", \"b\"],\n   *   //           ↓ \"someName\" here as well\n   *   {valueName: \"someName\"}\n   * )(Example)\n   * <Component {...props} />\n   * // is equivalent to\n   * <>\n   *   //                                ↓ \"someName\" here as well\n   *   <Example key={0} {...props} i={0} someName=\"a\" />\n   *   <Example key={1} {...props} i={1} someName=\"b\" />\n   * </>\n   */\n  valueName?: ValueName;\n  /**\n   * @default (props) => props[indexName] // which is props.i\n   *\n   * @example\n   * // makes key = 2^x\n   * const Component = withForEach(4, {key: ({i}) => 2 ** i})(Example)\n   * <Component {...props} />\n   * // is equivalent to\n   * <>\n   *   //            ↓ note key values\n   *   <Example key={1} {...props} someName={0} />\n   *   <Example key={2} {...props} someName={1} />\n   *   <Example key={4} {...props} someName={2} />\n   *   <Example key={8} {...props} someName={2} />\n   * </>\n   */\n  key?: (props: object) => Key;\n}\n\ninterface WithForEachHoc {\n  <IndexName extends string, ValueName extends string>(\n    target: number | Array<unknown> | object,\n    options?: Options<IndexName, ValueName>,\n  ): Hoc<\n    [\n      SetOptionalFn<\n        | ([IndexName & \"\"] extends [never] ? IndexName : \"i\")\n        | ([ValueName & \"\"] extends [never] ? ValueName : \"children\")\n      >,\n    ]\n  >;\n}\n\nconst isReactNode = (value: unknown): boolean =>\n  isValidElement(value) || typeof value !== \"object\" || value === null;\n\n/**\n * Like Array.map but for React components.\n * Concatenates components for each item in array, range or object\n *\n * @experimental It needs to discuss its name\n *\n * @example\n * const Component = withForEach(3)(Example)\n * <Component {...props} />\n * // is equivalent to\n * <>\n *   <Example key={0} {...props} i={0} />\n *   <Example key={1} {...props} i={1} />\n *   <Example key={2} {...props} i={2} />\n * </>\n *\n * @example\n * const Component = withForEach([\"a\", \"b\"])(Example)\n * <Component {...props} />\n * // is equivalent to\n * <>\n *   <Example key={0} {...props} i={0}>a</Example>\n *   <Example key={1} {...props} i={1}>b</Example>\n * </>\n *\n * @example\n * const Component = withForEach({\n *   a: 100,\n *   b: 200,\n * })(Example)\n * <Component {...props} />\n * // is equivalent to\n * <>\n *   <Example key=\"a\" {...props} i=\"a\">100</Example>\n *   <Example key=\"b\" {...props} i=\"b\">200</Example>\n * </>\n */\nexport const withForEach = newHoc<WithForEachHoc>(function withForEach(\n  Component: ComponentType,\n  target: number | Array<unknown> | object,\n  {\n    indexName = \"i\",\n    key = (props: any): Key => props[indexName],\n    valueName = \"children\",\n  } = {},\n): FunctionComponent {\n  function componentWithKey(newProps: any): ReactNode {\n    if (valueName === \"children\") {\n      const { children, ...rest } = newProps;\n      return (\n        <Component key={key(newProps)} {...rest}>\n          {children}\n        </Component>\n      );\n    }\n    return <Component key={key(newProps)} {...newProps} />;\n  }\n\n  return function WithForEach(props: any): ReactNode {\n    const children = ((): ReactNode[] => {\n      if (typeof target === \"number\") {\n        return Array.from({ length: target }).map((_v, i) =>\n          componentWithKey({ ...props, [indexName]: i }),\n        );\n      }\n\n      if (Array.isArray(target)) {\n        if (target.length === 0) {\n          return [];\n        }\n\n        if (isReactNode(target[0])) {\n          return target.map((v, i) =>\n            componentWithKey({ ...props, [indexName]: i, [valueName]: v }),\n          );\n        }\n\n        return target.map((v, i) =>\n          componentWithKey({ ...v, ...props, [indexName]: i }),\n        );\n      }\n\n      if (typeof target === \"object\" && target !== null) {\n        if (Object.keys(target).length === 0) {\n          return [];\n        }\n\n        if (isReactNode(Object.values(target)[0])) {\n          return Object.entries(target).map(([index, value]) => {\n            return componentWithKey({\n              ...props,\n              [indexName]: index,\n              [valueName]: value,\n            });\n          });\n        }\n\n        return Object.entries(target).map(([index, value]) => {\n          return componentWithKey({ ...value, ...props, [indexName]: index });\n        });\n      }\n\n      if (process.env.NODE_ENV !== \"production\") {\n        const never: never = target;\n        never;\n      }\n      throw new Error(\n        \"withForEach: should be used with number, Array or object\",\n      );\n    })();\n\n    return createElement(Fragment, null, ...children);\n  };\n});\n"],"mappings":"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"}
//# sourceMappingURL=withForEach.js.map