Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/sahilrajput03/usewhat


https://github.com/sahilrajput03/usewhat

Last synced: 16 days ago
JSON representation

Awesome Lists containing this project

README

        

# `useWhat` - global state management solution

**Before diving into anything in this package read this valuable question @ [stackoverflow](https://stackoverflow.com/questions/40819992/react-parent-component-re-renders-all-children-even-those-that-havent-changed), and you'll understand mostly how useWhat works underneath(though its very simple already!)**.

MOST ASKED QUESTION 😴 :

**Q. What to use for state management in react ?**

> Use `useWhat`, thats all.

**Believe me, nothing can be better than `useWhat` 😎**

## Installation

```bash
yarn add usewhat
# or
npm i usewhat
```

## Examples

- New feature added: useWhatPersistent.

With `useWhatPersistent` api, you can have `state` to be stored in `localStorage` as well in app memory and state will get updated in `localStorage` as soon as `state` is updated. So when you close the tab or refresh the tab, state will persists in the app as it was in the app. (Support for `async-storage` in react native will follow up in upcoming version.)

```js
import {useWhatPersistent} from 'usewhat';
//...
const App = () => {
const [db, setDb] = useWhatPersistent('db', 'one');
//..
};
```

### Example 1

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/usewhat-example-for-npmjscom-1fopu?file=/src/App.js)
s
Tip: In below example, I have used `initialState` i.e.,

```js
const [home, setHome] = useWhat('home', initialHomeState);
```

, but you may simply use `useWhat` api like

```js
const [home, setHome] = useWhat('home');
```

and that'll set the initial state as `undefined` (this is same ☂️ as using `const [home, setHome] = useState()` in general react).

```js
import React, {useEffect} from 'react';
import './styles.css';
import {useWhat, getWhat} from 'usewhat';

let log = console.log;

const initialHomeState = {rooms: 10};
const initialKitchenState = {cups: 200};

const Pretty = ({data}) =>

{JSON.stringify(data)}
;

export default function App() {
const [home, setHome] = useWhat('home', initialHomeState);
// Initializing `home` namespace with initial data as second parameter.
const [kitchen, setKitchen] = useWhat('kitchen', initialKitchenState);
// Initializing `kitchen` namespace with initial data as second parameter.

return (


App component






);
}

const PARENT_COMPONENT = () => {
// Accessing earlier initialized namespaces i.e., `home` and `kitchen` in `App` component.
const [home, setHome] = getWhat('home');
const [kitchen, setKitchen] = getWhat('kitchen');

const incrementHome = () => setHome({rooms: home.rooms + 1});

const incrementKitchen = () => setKitchen({cups: kitchen.cups + 1});

const incrementHomeBy2 = () =>
setHome((state) => ({
rooms: state.rooms + 2,
}));

const incrementKitchenBy2 = () =>
setKitchen((state) => ({cups: state.cups + 2}));

return (




Parent Component




Increment stateHome
Increment stateKitchen


Increment stateHome by 2
Increment stateKitchen by 2




);
};

const CHILD_COMPONENT = () => {
// Accessing earlier initialized namespaces i.e., `home` and `kitchen` in `App` component.
const [home, _] = getWhat('home');
const [kitchen, __] = getWhat('kitchen');

return (


Child Component




Increment stateHome
Increment stateKitchen


Increment stateHome by 2
Increment stateKitchen by 2

);
};

const incrementHome = () => {
const [home, setHome] = getWhat('home');
setHome({rooms: home.rooms + 1});
};

const incrementKitchen = () => {
const [kitchen, setKitchen] = getWhat('kitchen');
setKitchen({cups: kitchen.cups + 1});
};

const incrementHomeBy2 = () => {
const [_, setHome] = getWhat('home');
setHome((state) => ({
rooms: state.rooms + 2,
}));
};

const incrementKitchenBy2 = () => {
const [_, setKitchen] = getWhat('kitchen');
setKitchen((state) => ({cups: state.cups + 2}));
};
```

### Example 2

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/usewhat-example2-fetching-npmjscom-nkm6c?file=/src/App.js)

```js
import React, {useEffect} from 'react';
import './styles.css';
import axios from 'axios';
import {useWhat, getWhat} from 'usewhat';

let log = console.log;

export default function App() {
const [github, setGithub] = useWhat('gh');
// Initial state of `gh` namespace is set as `undefined` in this case.
useEffect(() => {
axios
.get('https://api.github.com')
.then((res) => setGithub(res.data))
.catch((e) => log('#got error#', e));
}, [setGithub]);

return (




);
}

const ChildComponent = () => {
const [github, setGithub] = getWhat('gh');

return (


Child Component


{JSON.stringify(github, null, 2)}


);
};
```

### Example 3 - What not to do with useWhat? This is exactly what it is.

[**(Beware this is about what not to do with `useWhat`) Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/usewhat-bad-usagenpm-page-exapmle3-f7khu?file=/src/App.js)

