compose-components
Combine multiple React components (like providers) while maintaining fully typed props support. Up to 10 times faster than using cloneElement
.
Installation
pnpm install compose-components
Usage
import { ComposeComponents, composeComponents } from "compose-components";
const App = () => (
<ComposeComponents
renderComponents={composeComponents(
[PaletteProvider, { mode: "light" }],
[LayoutProvider, { layout: "default" }],
[TypographyProvider, { font: "sans" }],
)}
>
<div>App</div>
</ComposeComponents>
);
declare function PaletteProvider({ children }: React.PropsWithChildren<{ mode: "light" | "dark" }>): React.ReactNode;
declare function LayoutProvider({ children }: React.PropsWithChildren<{ layout: "default" | "full" }>): React.ReactNode;
declare function TypographyProvider({ children }: React.PropsWithChildren<{ font: "sans" | "serif" }>): React.ReactNode;
Performance
Since cloneElement
is not used, this approach is nearly 10x faster than the version that uses cloneElement
.
compose-components [ main][+][📦 v0.0.0][ v22.6.0]
❯ pnpm benchmark
> compose-components@0.0.0 benchmark
> pnpm -F "./benchmark" run start
> benchmark@0.0.0 start compose-components/benchmark
> tsx ./main.tsx
┌─────────┬────────────────────────┬─────────────┬───────────────────┬──────────┬─────────┐
│ (index) │ Task Name │ ops/sec │ Average Time (ns) │ Margin │ Samples │
├─────────┼────────────────────────┼─────────────┼───────────────────┼──────────┼─────────┤
│ 0 │ 'with cloneElement' │ '222,705' │ 4490.229389930139 │ '±0.25%' │ 66812 │
│ 1 │ 'without cloneElement' │ '2,049,821' │ 487.8472469985147 │ '±0.35%' │ 614947 │
└─────────┴────────────────────────┴─────────────┴───────────────────┴──────────┴─────────┘