• Stars
    star
    2,518
  • Rank 18,215 (Top 0.4 %)
  • Language
    TypeScript
  • License
    MIT License
  • Created over 5 years ago
  • Updated 6 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

React useContextSelector hook in userland

use-context-selector

CI npm size discord

React useContextSelector hook in userland

Introduction

React Context and useContext is often used to avoid prop drilling, however it's known that there's a performance issue. When a context value is changed, all components that useContext will re-render.

To solve this issue, useContextSelector is proposed and later proposed Speculative Mode with context selector support. This library provides the API in userland.

Prior to v1.3, it uses changedBits=0 feature to stop propagation, v1.3 no longer depends on this undocumented feature.

Install

This package requires some peer dependencies, which you need to install by yourself.

yarn add use-context-selector react scheduler

Notes for library authors:

Please do not forget to keep "peerDependencies" and note instructions to let users to install peer dependencies.

Technical memo

To make it work like original React context, it uses useReducer cheat mode intentionally.

It also requires useContextUpdate to behave better in concurrent rendering. Its usage is optional and only required if the default behavior is unexpected.

Usage

import React, { useState } from 'react';
import ReactDOM from 'react-dom';

import { createContext, useContextSelector } from 'use-context-selector';

const context = createContext(null);

const Counter1 = () => {
  const count1 = useContextSelector(context, v => v[0].count1);
  const setState = useContextSelector(context, v => v[1]);
  const increment = () => setState(s => ({
    ...s,
    count1: s.count1 + 1,
  }));
  return (
    <div>
      <span>Count1: {count1}</span>
      <button type="button" onClick={increment}>+1</button>
      {Math.random()}
    </div>
  );
};

const Counter2 = () => {
  const count2 = useContextSelector(context, v => v[0].count2);
  const setState = useContextSelector(context, v => v[1]);
  const increment = () => setState(s => ({
    ...s,
    count2: s.count2 + 1,
  }));
  return (
    <div>
      <span>Count2: {count2}</span>
      <button type="button" onClick={increment}>+1</button>
      {Math.random()}
    </div>
  );
};

const StateProvider = ({ children }) => (
  <context.Provider value={useState({ count1: 0, count2: 0 })}>
    {children}
  </context.Provider>
);

const App = () => (
  <StateProvider>
    <Counter1 />
    <Counter2 />
  </StateProvider>
);

ReactDOM.render(<App />, document.getElementById('app'));

API

createContext

This creates a special context for useContextSelector.

Parameters

  • defaultValue Value

Examples

import { createContext } from 'use-context-selector';

const PersonContext = createContext({ firstName: '', familyName: '' });

useContextSelector

This hook returns context selected value by selector.

It will only accept context created by createContext. It will trigger re-render if only the selected value is referentially changed.

The selector should return referentially equal result for same input for better performance.

Parameters

  • context Context<Value>
  • selector function (value: Value): Selected

Examples

import { useContextSelector } from 'use-context-selector';

const firstName = useContextSelector(PersonContext, state => state.firstName);

useContext

This hook returns the entire context value. Use this instead of React.useContext for consistent behavior.

Parameters

  • context Context<Value>

Examples

import { useContext } from 'use-context-selector';

const person = useContext(PersonContext);

useContextUpdate

This hook returns an update function that accepts a thunk function

Use this for a function that will change a value in concurrent rendering in React 18. Otherwise, there's no need to use this hook.

Parameters

  • context Context<Value>

Examples

import { useContextUpdate } from 'use-context-selector';

const update = useContextUpdate();

// Wrap set state function
update(() => setState(...));

// Experimental suspense mode
update(() => setState(...), { suspense: true });

BridgeProvider

This is a Provider component for bridging multiple react roots

Parameters

  • $0 {context: Context<any>, value: any, children: ReactNode}

    • $0.context
    • $0.value
    • $0.children

Examples

