From 487dc4e7b32ed5eec192df639b1d54c47fdd590a Mon Sep 17 00:00:00 2001 From: Carlos Baraza Haro Date: Mon, 19 Sep 2022 12:54:23 +0100 Subject: [PATCH] feat: initial implementation Initial implementation of useLazyState BREAKING CHANGE: Initial implementation --- README.md | 103 ++++++++++++++++++++++++++++++--------------- package-lock.json | 59 ++++++++++++++++++++++++++ package.json | 1 + src/index.ts | 77 ++++++++++++++++++++++++++++++++- test/index.spec.ts | 12 ++---- 5 files changed, 208 insertions(+), 44 deletions(-) diff --git a/README.md b/README.md index 6b10a7a..c84c12f 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -# my-package-name +# use-lazy-state [![npm package][npm-img]][npm-url] [![Build Status][build-img]][build-url] @@ -8,55 +8,90 @@ [![Commitizen Friendly][commitizen-img]][commitizen-url] [![Semantic Release][semantic-release-img]][semantic-release-url] -> My awesome module +> Simple `useLazyState` hook. Similar to `useState` but each component using the state needs to opt-in the state changes. This is very useful to prevent re-rendering an entire tree when only a child should be actually re-rendered. ## Install ```bash -npm install my-package-name +npm install use-lazy-state ``` ## Usage -```ts -import { myPackage } from 'my-package-name'; +Create a state in the parent. Changes to the array will not trigger a re-render in the parent, because it is not using `useState`: -myPackage('hello'); -//=> 'hello from my package' -``` - -## API +```tsx +import { useLazyState } from 'use-lazy-state'; -### myPackage(input, options?) +const Parent = () => { + const numberList = useLazyState([0, 0, 0]); -#### input + return ( +
+ + + +
+ ); +}; +``` -Type: `string` +Opt-in the state changes with `state.useState`. Re-renders will only trigger when the state returned from the getter has changed (uses `===` to check changes): + +```tsx +type Props = { + numberList: UseLazyState; + index: number; +}; + +const Child = ({ numberList, index }: Props) => { + const n = numberList.useState(s => s[index]); + + const addOne = () => { + numberList.setState(prev => { + prev[index] = prev[index] + 1; + return [...prev]; + }); + }; + + return ( +
+ Number: {n} + +
+ ); +}; +``` -Lorem ipsum. +The `Child` component would only re-render when the state returned from your getter actually changes. -#### options +## Demo -Type: `object` +- [CodeSandbox demo](https://codesandbox.io/s/uselazystate-5ti537?file=/src/useLazyState.ts) -##### postfix +## API -Type: `string` -Default: `rainbows` +``` +export type UseLazyState = { + _stateRef: any; + setState: (action: T | UseLazyStateSetter) => void; + useState: (getter?: (state: T) => D) => D; +}; -Lorem ipsum. +export type UseLazyStateSetter = (prev: T) => T; +``` -[build-img]:https://github.com/carlosbaraza/use-lazy-state/actions/workflows/release.yml/badge.svg -[build-url]:https://github.com/carlosbaraza/use-lazy-state/actions/workflows/release.yml -[downloads-img]:https://img.shields.io/npm/dt/use-lazy-state -[downloads-url]:https://www.npmtrends.com/use-lazy-state -[npm-img]:https://img.shields.io/npm/v/use-lazy-state -[npm-url]:https://www.npmjs.com/package/use-lazy-state -[issues-img]:https://img.shields.io/github/issues/carlosbaraza/use-lazy-state -[issues-url]:https://github.com/carlosbaraza/use-lazy-state/issues -[codecov-img]:https://codecov.io/gh/carlosbaraza/use-lazy-state/branch/main/graph/badge.svg -[codecov-url]:https://codecov.io/gh/carlosbaraza/use-lazy-state -[semantic-release-img]:https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg -[semantic-release-url]:https://github.com/semantic-release/semantic-release -[commitizen-img]:https://img.shields.io/badge/commitizen-friendly-brightgreen.svg -[commitizen-url]:http://commitizen.github.io/cz-cli/ +[build-img]: https://github.com/carlosbaraza/use-lazy-state/actions/workflows/release.yml/badge.svg +[build-url]: https://github.com/carlosbaraza/use-lazy-state/actions/workflows/release.yml +[downloads-img]: https://img.shields.io/npm/dt/use-lazy-state +[downloads-url]: https://www.npmtrends.com/use-lazy-state +[npm-img]: https://img.shields.io/npm/v/use-lazy-state +[npm-url]: https://www.npmjs.com/package/use-lazy-state +[issues-img]: https://img.shields.io/github/issues/carlosbaraza/use-lazy-state +[issues-url]: https://github.com/carlosbaraza/use-lazy-state/issues +[codecov-img]: https://codecov.io/gh/carlosbaraza/use-lazy-state/branch/main/graph/badge.svg +[codecov-url]: https://codecov.io/gh/carlosbaraza/use-lazy-state +[semantic-release-img]: https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg +[semantic-release-url]: https://github.com/semantic-release/semantic-release +[commitizen-img]: https://img.shields.io/badge/commitizen-friendly-brightgreen.svg +[commitizen-url]: http://commitizen.github.io/cz-cli/ diff --git a/package-lock.json b/package-lock.json index aa79dcb..ea0041f 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13,6 +13,7 @@ "@ryansonshine/cz-conventional-changelog": "^3.3.4", "@types/jest": "^27.5.2", "@types/node": "^12.20.11", + "@types/react": "^18.0.20", "@typescript-eslint/eslint-plugin": "^4.22.0", "@typescript-eslint/parser": "^4.22.0", "conventional-changelog-conventionalcommits": "^5.0.0", @@ -2168,12 +2169,35 @@ "integrity": "sha512-eI5Yrz3Qv4KPUa/nSIAi0h+qX0XyewOliug5F2QAtuRg6Kjg6jfmxe1GIwoIRhZspD1A0RP8ANrPwvEXXtRFog==", "dev": true }, + "node_modules/@types/prop-types": { + "version": "15.7.5", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", + "integrity": "sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==", + "dev": true + }, + "node_modules/@types/react": { + "version": "18.0.20", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.0.20.tgz", + "integrity": "sha512-MWul1teSPxujEHVwZl4a5HxQ9vVNsjTchVA+xRqv/VYGCuKGAU6UhfrTdF5aBefwD1BHUD8i/zq+O/vyCm/FrA==", + "dev": true, + "dependencies": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + }, "node_modules/@types/retry": { "version": "0.12.1", "resolved": "https://registry.npmjs.org/@types/retry/-/retry-0.12.1.tgz", "integrity": "sha512-xoDlM2S4ortawSWORYqsdU+2rxdh4LRW9ytc3zmT37RIKQh6IHyKwwtKhKis9ah8ol07DCkZxPt8BBvPjC6v4g==", "dev": true }, + "node_modules/@types/scheduler": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.2.tgz", + "integrity": "sha512-hppQEBDmlwhFAXKJX2KnWLYu5yMfi91yazPb2l+lbJiwW+wdo1gNeRA+3RgNSO39WYX2euey41KEwnqesU2Jew==", + "dev": true + }, "node_modules/@types/stack-utils": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", @@ -3384,6 +3408,12 @@ "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", "dev": true }, + "node_modules/csstype": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.1.tgz", + "integrity": "sha512-DJR/VvkAvSZW9bTouZue2sSxDwdTN92uHjqeKVm+0dAqdfNykRzQ95tay8aXMBAAPpUiq4Qcug2L7neoRh2Egw==", + "dev": true + }, "node_modules/data-urls": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", @@ -14065,12 +14095,35 @@ "integrity": "sha512-eI5Yrz3Qv4KPUa/nSIAi0h+qX0XyewOliug5F2QAtuRg6Kjg6jfmxe1GIwoIRhZspD1A0RP8ANrPwvEXXtRFog==", "dev": true }, + "@types/prop-types": { + "version": "15.7.5", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", + "integrity": "sha512-JCB8C6SnDoQf0cNycqd/35A7MjcnK+ZTqE7judS6o7utxUCg6imJg3QK2qzHKszlTjcj2cn+NwMB2i96ubpj7w==", + "dev": true + }, + "@types/react": { + "version": "18.0.20", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.0.20.tgz", + "integrity": "sha512-MWul1teSPxujEHVwZl4a5HxQ9vVNsjTchVA+xRqv/VYGCuKGAU6UhfrTdF5aBefwD1BHUD8i/zq+O/vyCm/FrA==", + "dev": true, + "requires": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + }, "@types/retry": { "version": "0.12.1", "resolved": "https://registry.npmjs.org/@types/retry/-/retry-0.12.1.tgz", "integrity": "sha512-xoDlM2S4ortawSWORYqsdU+2rxdh4LRW9ytc3zmT37RIKQh6IHyKwwtKhKis9ah8ol07DCkZxPt8BBvPjC6v4g==", "dev": true }, + "@types/scheduler": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.2.tgz", + "integrity": "sha512-hppQEBDmlwhFAXKJX2KnWLYu5yMfi91yazPb2l+lbJiwW+wdo1gNeRA+3RgNSO39WYX2euey41KEwnqesU2Jew==", + "dev": true + }, "@types/stack-utils": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", @@ -14957,6 +15010,12 @@ } } }, + "csstype": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.1.tgz", + "integrity": "sha512-DJR/VvkAvSZW9bTouZue2sSxDwdTN92uHjqeKVm+0dAqdfNykRzQ95tay8aXMBAAPpUiq4Qcug2L7neoRh2Egw==", + "dev": true + }, "data-urls": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", diff --git a/package.json b/package.json index a490cfd..99bf635 100644 --- a/package.json +++ b/package.json @@ -46,6 +46,7 @@ "@ryansonshine/cz-conventional-changelog": "^3.3.4", "@types/jest": "^27.5.2", "@types/node": "^12.20.11", + "@types/react": "^18.0.20", "@typescript-eslint/eslint-plugin": "^4.22.0", "@typescript-eslint/parser": "^4.22.0", "conventional-changelog-conventionalcommits": "^5.0.0", diff --git a/src/index.ts b/src/index.ts index 0349cbc..299045e 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1 +1,76 @@ -export const myPackage = (taco = ''): string => `${taco} from my package`; +import { useEffect, useRef, useState } from 'react'; + +export type UseLazyStateSetter = (prev: T) => T; +export type UseLazyState = { + _stateRef: any; + setState: (action: T | UseLazyStateSetter) => void; + useState: (getter?: (state: T) => D) => D; +}; + +type Subscriber = { + id: number; + triggerUpdate: () => void; +}; + +export const useLazyState = (initialState: T): UseLazyState => { + const subscriberId = useRef(0); + const ref = useRef(initialState); + + const subscribers = useRef([]); + + const setState: UseLazyState['setState'] = action => { + if (typeof action === 'function') { + ref.current = (action as UseLazyStateSetter)(ref.current); + } else { + const state = action; + ref.current = state; + } + for (const subscriber of subscribers.current) { + subscriber.triggerUpdate(); + } + }; + + const _useState = function _useState(_getter?: (state: T) => D): D { + const getter = + typeof _getter === 'undefined' ? (state: T): T => state : _getter; + + if (typeof getter !== 'function') + throw new Error('Getter must be a function'); + + const innerInitialState = getter(initialState); + + // stateRef to maintain a reference to an object that triggerUpdate could point to + const currentStateRef = useRef(innerInitialState); + + // useState to trigger react rendering when it is updated + const [currentState, setCurrentState] = useState(innerInitialState); + + // Subscribe the hook to get global updates when changes happen in the state + useEffect(() => { + const id = subscriberId.current; + subscribers.current.push({ + id, + triggerUpdate: () => { + const newState = getter(ref.current); + const currentState = currentStateRef.current; + if (currentState && newState === currentState) return; // no need to update state + // update state + currentStateRef.current = newState; + setCurrentState(newState); + }, + }); + subscriberId.current = subscriberId.current + 1; + return () => { + subscribers.current = subscribers.current.filter(s => s.id !== id); + }; + }, []); + + return currentState as D; + }; + + return { + _stateRef: ref, + setState, + useState: _useState, + }; +}; diff --git a/test/index.spec.ts b/test/index.spec.ts index d305240..447f42a 100644 --- a/test/index.spec.ts +++ b/test/index.spec.ts @@ -1,13 +1,7 @@ -import { myPackage } from '../src'; - describe('index', () => { - describe('myPackage', () => { - it('should return a string containing the message', () => { - const message = 'Hello'; - - const result = myPackage(message); - - expect(result).toMatch(message); + describe('useLazyState', () => { + it('true === true', () => { + expect(true).toBe(true); }); }); });