react-with-hoc
Version:
Work with React and HOCs (Higher-Order Components)
123 lines (121 loc) • 15.3 kB
JavaScript
;
var _react = require("react");
var _newHoc = require("../utils/newHoc");
const isReactNode = value => (0, _react.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>
* </>
*/
const withForEach = exports.withForEach = (0, _newHoc.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.default.createElement(Component, {
key: key(newProps),
...rest
}, children);
}
return /*#__PURE__*/_react.default.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 (0, _react.createElement)(_react.Fragment, null, ...children);
};
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","require","_newHoc","isReactNode","value","isValidElement","withForEach","exports","newHoc","Component","target","indexName","key","props","valueName","componentWithKey","newProps","children","rest","default","createElement","WithForEach","Array","from","length","map","_v","i","isArray","v","Object","keys","values","entries","index","process","env","NODE_ENV","never","Error","Fragment"],"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