An Engine for DOM Recombobulation.
import { createRoot, useState } from 'endr';
const Root = () => {
const [count, setCount] = useState(0);
return (
<div>
<button onclick={() => setCount(count + 1)}>This button</button>{' '}
has been clicked {count} {count === 1 ? 'time' : 'times'}
</div>
);
};
createRoot(document.body).render(<Root />);
Endr takes the best parts of React and ditches the rest to result in a lean and fast virtual DOM rendering library. React is a great tool but has acquired some baggage over its long career that it must maintain for backwards compatibility. Endr loses the baggage and keeps the modern API.
Package | Size |
---|---|
Endr | |
Preact Preact Hooks |
|
React React DOM |
Use jsx: automatic
and jsxImportSource: 'endr'
in your tsconfig.json
and
JSX transpiler (babel
, esbuild
, etc) to get correct autocomplete and
rendered output. If you forget to do this you'll see errors about React
not
being found.
- There are no class components.
- There are no synthetic events.
- There is no property redirection.
- All element properties should be passed as if you were setting them directly
on the element. For example
onclick
instead ofonClick
andondblclick
instead ofonDoubleClick
.
- All element properties should be passed as if you were setting them directly
on the element. For example
ref
is not a special property on function components (it is passed through without something likeForwardRef
).useRef
accepts an initializer function.createContext
returns the equivalent of a ReactContext.Provider
component.- There is no
Context.Consumer
component. Access context values throughuseContext(Context)
.
- There is no
useCallback
does not take a second argument and will return a constant function that will call the last seen function passed touseCallback
. The less common case of memoizing a callback that creates a new function when dependencies change can be achieved withconst sumAB = useMemo(() => () => a + b, [a, b])
.useMemo
can be called without a second argument to default to an empty dependency array.setState
returns the most recently set value.setState
will not queue a re-render when it is called during the render function.- There is no
useLayoutEffect
. useEffect
is called immediately after the DOM is reconciled.- Portals can be used with the
Portal
component instead ofcreatePortal
.is equivalent to React's<Portal to={parentElement}><div /></Portal>
createPortal(<div />, parentElement)
- Exceptions thrown during render can be caught by the nearest Try component.
can be used, for example, to show an error message whenever rendering
const MyComponent () => { const [error, setError] = useState(undefined); if (error) return `An error occurred! ${error}`; return ( <Try catch={setError}> <AllMyChildren /> </Try> ); };
<AllMyChildren />
or any descendents throws an exception.- React's
Suspense
can be recreated withTry
by awaiting all thrown promises, if desired.
- React's