```js
import React from 'react';
import './styles.css';
import {useWhat, getWhat} from 'usewhat';

export default function App() {
const [name, setName] = getWhat('person_name'); //initial state = undefined.
// *LEARN: Bad usage of `usewhat`, you must only intialise state
// using `useWhat` api and use `getWhat` api in nested components in
// such component tree.
return (


state: {name ?? 'undefined'}


{
setName('tom');
}}
>
Set name as "tom"





);
}

const ChildComponent = () => {
const [name, setName] = useWhat('person_name', 'jerry');
return (
{
setName('jerry');
}}
>
Set name as 'jerry'

);
};
```

### Example 4 (Correcting example 3)

[**(This is correct version of Example 3) Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/usewhat-bad-usage-fixed-npm-page-exapmle4-x26de?file=/src/App.js:0-921)

```js
import React, {useEffect, useRef} from 'react';
import './styles.css';
import {useWhat, getWhat, log} from 'usewhat';

export default function App() {
const [name, setName] = useWhat('person_name'); //initial state = undefined.
// *LEARN: Correct way of using useWhat.
return (


state: {name ?? 'undefined'}


{
setName('tom');
}}
>
Set name as "tom"





);
}

const ChildComponent = () => {
const [name, setName] = getWhat('person_name');

useEffect(() => {
setName('jerry');
// *LEARN: Correct way initializing state in a child component.
}, [setName]);

return (
{
setName('jerry');
}}
>
Set name as 'jerry'

);
};
```

### Example 5 Bad usage of useWhat

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/example-4-usewhat-npmjs-package-readme-lpt9y?file=/src/App.js:83-126)

```js
import React from 'react';
import {useWhat} from 'usewhat';
// THIS IS AN ANTIPATTERN CASE FOR USEWHAT
// You should note that count namespace is initialized twice
// as Child component App compoment and its a antipatter to
// usage of useWhat.

// SOLUTION(Refer Example 6): You should host `count` namespace in
// App component using `useWhat('count', 1)` and utilise `count` in
// nested components using getWhat api.

function App() {
return (





);
}

const Child = () => {
const [count, setCount] = useWhat('count', 1);

return (
{
setCount(count + 1);
}}
>
{count}

);
};

export default App;
```

### Example 6 (Fix for e.g. 5)

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/fix-for-eg-5-usewhat-npm-packagereadme-wfj0p?file=/src/App.js)

```js
import './styles.css';
import {useWhat, getWhat} from 'usewhat';

const Child = () => {
const [count, setCount] = getWhat('count');

return (
{
setCount(count + 1);
}}
>
{count}

);
};

export default function App() {
useWhat('count', 1); // hoisting the state!
return (
<>


>
);
}
```

### Example 7 Deeply nesting components

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/deep-nested-comp-case-usewhat-npm-package-exampleto-be-added-to-readme-puiwo)

```js
// Code is large to display here, refer above codesnadbox link please!!
```

### Example 8 If you need performace optimization, you can use [memo](https://reactjs.org/docs/react-api.html#reactmemo).

- Also, I would suggest if you should optimise your components at all or not, by reading [@stackoverflow question](https://stackoverflow.com/questions/53074551/when-should-you-not-use-react-memo) and '#reactmemo-and-friends: When to useMemo and useCallback' article from [kentcdodds](https://kentcdodds.com/blog/usememo-and-usecallback#reactmemo-and-friends).

[**Click here to see this example in codesandbox 🔥**](https://codesandbox.io/s/optimization-and-usewhatnpm-package-eg-7-81yxv?file=/src/App.js)

```js
// You can safely remove all the log statements, they are just for learning phase only.
import {useEffect, memo} from 'react';
import './styles.css';
import {useWhat, getWhat} from 'usewhat';
// You can use wrap any component(FOR OPTIMIZATION) to use last rendered result if its props
// has not changed, yikes!!
// Read at react docs: https://reactjs.org/docs/react-api.html#reactmemo
let log = console.log;

export default function App() {
const [count, setCount] = useWhat('count', 1);
const [label, setLabel] = useWhat('label', 'Optimized component...');

return (


Hello CodeSandbox


setCount(count + 1)}> incr
setLabel(Math.random)}> change label

{/* */} {/* ← ← ← ← ← ← non-optimized way */}


);
}

const Child1 = () => {
const [count, setCount] = getWhat('count');
useEffect(() => {
log('Child 1 updated(re-rendered!)');
});
return

Child1 - Count: {count}
;
};

const Child2_Optimized = memo(Child2);

function Child2({label}) {
const [count, setCount] = getWhat('count');
useEffect(() => {
log('Child 2 updated(re-rendered!)');
});
return (


Child2 - Count: {count} {label ?? label}


);
}

const Child3_Optimized = memo(Child3);

function Child3() {
const [count, setCount] = getWhat('count');
useEffect(() => {
log('Child 3 updated(re-rendered!)');
});
return

Child3 - Count: {count}
;
}
```

## Why use `useWhat` ?

- Only two apis: useWhat and getWhat.
- No need to pass props down ever
- `getWhat` can be called in handler functions as well unlike useState requires you top-level declaration (eslint shit)
- `getWhat` can be called outside react components, thus fuction callbacks get more neat( as they don't need state to be passed as params, coz they can have their own getWhat inside them )
- No clutter for maintaing big component trees.
- Also works for react-native

**Show your support by star the [repo](https://github.com/sahilrajput03/usewhat)**.

Thanks, for being here.

~Sahil Rajput