Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/fahad19/proppy

Functional props composition for UI components (React.js & Vue.js)
https://github.com/fahad19/proppy

functional-programming javascript preact props react redux rxjs vue

Last synced: 3 days ago
JSON representation

Functional props composition for UI components (React.js & Vue.js)

Awesome Lists containing this project

README

        

![ProppyJS](./site/assets/img/banner.png)


Functional props composition for components


A 1.5kB library integrating with your favourite UI framework





NPM version



Build Status



Gitter




Website

|

Documentation

|

Packages

|

Contributing



Built with ❤︎ by
@fahad19 and

contributors


# What is ProppyJS?

ProppyJS is a tiny `1.5kB` JavaScript library for composing props (object that components receive to render themselves).

It comes with various integration packages giving you the freedom to use it popular rendering libraries.

# What does it do?

[![ProppyJS flow of props](./site/assets/img/proppy-flow.gif)](https://proppyjs.com)

The idea is you express the behaviour of your Component as props first, and then connect it to your Component (which can be either React, Vue.js or Preact) using the same API of Proppy.

The API gives you access to other application-wide dependencies too (like a store using Redux) for convenience anywhere in the components tree.

# Packages

| Package | Status | Size | Description |
|----------------------|------------------------------------------------------------|---------|---------------------------|
| [proppy] | [![proppy-status]][proppy-package] | `1.5K` | Core package |
| [proppy-react] | [![proppy-react-status]][proppy-react-package] | `1.0K` | React integration |
| [proppy-vue] | [![proppy-vue-status]][proppy-vue-package] | `0.7K` | Vue.js integration |
| [proppy-preact] | [![proppy-preact-status]][proppy-preact-package] | `1.1K` | Preact integration |
| [proppy-redux] | [![proppy-redux-status]][proppy-redux-package] | `0.6K` | Redux integration |
| [proppy-rx] | [![proppy-rx-status]][proppy-rx-package] | `0.6K` | RxJS integration |

[proppy]: https://proppyjs.com/docs/packages/proppy
[proppy-react]: https://proppyjs.com/docs/packages/proppy-react
[proppy-vue]: https://proppyjs.com/docs/packages/proppy-vue
[proppy-preact]: https://proppyjs.com/docs/packages/proppy-preact
[proppy-redux]: https://proppyjs.com/docs/packages/proppy-redux
[proppy-rx]: https://proppyjs.com/docs/packages/proppy-rx
[proppy-frint-react]: https://proppyjs.com/docs/packages/proppy-frint-react

[proppy-status]: https://img.shields.io/npm/v/proppy.svg
[proppy-react-status]: https://img.shields.io/npm/v/proppy-react.svg
[proppy-vue-status]: https://img.shields.io/npm/v/proppy-vue.svg
[proppy-preact-status]: https://img.shields.io/npm/v/proppy-preact.svg
[proppy-redux-status]: https://img.shields.io/npm/v/proppy-redux.svg
[proppy-rx-status]: https://img.shields.io/npm/v/proppy-rx.svg
[proppy-frint-react-status]: https://img.shields.io/npm/v/proppy-frint-react.svg

[proppy-package]: https://npmjs.com/package/proppy
[proppy-react-package]: https://npmjs.com/package/proppy-react
[proppy-vue-package]: https://npmjs.com/package/proppy-vue
[proppy-preact-package]: https://npmjs.com/package/proppy-preact
[proppy-redux-package]: https://npmjs.com/package/proppy-redux
[proppy-rx-package]: https://npmjs.com/package/proppy-rx
[proppy-frint-react-package]: https://npmjs.com/package/proppy-frint-react

# Quick start

- [Installation](#installation)
- [Basics](#basics)
- [Factories](#factories)
- [Instances](#instances)
- [Subscription](#subscription)
- [Destroy](#destroy)
- [Dynamic props](#dynamic-props)
- [Composition](#composition)
- [Providers](#providers)
- [Rendering with React](#rendering-with-react)
- [Set providers](#set-providers)
- [Attach factory](#attach-factory)

## Installation

With [`npm`](https://www.npmjs.com/):

```
$ npm install --save proppy
```

## Basics

### Factories

For the simplicity of this example, we can start with `withProps` function from [`proppy`](https://proppyjs.com/docs/packages/proppy) first:

```js
import { withProps } from 'proppy';

const P = withProps({ counter: 1 });
```

### Instances

Now we can get an instance from our factory function by calling it:

```js
const p = P();
```

To access the props synchronously:

```js
console.log(p.props); // { counter: 1 }
```

### Subscription

Given the nature of our instance having static props, if we subscribe to it, it will emit the props only once:

```js
const unsubscribe = p.subscribe(props => console.log(props));
// { counter: 1 }
```

To unsubscribe, just call the returned function:

```js
unsubscribe();
```

### Destroy

To cancel all listeners and clear the internal state, just call:

```js
p.destroy();
```

## Dynamic props

There are times when your props require a bit of interactivity. Imagine your component wants to render the current state of `counter`, but also want to be able to update the value.

We can achieve that using the `withState` function for example:

```js
import { withState } from 'proppy';

// withState(stateName, setterName, initialState)
const P = withState('counter', 'setCounter', 0);

const p = P();
```

Initially, the instance will only show you the default initial state for `counter`:

```js
console.log(p.props);
// { counter: 0, setCounter: Function }
```

But we can update it too:

```js
p.props.setCounter(5);

console.log(p.props);
// { counter: 5, setCounter: Function }
```

If you subscribed to the instance, it will keep emitting the new props every time a change occurs:

```js
p.subscribe(props => console.log(props));
// { counter: 0, setCounter: Function }
// { counter: 5, setCounter: Function }
```

## Composition

Proppy also ships with a handy `compose` function which allows you to compose multiple factories into a single one:

```js
import { compose, withProps, withState } from 'proppy';

const P = compose(
withProps({ foo: 'foo value' }),
withProps({ bar: 'bar value' }),
withState('counter', 'setCounter', 0)
);
```

Once you create an instance, all the props from those factories will be accessible in a single object:

```js
const p = P();

console.log(p.props);
// {
// foo: 'foo value',
// bar: 'bar value',
// counter: 0,
// setCounter: Function
// }
```

## Providers

Providers are application-wide dependencies that all Proppy instances can access anywhere.

They are useful especially when you want to maintain them in a single place, but you want Components from anywhere to access them at will.

Imagine setting your application's name, in a providers object like this:

```js
const myProviders = {
appName: 'My Super Awesome App!'
};
```

Now when composing your props with Proppy, you want to be able to access them:

```js
import { compose, withProps } from 'proppy';

const P = compose(
withProps({ foo: 'foo value' }),

// `withProps` can also generate props using a function
withProps((props, providers) => {
return {
name: providers.appname
};
})
);

// pass `myProviders` when calling the factory
const p = P(myProviders);
```

Your Proppy instance now has these props:

```js
console.log(p.props);
// {
// foo: 'foo value',
// name: 'My Super Awesome App!'
// }
```

## Rendering with React

Proppy integrates with React via [`proppy-react`](https://proppyjs.com/docs/packages/proppy-react) package.

You can install it with `npm` first:

```
$ npm install --save react proppy-react
```

### Set providers

Then we need to set our custom providers in our root React component:

```js
// components/Root.js
import React from 'react';
import { ProppyProvider } from 'proppy-react';

import Counter from './Counter';

const myProviders = {};

export default function Root() {
return (



);
}
```

### Attach factory

Now from anywhere in our components tree, we can use the `attach` higher-order component to connect your Proppy factory to React component:

```js
// components/Counter.js
import React from 'react';
import { attach } from 'proppy-react';
import { compose, withProps, withState } from 'proppy';

const P = compose(
withProps({ foo: 'foo value' }),
withState('counter', 'setCounter', 0),
);

function Counter(props) {
const { foo, counter, setCounter } = props;

return (


Foo: {foo}

Counter: {counter}

setCounter(counter + 1)}>
Increment


);
}

export default attach(P)(Counter);
```

You can do the same with Vue.js and Preact too via [`proppy-vue`](https://proppyjs.com/docs/packages/proppy-vue) and [`proppy-preact`](https://proppyjs.com/docs/packages/proppy-preact) packages respectively.

# Inspiration

Original inspiration for this project has been [`recompose`](https://github.com/acdlite/recompose). If your project is using React, you should definitely check it out.

Learn more about the differences between Proppy and other libraries (like Redux and Recompose) in our [FAQ page](https://proppyjs.com/docs/faq/).

# License

MIT