!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! !!!!!!!!!!!!!!!!!!!! !!!!!!!!!!!!!!!!!!!! !!!!!!!!!!!!!!!!!!!! !!!!!!!!!!!!!!!!!!!!!!!
!!!! Concept code. Not yet working. Greetings to Brillout and aleclarson !!!!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Deeply watches your state-object and props for changes. Re-renders automatically😎 and makes you write less code 😊.
- Performance friendly
React Deepwatch uses proxy facades to watch only for those properties that are actually used in your component function. It doesn't matter how complex and deep the graph behind your state or props is. - Can watch your -model- as well
If a (used) property in props points to your model, a change there will also trigger a re-render. In fact, you can watch anything ;)
npm install --save react-deepwatch
import {watchedComponent, watched, useWatchedState} from "react-deepwatch"
const MyComponent = watchedComponent(props => {
const state = useWatchedState( {myDeep: {counter: 0, b: 2}}, {/* WatchedOptions (optional) */} );
return <div>
Counter is: {state.myDeep.counter}
<button onClick={ () => state.myDeep.counter++ /* will trigger a rerender */ }>Increase counter</button>
</div>
}, {/* WatchedComponentOptions (optional) */});
<MyComponent/> // Use MyComponent
Now that we already have the ability to deeply record our reads, let's see if there's also a way to cut away the boilerplate code for useEffect
.
import {watchedComponent, load, poll, isLoading, loadFailed, preserve} from "react-deepwatch"
const MyComponent = watchedComponent(props => {
return <div>
Here's something fetched from the Server: { load( async () => await myFetchFromServer(props.myProperty), {/* LoadOptions (optional) */} ) }
</div>
});
<MyComponent/> // Use MyComponent
load(...)
re-executes myFetchFromServer
, when a dependent value changes. That means, it records all reads from previous code in your component function plus the reads immediately inside the load(...)
call. Here: props.myProperty.
The returned Promise will be await'ed and the component will be put into suspense that long.
👍 load(...) can be inside a conditional block or a loop. Then it has already recorded the condition + everything else that leads to the computation of load(...)'s point in time and state 😎._
For this mechanic to work, make sure, all sources are watched: props
and load(...)
's result are already automatically watched; For state, use useWatchedState(...)
; For context, use watched(useContext(...))
.
To show a 🌀loading spinner / placeholder during load, either...
- wrap your component in a
<Suspense fallback={<div>🌀</div>}>...<MyComponent/>...</Suspense>
. It can be wrapped at any parent level😎. Or... - call isLoading() inside your component, to probe if any or a certain
load(...)
statement is loading. See jsDoc for usage example. Mind the caveat of not using it for a condition to cut off a load statement. and/or... - specify a fallback value via
load(..., {fallback:"🌀"})
.
either...
- wrap your component in a
<ErrorBoundary fallback={<div>Something went wrong</div>}>...<MyComponent/>...</ErrorBoundary>
from the react-error-boundary package. It can be wrapped at any parent level😎.
It tries to recover from errors and re- runs theloaderFn
, whenever a dependency changes. Note that recovering works only with the mentioned react-error-boundary 4.x and not with 3rd party error-boundary libraries. Or... - try/catch around the load(...) statement. Caveat: You must check, if caught is
instanceof Promise
and re-throw it then. Because this is the way forload
to signal, that things are loading. Or... - call the loadFailed() probing function. This looks more elegant than the above. See jsDoc for usage example.
To reduce the number of expensive myFetchFromServer
calls, try the following:
- Move the load(...) call as upwards in the code as possible, so it depends on fewer props / state / watched objects.
- See the
LoadOptions#fallback
,LoadOptions#silent
andLoadOptions#critical
settings. - Use the
preserve
function on all your fetched data, to smartly ensure non-changing object instances in your app (newFetchResult
===oldFetchResult
; Triple-equals. Also for the deep result_). Changed object instances can either cascade to a lot of re-loads or result in your component still watching the old instance. Think of it like: The preserve function does for your data, what React does for your component tree: It smartly remembers the instances, if needed with the help of an id or key, and re-applies the re-fetched/re-rendered properties to them, so the object-identity/component-state stays the same.
👍load(...)
does callpreserve
by default to enforce this paradigm and give you the best, trouble free experience.
- The component function might return and empty
</>
on the first load and produce a short screen flicker. This is because React's Suspense mechasim is not able to remeber state at that time. To circumvent this, specifyWatchedComponentOptions#fallback
. <Suspense>
and<ErrorBoundary>
inside your component function do not handle/catch loads in that same function. Means: You must place them outside to handle/catch them.- If your app is a mixed scenario with non-watchedComponents and relies on the old way of fully re-rendering the whole tree to pass deep model data (=more than using shallow, primitive props) to the leaves, mind disabling the WatchedComponentOptions#memo flag.
- SSR is not supported.
- startTransition is not supported (has no effect).
- As said: Keep in mind that
load(...)
callspreserve
on its result. It also invalidates (destroys) the "unused" objects. When they're not really unused any you are trying to access them, You'll get the proper error message how to disable it.
Not working on Firefox with Stackblitz currently. Ignore the ever-spinning "Installing dependencies".
You can also use watched
similarly to useWatchedState
to watch any global object. But in React paradigm, this is rather rare, because values are usually passed as props into your component function.
Besides load
, there's also the poll
function, which works similar, but re-loads in regular intervals. See jsDoc
If you like, how this library simplifies things for you and want to write the backend (http) endpoints behind your load(...) statements simply as typescript methods, have a look at my flagship project Restfuncs. Example:
// In your watchedComponent function:
return <div>The greeting's result from server is: { load( async () => await myRemoteSession.greet(state.name) ) }</div>
// On the server:
...
@remote greet(name: string) {
return `Hello ${name}`
}
...
The example leaves away all the the setup-once boilerplate code.
Also in your tsx, you can enjoy type awareness / type safety and IDE's code completion around myRemoteSession.greet
and all its parameters and returned types, which is a feature that only rpc libraries can offer (Restfuncs is such one)😎