Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/fpapado/precompose-props
Prop mapping made easy at <1k
https://github.com/fpapado/precompose-props
contramap preact props react theming
Last synced: about 2 months ago
JSON representation
Prop mapping made easy at <1k
- Host: GitHub
- URL: https://github.com/fpapado/precompose-props
- Owner: fpapado
- License: mit
- Created: 2018-02-11T21:11:05.000Z (almost 7 years ago)
- Default Branch: master
- Last Pushed: 2018-02-13T14:10:16.000Z (almost 7 years ago)
- Last Synced: 2024-09-26T07:51:38.950Z (3 months ago)
- Topics: contramap, preact, props, react, theming
- Language: JavaScript
- Homepage: https://npm.im/precompose-props
- Size: 94.7 KB
- Stars: 0
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE.md
Awesome Lists containing this project
README
# Precompose-props
> Prop mapping made easy.[![travis](https://travis-ci.org/fpapado/precompose-props.svg?branch=master)](https://travis-ci.org/fpapado/precompose-props)
[![npm](https://img.shields.io/npm/v/precompose-props.svg)](https://www.npmjs.com/package/precompose-props)## Table of Contents
- [Install](#install)
- [Motivation](#install)
- [Usage](#usage)
- [API](#api)
- [License](#license)## Install
This package is distributed via [npm](https://www.npmjs.com/get-npm).```shell
$ npm install --save precompose-props
# or
$ yarn add precompose-props
```Then import according to your modules model and bundler, such as [Rollup](https://rollupjs.org/guide/en) and [Webpack](https://webpack.js.org/)
```js
// ES6 Modules
// For all possible functions to import, look at "export" in src/index.js
import { contramap, withTheme } from 'precompose-props';/// CommonJS modules
var precomposeProps = require('precompose-props');
```A [UMD](https://github.com/umdjs/umd) version is also available on [unpkg](https://unpkg.com/):
```html```
## Motivation
When working with React, I often want to make a component that "wraps" the props of another component, mapping from one set to another.
This is especially true when composing styles; I want to have a higher-level "theme" prop, that collects a set of lower-level "style" props.
Doing this manually has some annoying boilerplate, and leads to more noise than I'd like.Ideally, I would be able to do something like this:
```jsx
const Th = ({fontWeight, color, children}) => {children};const Cell = withTheme({bold: {fontWeight: '7', color: 'black'}})(Th);
```### Ideas
In his talk ["Oh Composable World!"](https://www.youtube.com/watch?v=SfWR3dKnFIo), Brian Lonsdorf (aka Dr. Boolean) goes over how to use `contramap` to compose props.
A contramap in this context allows us to use a function pre-process a set of props, before passing the result on to a component. This is exactly what we want!
(Also, that sounds like `mapStateToProps()` from `react-redux`).Let's define contramap:
```js
const contramap = mapFn => Component => props => Component(mapFn(props));
```In his talk, he defines a `Comp = x => {...}` function, because there is a larger point about composition and concatenation for React (contravariance and monoids).
I am not doing that, because calling .fold() would look out of place in our codebase.
Would be simple to add and fun to try though :)## Usage
```jsx
import React from "react";
import { render } from "react-dom";
import {
withTheme,
named,
toggle,
contramap,
concatAndMergeProps
} from "precompose-props";// The lower component
const P = ({ measure, lineHeight, fontSize, fontWeight, children }) => (
{children}
);// Higher component
const Text = withTheme({
bold: toggle({ fontWeight: "700" }),
kind: named(
{ copy: { lineHeight: "1.5", measure: "34em" } },
{ heading: { lineHeight: "1.25" } }
)
})(P);// Equivalent forms
// Using concatAndMergeProps directly with contramap
const TextAlt = contramap(
concatAndMergeProps({
bold: toggle({ fontWeight: "700" }),
kind: named(
{ copy: { lineHeight: "1.5", measure: "34em" } },
{ heading: { lineHeight: "1.25" } }
)
})
)(P);// Using contramap with a custom function
/*
mapFakeToProps(({bold, kind, ...rest}) => "Fill this in");
const TextManual = contramap(mapFakeToProps)(P);
*/const App = () => (
I am a Heading
This is a text oh this is a text, and would you guess, this is copy text
with max widths and stuff.
This is a text oh this is a text, and would you guess, this is copy text
with max widths and stuff.
);render(, document.getElementById("root"));
```### Any abstraction you like
Another piece of wisdom by Dr. Boolean (I'm a fan, can't you tell?) is on API design (paraphrasing):
- A set of primitives
- A way to compose them
- A set of precomposed thingsArmed with that thought, here are the functions provided and their use cases, lower- to higher- level.
**Low-level**
- `contramap`
apply a mapping function from one set of props to another, pass to component- `concatProps`
concatenate props according to a specification- `concatAndMergeProps`
concatenate props according to a specification, merges**Specification Utilities**
A specification is of the form
```ts
{[K in higherProp]: mapFn}mapFn: higherPropValue => Partial`.
```- `toggle(obj)(value)`
return obj if value is true- `named(obj)(value)`
return the entry at obj[value]**Higher-level**
- `withTheme`
maps props with specification, merge other props- `mapTheme`
maps props with specification, do not merge other propsIf you see a pattern there, it's because the latter are simply compositions of the former!
Functions are curried by default. I have not used Ramda's `curry`; this is open to change.### Examples
[Edit the example above on CodeSandbox](https://codesandbox.io/s/0107ywo83l)### API
#### Table of Contents
## TODO
- Types
- Example folder## Inspiration
- Brian Lonsdorf (drboolean) for his talks on functional JS. Even when the language is not the best for it, I
will occasionally run into cases where I think about concatenation or the order of operations.
- Jason Miller (developit) for his package build setups, and `microbundle`. They were a great starting
point to figure out how to publish this damn thing.