react-with-hoc
Version:
Work with React and HOCs (Higher-Order Components)
88 lines • 11.3 kB
JavaScript
import React from "react";
import { newHoc } from "../utils/newHoc";
/**
* Wraps the component with the provided Component Wrapper
*
* @example
* const NewComponent = withWrapper(Wrapper)(Component)
* <NewComponent a="a" b="b" c="c" />
* // is equivalent to
* <Wrapper> // by default, it does not bring any prop
* <Component a="a" b="b" c="c" />
* </Wrapper>
*
* @example
* const NewComponent = withWrapper(Wrapper, { pickProps: ['a', 'b'] })(Component)
* <NewComponent a="a" b="b" c="c" />
* // is equivalent to
* <Wrapper a="a" b="b">
* <Component a="a" b="b" c="c" />
* </Wrapper>
*
* @example
* const NewComponent = withWrapper(Wrapper, { omitProps: ['a', 'b'] })(Component)
* <NewComponent a="a" b="b" c="c" />
* // is equivalent to
* <Wrapper c="c">
* <Component a="a" b="b" c="c" />
* </Wrapper>
*
* @example
* // to carry all props, use omitProps with empty array
* const NewComponent = withWrapper(Wrapper, { omitProps: [] })(Component)
* <NewComponent a="a" b="b" c="c" />
* // is equivalent to
* <Wrapper a="a" b="b" c="c">
* <Component a="a" b="b" c="c" />
* </Wrapper>
*/
export const withWrapper = newHoc(function withWrapper(Component, Wrapper, {
pickProps,
omitProps
} = {
pickProps: []
}) {
if (process.env.NODE_ENV !== "production") {
const bothAssigned = pickProps && omitProps;
const noneAssigned = !pickProps && !omitProps;
if (bothAssigned || noneAssigned) {
throw new Error("withWrapper should have either pickProps or omitProps assigned");
}
}
const set = new Set(pickProps || omitProps);
return function WithWrapper(props) {
if (set.size === 0 && pickProps) {
return /*#__PURE__*/React.createElement(Wrapper, null, /*#__PURE__*/React.createElement(Component, {
...props
}));
}
if (set.size === 0 && omitProps) {
return /*#__PURE__*/React.createElement(Wrapper, {
...props
}, /*#__PURE__*/React.createElement(Component, {
...props
}));
}
const parentProps = {};
if (pickProps) {
for (const prop in props) {
if (set.has(prop)) {
parentProps[prop] = props[prop];
}
}
} else if (omitProps) {
for (const prop in props) {
if (!set.has(prop)) {
parentProps[prop] = props[prop];
}
}
}
return /*#__PURE__*/React.createElement(Wrapper, {
...parentProps
}, /*#__PURE__*/React.createElement(Component, {
...props
}));
};
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","newHoc","withWrapper","Component","Wrapper","pickProps","omitProps","process","env","NODE_ENV","bothAssigned","noneAssigned","Error","set","Set","WithWrapper","props","size","createElement","parentProps","prop","has"],"sources":["../../../src/hocs/withWrapper.tsx"],"sourcesContent":["import React, { ComponentType, FunctionComponent } from \"react\";\nimport { IntersectionFn, ToSchema } from \"../types/Fn\";\nimport { Hoc } from \"../types/Hoc\";\nimport { newHoc } from \"../utils/newHoc\";\n\nexport interface WithWrapperHoc {\n  <\n    const WrapperProps,\n    const PickOptions extends string[] = [never],\n    const OmitOptions extends string[] = [never],\n  >(\n    Wrapper: ComponentType<WrapperProps>,\n    options?:\n      | {\n          /**\n           * @example\n           * const NewComponent = withWrapper(Wrapper, { pickProps: ['a', 'b'] })(Component)\n           * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n           * // is equivalent to\n           * <Wrapper a=\"a\" b=\"b\">\n           *   <Component a=\"a\" b=\"b\" c=\"c\" />\n           * </Wrapper>\n           */\n          pickProps: PickOptions;\n          omitProps?: undefined;\n        }\n      | {\n          /**\n           * @example\n           * const NewComponent = withWrapper(Wrapper, { omitProps: ['a', 'b'] })(Component)\n           * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n           * // is equivalent to\n           * <Wrapper c=\"c\">\n           *   <Component a=\"a\" b=\"b\" c=\"c\" />\n           * </Wrapper>\n           */\n          omitProps: OmitOptions;\n          pickProps?: undefined;\n        },\n  ): PickOptions extends [never]\n    ? OmitOptions extends [never]\n      ? // has none\n        Hoc<[]>\n      : // has omit\n        Hoc<\n          [\n            IntersectionFn<\n              Exclude<\n                ToSchema<WrapperProps>,\n                [OmitOptions[number] | \"children\", any]\n              >\n            >,\n          ]\n        >\n    : // has pick\n      Hoc<\n        [\n          IntersectionFn<\n            Extract<\n              Exclude<ToSchema<WrapperProps>, [\"children\", any]>,\n              [PickOptions[number], any]\n            >\n          >,\n        ]\n      >;\n}\n\n/**\n * Wraps the component with the provided Component Wrapper\n *\n * @example\n * const NewComponent = withWrapper(Wrapper)(Component)\n * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n * // is equivalent to\n * <Wrapper> // by default, it does not bring any prop\n *   <Component a=\"a\" b=\"b\" c=\"c\" />\n * </Wrapper>\n *\n * @example\n * const NewComponent = withWrapper(Wrapper, { pickProps: ['a', 'b'] })(Component)\n * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n * // is equivalent to\n * <Wrapper a=\"a\" b=\"b\">\n *   <Component a=\"a\" b=\"b\" c=\"c\" />\n * </Wrapper>\n *\n * @example\n * const NewComponent = withWrapper(Wrapper, { omitProps: ['a', 'b'] })(Component)\n * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n * // is equivalent to\n * <Wrapper c=\"c\">\n *   <Component a=\"a\" b=\"b\" c=\"c\" />\n * </Wrapper>\n *\n * @example\n * // to carry all props, use omitProps with empty array\n * const NewComponent = withWrapper(Wrapper, { omitProps: [] })(Component)\n * <NewComponent a=\"a\" b=\"b\" c=\"c\" />\n * // is equivalent to\n * <Wrapper a=\"a\" b=\"b\" c=\"c\">\n *   <Component a=\"a\" b=\"b\" c=\"c\" />\n * </Wrapper>\n */\nexport const withWrapper = newHoc<WithWrapperHoc>(function withWrapper(\n  Component: ComponentType<any>,\n  Wrapper: ComponentType<any>,\n  {\n    pickProps,\n    omitProps,\n  }: {\n    pickProps?: string[];\n    omitProps?: string[];\n  } = { pickProps: [] },\n): FunctionComponent {\n  if (process.env.NODE_ENV !== \"production\") {\n    const bothAssigned = pickProps && omitProps;\n    const noneAssigned = !pickProps && !omitProps;\n    if (bothAssigned || noneAssigned) {\n      throw new Error(\n        \"withWrapper should have either pickProps or omitProps assigned\",\n      );\n    }\n  }\n\n  const set = new Set(pickProps || omitProps);\n\n  return function WithWrapper(props: any): JSX.Element {\n    if (set.size === 0 && pickProps) {\n      return (\n        <Wrapper>\n          <Component {...props} />\n        </Wrapper>\n      );\n    }\n\n    if (set.size === 0 && omitProps) {\n      return (\n        <Wrapper {...props}>\n          <Component {...props} />\n        </Wrapper>\n      );\n    }\n\n    const parentProps = {} as any;\n    if (pickProps) {\n      for (const prop in props) {\n        if (set.has(prop)) {\n          parentProps[prop] = props[prop];\n        }\n      }\n    } else if (omitProps) {\n      for (const prop in props) {\n        if (!set.has(prop)) {\n          parentProps[prop] = props[prop];\n        }\n      }\n    }\n\n    return (\n      <Wrapper {...parentProps}>\n        <Component {...props} />\n      </Wrapper>\n    );\n  };\n});\n"],"mappings":"AAAA,OAAOA,KAAK,MAA4C,OAAO;AAG/D,SAASC,MAAM,QAAQ,iBAAiB;AAgExC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,MAAMC,WAAW,GAAGD,MAAM,CAAiB,SAASC,WAAWA,CACpEC,SAA6B,EAC7BC,OAA2B,EAC3B;EACEC,SAAS;EACTC;AAIF,CAAC,GAAG;EAAED,SAAS,EAAE;AAAG,CAAC,EACF;EACnB,IAAIE,OAAO,CAACC,GAAG,CAACC,QAAQ,KAAK,YAAY,EAAE;IACzC,MAAMC,YAAY,GAAGL,SAAS,IAAIC,SAAS;IAC3C,MAAMK,YAAY,GAAG,CAACN,SAAS,IAAI,CAACC,SAAS;IAC7C,IAAII,YAAY,IAAIC,YAAY,EAAE;MAChC,MAAM,IAAIC,KAAK,CACb,gEACF,CAAC;IACH;EACF;EAEA,MAAMC,GAAG,GAAG,IAAIC,GAAG,CAACT,SAAS,IAAIC,SAAS,CAAC;EAE3C,OAAO,SAASS,WAAWA,CAACC,KAAU,EAAe;IACnD,IAAIH,GAAG,CAACI,IAAI,KAAK,CAAC,IAAIZ,SAAS,EAAE;MAC/B,oBACEL,KAAA,CAAAkB,aAAA,CAACd,OAAO,qBACNJ,KAAA,CAAAkB,aAAA,CAACf,SAAS;QAAA,GAAKa;MAAK,CAAG,CAChB,CAAC;IAEd;IAEA,IAAIH,GAAG,CAACI,IAAI,KAAK,CAAC,IAAIX,SAAS,EAAE;MAC/B,oBACEN,KAAA,CAAAkB,aAAA,CAACd,OAAO;QAAA,GAAKY;MAAK,gBAChBhB,KAAA,CAAAkB,aAAA,CAACf,SAAS;QAAA,GAAKa;MAAK,CAAG,CAChB,CAAC;IAEd;IAEA,MAAMG,WAAW,GAAG,CAAC,CAAQ;IAC7B,IAAId,SAAS,EAAE;MACb,KAAK,MAAMe,IAAI,IAAIJ,KAAK,EAAE;QACxB,IAAIH,GAAG,CAACQ,GAAG,CAACD,IAAI,CAAC,EAAE;UACjBD,WAAW,CAACC,IAAI,CAAC,GAAGJ,KAAK,CAACI,IAAI,CAAC;QACjC;MACF;IACF,CAAC,MAAM,IAAId,SAAS,EAAE;MACpB,KAAK,MAAMc,IAAI,IAAIJ,KAAK,EAAE;QACxB,IAAI,CAACH,GAAG,CAACQ,GAAG,CAACD,IAAI,CAAC,EAAE;UAClBD,WAAW,CAACC,IAAI,CAAC,GAAGJ,KAAK,CAACI,IAAI,CAAC;QACjC;MACF;IACF;IAEA,oBACEpB,KAAA,CAAAkB,aAAA,CAACd,OAAO;MAAA,GAAKe;IAAW,gBACtBnB,KAAA,CAAAkB,aAAA,CAACf,SAAS;MAAA,GAAKa;IAAK,CAAG,CAChB,CAAC;EAEd,CAAC;AACH,CAAC,CAAC"}
//# sourceMappingURL=withWrapper.js.map