const valueToBridge = useBridgeValue(PersonContext);
return (
  <Renderer>
    <BridgeProvider context={PersonContext} value={valueToBridge}>
      {children}
    </BridgeProvider>
  </Renderer>
);

useBridgeValue

This hook return a value for BridgeProvider

Parameters

  • context Context<any>

Limitations

  • In order to stop propagation, children of a context provider has to be either created outside of the provider or memoized with React.memo.
  • Provider trigger re-renders only if the context value is referentially changed.
  • Neither context consumers or class components are supported.
  • The stale props issue can't be solved in userland.
  • Tearing is only avoided if all consumers get data using useContextSelector. If you use both props and use-context-selector to pass the same data, they may provide inconsistence data for a brief moment. (02_tearing_spec fails)

Examples

The examples folder contains working examples. You can run one of them with

PORT=8080 yarn run examples:01_minimal

and open http://localhost:8080 in your web browser.

You can also try them in codesandbox.io: 01 02 03

Projects that use use-context-selector

More Repositories

1

waku

⛩️ The minimal React framework
TypeScript
4,363
star
2

react-tracked

State usage tracking with Proxies. Optimize re-renders for useState/useReducer, React Redux, Zustand and others.
TypeScript
2,663
star
3

excalidraw-animate

A tool to animate Excalidraw drawings
TypeScript
1,252
star
4

react-hooks-global-state

[NOT MAINTAINED] Simple global state for React with Hooks API without Context API
TypeScript
1,099
star
5

proxy-memoize

Intuitive magical memoization library with Proxy and WeakMap
TypeScript
715
star
6

react-hooks-worker

React custom hooks for web workers
TypeScript
698
star
7

will-this-react-global-state-work-in-concurrent-rendering

Test tearing and branching in React concurrent rendering
JavaScript
546
star
8

reactive-react-redux

[NOT MAINTAINED] React Redux binding with React Hooks and Proxy
TypeScript
502
star
9

react-hooks-async

[NOT MAINTAINED] React custom hooks for async functions with abortability and composability
JavaScript
496
star
10

excalidraw-claymate

A tool based on Excalidraw to create stop motion animations and slides.
TypeScript
482
star
11

react-hooks-fetch

[NOT MAINTAINED] Minimal data fetching library with React Suspense
TypeScript
397
star
12

react-worker-components

React Worker Components simplify using Web Workers
TypeScript
337
star
13

use-signals

An experimental React hook for TC39 signals
TypeScript
309
star
14

react-suspense-fetch

[NOT MAINTAINED] A low-level library for React Suspense for Data Fetching
TypeScript
297
star
15

katachidraw

SVG based drawing tool and react-native component
TypeScript
290
star
16

proxy-compare

Compare two objects using accessed properties with Proxy
TypeScript
268
star
17

redux-in-worker

Entire Redux in Web Worker
TypeScript
258
star
18

social-cms-backend

Express middleware to provide schema-less REST APIs for creating a social networking website primarily using angular.js. It comes with built-in authentication, authorization and notification features.
JavaScript
216
star
19

use-reducer-async

React useReducer with async actions
TypeScript
178
star
20

valtio-yjs

valtio-yjs makes yjs state easy
TypeScript
177
star
21

connect-prerenderer

Express/connect middleware to pre-render ajax page for non-ajax browsers, especially using angular.js
JavaScript
161
star
22

typescript-expo-apollo-boilerplate

[NOT MAINTAINED] Clean boilerplate for TypeScript + Expo (React Native) + React Apollo (GraphQL)
TypeScript
144
star
23

es-beautifier

[NOT MAINTAINED] ECMAScript beautifier based on eslint
JavaScript
120
star
24

create-react-signals

A factory function to create signals for React
TypeScript
111
star
25

zustand-signal

Another React binding for Zustand
TypeScript
109
star
26

react18-use

React 19 use hook shim
TypeScript
109
star
27

use-atom

Yet another implementation for Jotai atoms without side effects
TypeScript
108
star
28

remote-faces

A tool for Working From Home: Share your webcam images with your colleagues
JavaScript
95
star
29

