Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/typeofweb/react-with-observable

Use Observables with React declaratively!
https://github.com/typeofweb/react-with-observable

jsx observable react reactjs rxjs

Last synced: 19 days ago
JSON representation

Use Observables with React declaratively!

Awesome Lists containing this project

README

        

# react-with-observable
[![buddy pipeline](https://app.buddy.works/mmiszy/react-with-observable/pipelines/pipeline/225575/badge.svg?token=dfc09b1458ffdb6655820015738f0e3bc1a515874804b85799fe6af214f3473c "buddy pipeline")](https://app.buddy.works/mmiszy/react-with-observable/pipelines/pipeline/225575)
[![codecov](https://codecov.io/gh/mmiszy/react-with-observable/branch/master/graph/badge.svg)](https://codecov.io/gh/mmiszy/react-with-observable)
[![cypress dashboard](https://img.shields.io/badge/cypress-dashboard-brightgreen.svg)](https://dashboard.cypress.io/#/projects/dmnv1v/runs)
[![npm](https://img.shields.io/npm/v/react-with-observable.svg)](https://www.npmjs.com/package/react-with-observable)
[![npm bundle size (minified + gzip)](https://img.shields.io/bundlephobia/minzip/react-with-observable.svg)](https://www.npmjs.com/package/react-with-observable)

`react-with-observable`: use Observables declaratively in ⚛️ React!

* ✅ Supports any Observable implementation compatible with ECMAScript Observable (e.g. **RxJS**)!
* ✅ Inspired by the `AsyncPipe` from Angular!
* ✅ Very extensible by composing Observable operators!
* ✅ TypeScript definitions included!

It handles subscribing and unsubscribing automatically and, hence, you don't have to worry about memory leaks or updating state when new values come!

Inspired by the `AsyncPipe` from Angular. Uses React's [`create-subscription`](https://github.com/facebook/react/tree/master/packages/create-subscription) under the hood.

## Install
```javascript
npm install --save react-with-observable create-subscription
```

Get a polyfill for `Symbol.observable` if you need one (you most likely do).

```javascript
npm install --save symbol-observable
```

Remember to `import 'symbol-observable'` **before** `rxjs` or `react-with-observable`!

## Usage
The component supports any Observable library compatible with the [Observables for ECMAScript draft proposal](https://github.com/tc39/proposal-observable).

### Basics
This package exports a single named component `Subscribe`. It expects you to provide an Observable as its only child:

```javascript
const source$ = Observable.of('Hello, world!');
// …
{source$}
```

This results in "Hello, world!" being displayed.

### Reactivity
The component automatically updates whenever a new value is emitted by the Observable:

```javascript
const source$ = Observable.interval(1000);
// …
{source$}
```

As a result, the next integer is displayed every second.

### Operators
You can transform the Observable as you wish, as long as the final result is also an Observable:

```javascript
const source$ = Observable.interval(1000);
// …

{source$.pipe(
map(val => 10 * val),
scan((acc, val) => acc + val, 0),
map(val => )
)}

```

As the result, an `` element is rendered. Its value is changed every second to 0, 10, 30, 60, 100, and so on.

### Initial value
Use your Observable library! `react-with-observable` doesn't implement any custom way to provide the default value and it doesn't need to. For example, with RxJS, you can use the `startWith` operator:

```javascript

{source$.pipe(
startWith(null)
)}

```

## Example
You can find more interactive examples here: https://mmiszy.github.io/react-with-observable/

```javascript
import 'symbol-observable';
import * as React from 'react';
import { Link } from 'react-router-dom';
import { map, startWith } from 'rxjs/operators';
import { Subscribe } from 'react-with-observable';

// myContacts$ is an Observable of an array of contacts

export class ContactsList extends React.Component {
render() {
return (


My Contacts



{myContacts$.pipe(
startWith(null),
map(this.renderList)
)}


);
}

renderList = (contacts) => {
if (!contacts) {
return 'Loading…';
}

if (!contacts.length) {
return 'You have 0 contacts. Add some!';
}

return (


    {contacts.map(contact => (


  • {contact.fullName} — {contact.description}


  • ))}

);
};
}
```

## Bugs? Feature requests?
Feel free to create a new issue: [issues](https://github.com/mmiszy/react-with-observable/issues). Pull requests are also welcome!