react-suspense-router

[NOT MAINTAINED] React Router for React Suspense and Render-as-You-Fetch
TypeScript
87
star
30

continuation.js

A module for tail call optimization by Continuation Passing Style (CPS) transformation with trampoline technique for Node.js
JavaScript
78
star
31

rss-pipes

RSS feed aggregator by Node.js
JavaScript
74
star
32

derive-zustand

A function to create a derived Zustand store from stores
TypeScript
71
star
33

lets-compare-global-state-with-react-hooks

Comparing global state libraries with React Hooks
59
star
34

use-zustand

Another custom hook to use Zustand vanilla store
TypeScript
57
star
35

easy-livereload

Express middleware to use livereload2 easily (both server and client)
JavaScript
56
star
36

codeonmobile

A coding tool on mobile devices targeting GitHub/Codeship/Heroku
JavaScript
55
star
37

valtio-signal

Another React binding for Valtio proxy state
TypeScript
50
star
38

connect-cache-manifest

Express/connect middleware to generate HTML5 cache manifest file.
JavaScript
49
star
39

excalidraw-gallery

A tool to display Excalidraw drawings
JavaScript
37
star
40

excalidraw-layers

A tool to view layers of Excalidraw drawings
TypeScript
33
star
41

notes-app-sample

A sample HTML5 web app using social-cms-backend (the BMEAN stack)
JavaScript
33
star
42

express-react-redux

Express middleware for React/Redux applications
JavaScript
30
star
43

svelte3-redux

[NOT MAINTAINED] Redux for Svelte 3
TypeScript
30
star
44

react-suspense-worker

[NOT MAINTAINED] React Suspense for Web Worker with Comlink
TypeScript
26
star
45

react-native-dom-expo

[NOT MAINTAINED] A patch library to make Expo work with react-native-dom
JavaScript
25
star
46

react-context-global-state

[NOT MAINTAINED] Simple global state for React with Context API
JavaScript
23
star
47

gqless-hook

[NOT MAINTAINED] Yet another React hook for gqless
TypeScript
22
star
48

use-valtio

Another custom hook to use Valtio proxy state
TypeScript
21
star
49

react-hooks-render-props

[NOT MAINTAINED] A hacking custom hook to emulate render props
JavaScript
18
star
50

benchmark-octane

Octane benchmark for Node.js
JavaScript
18
star
51

recoildux

[NOT MAINTAINED] Recoil inspired implementation with Redux
TypeScript
16
star
52

react-compose-state

A helper function to attach state to stateless function components
JavaScript
16
star
53

gunosy-rss

[OBSOLETE] Gunosy RSS feed web service by Node.js
JavaScript
15
star
54

apollo-link-lazy

Apollo Link for lazy loading
JavaScript
14
star
55

react-suspense-router-demo

[NOT MAINTAINED] A demo app with react-suspense-router
TypeScript
11
star
56

react-apollo-github-api-infinite-scroll-example

React Apollo GitHub GraphQL API infinite scroll example code
JavaScript
11
star
57

twitter-clone-sample

JavaScript
7
star
58

blog

Personal blog
Less
6
star
59

waku-vercel-starter

TypeScript
5
star
60

meteor-fan

HTML
4
star
61

waku-netlify-starter

TypeScript
4
star
62

dai-shi

3
star
63

minimal-apollo-example

Minimal Apollo stack example code
JavaScript
2
star
64

react-compose-onmount

A helper function to attach onmount handler to stateless function components
JavaScript
2
star
65

GBrain

A Google AppEngine web app for sharing brains.
Java
1
star
66

meteor-google-maps-example

JavaScript
1
star
67

meteor-blaze-google-maps

Easy Blaze template for Google Maps with reactivity
JavaScript
1
star
68

meteor-blaze-showhide

Handy block helpers for show/hide functionality in Blaze
JavaScript
1
star
69

meteor-facebook-server-api

A Meteor package to provide Facebook Graph API
JavaScript
1
star