Post contents
Components are awesome. They allow you to make your code logic more modular and associate that logic with a related collection of DOM nodes. More importantly than that, components are composable; You can take two components and use them together to build a third that employs them both.
Sometimes, while building components, you may find yourself needing to share logic between multiple components.
We're not talking about sharing state or logic between instances of the same component:
IE: Two instances of the same component sharing the same data.
Instead, we're talking about a way to share logic for each instance of a component.
IE: Two instances of the same component have their own data.
For example, let's say that you have some component code that detects the current window size. While this might seem like a simple problem at first, it requires you to:
- Get the initial window size and share that data with the component
- Add and clean up event listeners for when the user resizes their browser window
- Compose the window sizing logic inside other shared logic, such as an
onlyShowOnMobile
boolean
The method of how this logic is shared between components differs from framework to framework.
Framework | Method Of Logic Sharing |
---|---|
React | Custom Hooks |
Angular | Signal Functions |
Vue | Compositions |
We'll spend the chapter talking about how to do all of this and see how we can apply these methods to production code.
But here's my favorite part about these methods: We don't have to introduce any new APIs to use them. Instead, we'll combine a culmination of other APIs we've learned to this point.
Without further ado, let's build the window size shared logic.
Sharing Data Storage Methods
The first step of creating composable shared logic is to create a way to store data in an instance of the logic:
- React
- Angular
- Vue
In a typical React component, we'd store data using the useState
or useReducer
hook. Using React's custom hooks, we'll use the same APIs to create our own hook that combines (or composes) these other APIs:
const useWindowSize = () => { const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); return { height, width };};
We can then use this useWindowSize
custom hook just as we would any other hook:
const App = () => { const { height, width } = useWindowSize(); return ( <p> The window is {height}px high and {width}px wide </p> );};
Rules of Custom Hooks
In our "Intro to Components" chapter, we covered the rules of React's built-in hooks.
We mentioned that there are a few rules for any hooks:
- Be called from a component* (no normal functions)
- Not be called conditionally inside a component (no
if
statements) - Not be called inside a loop (no
for
orwhile
loops)
While these rules are mostly true, let's expand the first point to include "other hooks" as a place you're allowed to call a hook from.
Now that we've corrected that, it's a good time to mention that custom hooks also follow these rules.
There's one additional rule that custom hooks must follow, and that's:
- Your custom Hook's name must start with
use
.
To recap, this means that the following custom hooks are not allowed:
// ❌ Not allowed, the function name must start with `use`const getWindowSize = () => { const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); return { height, width };};
const useWindowSize = () => { const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); return { height, width };};// ❌ Not allowed, you must use a hook _inside_ a component or another hookconst { height, width } = useWindowSize();const Component = () => { return <p>Height is: {height}</p>;};
const useWindowSize = () => { const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); return { height, width };};function getWindowSize() { // ❌ Not allowed, you cannot use a hook inside a non-hook function const { height, width } = useWindowSize(); return { height, width };}
const useWindowSize = () => { // ❌ Not allowed, you cannot `return` before using a hook if (bool) return { height: 0, width: 0 }; const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); return { height, width };};
Signals enable intense levels of code-sharing between components. Not only can signal
s be used outside of a component, but effects
can be triggered and cleaned up from the calling parent even when wrapped in many levels of function nesting.
What does this look like in practice?
Something like this:
const useWindowSize = () => { const height = signal(window.innerHeight); const width = signal(window.innerWidth); return { height, width };};@Component({ selector: "app-root", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <p> The window is {{ windowSize.height() }}px high and {{ windowSize.width() }}px wide </p> `,})class AppComponent { windowSize = useWindowSize();}
Because Vue's ref
and reactive
data reactivity systems work anywhere, we can extract these values to a dedicated function called useWindowSize
.
// use-window-size.jsimport { ref } from "vue";export const useWindowSize = () => { const height = ref(window.innerHeight); const width = ref(window.innerWidth); return { height, width };};
This custom function is often called a "composition" since we're using Vue's Composition API inside it. We can then use this composition inside our setup script
like so:
<!-- App.vue --><script setup>import { useWindowSize } from "./use-window-size";const { height, width } = useWindowSize();</script><template> <p>The window is {{ height }}px high and {{ width }}px wide</p></template>
While React requires you to name your custom hooks "useX," you don't have to do the same with custom compositions. We could have easily called this code
createWindowSize
and have it work just as well.We still use the
use
composition prefix to keep things readable. While this is subjective, it's the naming convention the ecosystem seems to favor for compositions like this.
Sharing Side Effect Handlers
While sharing data between a consuming component is helpful in its own right, this is only a fraction of the capabilities these frameworks have for cross-component logic reuse.
One of the most powerful things that can be reused between components is side effect logic.
Using this, we can say something along the lines of:
When a component that implements this shared bit of code renders, do this behavior.
And combine it with our data storage to say:
When the component renders, store some calculation and expose it back to the consuming component.
This can be a bit vague to discuss without code, so let's dive in.
While our last code sample was able to expose the browser window's height and width, it didn't respond to window resizing. This means that if you resized the browser window, the value of height
and width
would no longer be accurate.
Let's use the window listener side effect we built in our "Side Effects" chapter to add an event handler to listen for window resizing.
- React
- Angular
- Vue
const useWindowSize = () => { const [height, setHeight] = useState(window.innerHeight); const [width, setWidth] = useState(window.innerWidth); useEffect(() => { function onResize() { setHeight(window.innerHeight); setWidth(window.innerWidth); } window.addEventListener("resize", onResize); // Remember to cleanup the listener return () => window.removeEventListener("resize", onResize); }, []); return { height, width };};
... That's it!
There's nothing more we need to do inside our useWindowSize
consuming component; it simply works transparently as if we had placed the useEffect
in the component itself.
const App = () => { const { height, width } = useWindowSize(); return ( <p> The window is {height}px high and {width}px wide </p> );};
Notice that we've changed exactly zero lines of code from our previous example of this component! ✨ Magic ✨
Now that we know we have access to all of the Signal-based APIs in functions called within a component, we can do powerful things with those APIs.
const useWindow = () => { const height = signal(0); const width = signal(0); const onResize = () => { height.set(window.innerHeight); width.set(window.innerWidth); }; effect((onCleanup) => { height.set(window.innerHeight); width.set(window.innerWidth); window.addEventListener("resize", onResize); onCleanup(() => { window.removeEventListener("resize", onResize); }); }); return { height, width, };};@Component({ selector: "app-root", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <p> The window is {{ windowSize.height() }}px high and {{ windowSize.width() }}px wide </p> `,})class AppComponent { windowSize = useWindow();}
This code isn't ideal; the Angular team knows this. This is why they're working on introducing a new method of side effect handling (and data storage) called "Signals". At the time of writing, Signals are still in the experimental phase, but they're worth keeping an eye on.
While this is the only method we'll be looking at in this book for writing this code, Lars Gyrup Brink Nielsen showcased how we could improve this code using RxJS in another article on the Playful Programming site.
Sharing side effects handling within custom compositions is just as straightforward as using them within components. We can simply use the same onMounted
and onUnmounted
lifecycle methods as we do within our setup
script
.
// use-window-size.jsimport { onMounted, onUnmounted, ref } from "vue";export const useWindowSize = () => { const height = ref(window.innerHeight); const width = ref(window.innerWidth); function onResize() { height.value = window.innerHeight; width.value = window.innerWidth; } onMounted(() => { window.addEventListener("resize", onResize); }); onUnmounted(() => { window.removeEventListener("resize", onResize); }); return { height, width };};
<!-- App.vue --><script setup>import { useWindowSize } from "./use-window-size";const { height, width } = useWindowSize();</script><template> <p>The window is {{ height }}px high and {{ width }}px wide</p></template>
We could have also used the
watch
orwatchEffect
composition methods, but chose not to for this example.
Composing Custom Logic
We've covered how shared logic can access data storage and side-effect handlers. Now let's talk about the fun stuff: Composability.
Not only can you call your custom logic from components, but you can call them from other shared logic fragments.
For example, let's say that we want to take our window size getter and create another custom logic fragment that composes it.
If we were using plain-ole functions, it might look something like this:
function getWindowSize() { return { height: window.innerHeight, width: window.innerWidth, };}function isMobile() { const { height, width } = getWindowSize(); if (width <= 480) return true; else return false;}
But this comes with downsides when trying to include this logic in a framework, such as:
- No access to side effect cleanup
- No automatic-re-rendering when
height
orwidth
changes
Luckily for us, we can do this with our frameworks with full access to all the other framework-specific APIs we've covered until now.
- React
- Angular
- Vue
So, do you remember how we used useState
inside of useWindowSize
? That's because all hooks are composable.
This is true for custom hooks as well, meaning that we can do the following code:
const useMobileCheck = () => { const { width } = useWindowSize(); if (width <= 480) return { isMobile: true }; else return { isMobile: false };};
Without modifying the useWindowSize
component.
To consume our new useMobileCheck
component is just as straightforward as it was to use useWindowSize
:
const Component = () => { const { isMobile } = useMobileCheck(); return <p>Is this a mobile device? {isMobile ? "Yes" : "No"}</p>;};
Because useWindowSize
before is a standard JavaScript function, we can call it from another Signal function to compose them together:
const useWindowSize = () => { const height = signal(0); const width = signal(0); const onResize = () => { height.set(window.innerHeight); width.set(window.innerWidth); }; effect((onCleanup) => { onResize(); window.addEventListener("resize", onResize); onCleanup(() => { window.removeEventListener("resize", onResize); }); }); return { height, width, };};
const useMobileCheck = () => { const windowSize = useWindowSize(); const isMobile = computed(() => this.windowSize.width() <= 480); return { isMobile };};
@Component({ selector: "app-root", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <p>Is mobile? {{ mobileCheck.isMobile() }}</p> `,})class AppComponent { mobileCheck = useMobileCheck();}
Composing custom composables (say that 10 times fast) is a straightforward task, thanks to custom composables acting like normal functions.
// use-mobile-check.jsimport { computed } from "vue";import { useWindowSize } from "./use-window-size";export const useMobileCheck = () => { const { height, width } = useWindowSize(); const isMobile = computed(() => { if (width.value <= 480) return true; else return false; }); return { isMobile };};
Notice that we aren't showing the source code for
useWindowSize
again, that's because we haven't changed it!
Then, to use this new composable in our components, we use it just like we did our previous composables:
<!-- App.vue --><script setup>import { useMobileCheck } from "./use-mobile-check";const { isMobile } = useMobileCheck();</script><template> <p>Is this a mobile device? {{ isMobile ? "Yes" : "No" }}</p></template>
Challenge
Let's take everything we've learned about shared-component logic and use it to recreate our ContextMenu
component from the "Component Reference" chapter in smaller pieces.
Let's break these components into smaller pieces that we'll create composable logic for:
- A listener for clicks outside the context menu
- A composition that gets the bounds of the context menu's parent element
Step 1: Create an Outside Click Composition
To listen for clicks outside the context menu, we can leverage some JavaScript akin to the following:
const closeIfOutsideOfContext = (e) => { const isClickInside = ref.value.contains(e.target); if (isClickInside) return; closeContextMenu();};document.addEventListener("click", closeIfOutsideOfContext);
Let's turn this into a composition that we can use in our ContextMenu
component.
- React
- Angular
- Vue
const useOutsideClick = ({ ref, onClose }) => { useEffect(() => { const closeIfOutsideOfContext = (e) => { const isClickInside = ref.current.contains(e.target); if (isClickInside) return; onClose(); }; document.addEventListener("click", closeIfOutsideOfContext); return () => document.removeEventListener("click", closeIfOutsideOfContext); }, [onClose]);};
Now that we have our useOutsideClick
hook written, we can use it in our ContextMenu
component:
const ContextMenu = forwardRef(({ x, y, onClose }, ref) => { const divRef = useRef(); useImperativeHandle(ref, () => ({ focus: () => divRef.current && divRef.current.focus(), })); useOutsideClick({ ref: divRef, onClose }); return ( <div tabIndex={0} ref={divRef} style={{ position: "fixed", top: y + 20, left: x + 20, background: "white", border: "1px solid black", borderRadius: 16, padding: "1rem", }} > <button onClick={() => onClose()}>X</button> This is a context menu </div> );});
Let's start by moving our "Outside click" behavior into a Signal function:
const useOutsideClick = ( contextMenu: Signal<ElementRef>, onClose: () => void,) => { afterRenderEffect((onCleanup) => { const closeIfOutsideOfContext = (e: MouseEvent) => { const contextMenuEl = contextMenu()?.nativeElement; if (!contextMenuEl) return; const isClickInside = contextMenuEl.contains(e.target as HTMLElement); if (isClickInside) return; onClose(); }; document.addEventListener("click", closeIfOutsideOfContext); onCleanup(() => { document.removeEventListener("click", closeIfOutsideOfContext); }); });};
That we can then use in our ContextMenu
component:
@Component({ selector: "context-menu", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div #contextMenu tabIndex="0" [style]="{ position: 'fixed', top: y() + 20, left: x() + 20, background: 'white', border: '1px solid black', borderRadius: 16, padding: '1rem', }" > <button (click)="close.emit()">X</button> This is a context menu </div> `,})class ContextMenuComponent { contextMenu = viewChild.required("contextMenu", { read: ElementRef<HTMLElement>, }); x = input(0); y = input(0); close = output(); constructor() { useOutsideClick(this.contextMenu, () => this.close.emit()); } focus() { this.contextMenu().nativeElement.focus(); }}
// use-outside-click.jsimport { onMounted, onUnmounted } from "vue";export const useOutsideClick = ({ ref, onClose }) => { const closeIfOutsideOfContext = (e) => { const isClickInside = ref.value.contains(e.target); if (isClickInside) return; onClose(); }; onMounted(() => { document.addEventListener("click", closeIfOutsideOfContext); }); onUnmounted(() => { document.removeEventListener("click", closeIfOutsideOfContext); });};
Then, we can use this composition in our ContextMenu
component:
<!-- ContextMenu.vue --><script setup>import { onMounted, onUnmounted, ref } from "vue";import { useOutsideClick } from "./use-outside-click";const props = defineProps(["x", "y"]);const emit = defineEmits(["close"]);const contextMenuRef = ref(null);useOutsideClick({ ref: contextMenuRef, onClose: () => emit("close") });function focusMenu() { contextMenuRef.value.focus();}defineExpose({ focusMenu,});</script><template> <div tabIndex="0" ref="contextMenuRef" :style="{ position: 'fixed', top: props.y + 20, left: props.x + 20, background: 'white', border: '1px solid black', borderRadius: 16, padding: '1rem', }" > <button @click="$emit('close')">X</button> This is a context menu </div></template>
Step 2: Create a Bounds Composable
Now let's move the bounds' size checking into a composable as well. This is done in JavaScript like so:
const resizeListener = () => { if (!el) return; const localBounds = el.getBoundingClientRect(); setBounds(localBounds);};resizeListener();window.addEventListener("resize", resizeListener);window.removeEventListener("resize", resizeListener);
- React
- Angular
- Vue
const useBounds = () => { const [bounds, setBounds] = useState({ height: 0, width: 0, x: 0, y: 0, }); const [el, setEl] = useState(null); const ref = useCallback((el) => { setEl(el); }, []); useEffect(() => { const resizeListener = () => { if (!el) return; const localBounds = el.getBoundingClientRect(); setBounds(localBounds); }; resizeListener(); window.addEventListener("resize", resizeListener); window.removeEventListener("resize", resizeListener); }, [el]); return { ref, bounds };};
Now we wrap this into our App
root:
function App() { const { ref, bounds } = useBounds(); // An addEventListener is easier to tackle when inside the conditional render // Add that as an exploration for `useImperativeHandle` const [isOpen, setIsOpen] = useState(false); function onContextMenu(e) { e.preventDefault(); setIsOpen(true); } const contextMenuRef = useRef(); useEffect(() => { if (isOpen && contextMenuRef.current) { contextMenuRef.current.focus(); } }, [isOpen]); return ( <> <div style={{ marginTop: "5rem", marginLeft: "5rem" }}> <div ref={ref} onContextMenu={onContextMenu}> Right click on me! </div> </div> {isOpen && ( <ContextMenu x={bounds.x} y={bounds.y} ref={contextMenuRef} onClose={() => setIsOpen(false)} /> )} </> );}
Final code output
Then we wrap this boundary getting logic into it's own function:
const useBounds = (contextOrigin: Signal<ElementRef>) => { const bounds = signal({ height: 0, width: 0, x: 0, y: 0, }); const resizeListener = () => { if (!contextOrigin()) return; bounds.set(contextOrigin().nativeElement.getBoundingClientRect()); }; afterRenderEffect((onCleanup) => { bounds.set(contextOrigin().nativeElement.getBoundingClientRect()); window.addEventListener("resize", resizeListener); onCleanup(() => { window.removeEventListener("resize", resizeListener); }); }); return { bounds };};
And call it into our App
component:
@Component({ selector: "app-root", imports: [ContextMenuComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div [style]="{ marginTop: '5rem', marginLeft: '5rem' }"> <div #contextOrigin (contextmenu)="open($event)">Right click on me!</div> </div> @if (isOpen()) { <context-menu #contextMenu [x]="boundsContext.bounds().x" [y]="boundsContext.bounds().y" (close)="close()" /> } `,})class AppComponent { contextOrigin = viewChild.required("contextOrigin", { read: ElementRef<HTMLElement>, }); contextMenu = viewChildren("contextMenu", { read: ContextMenuComponent }); isOpen = signal(false); boundsContext = useBounds(this.contextOrigin); constructor() { afterRenderEffect(() => { this.contextMenu().forEach(() => { const isLoaded = this?.contextMenu()[0]; if (!isLoaded) return; this.contextMenu()[0].focus(); }); }); } close() { this.isOpen.set(false); } open(e: UIEvent) { e.preventDefault(); this.isOpen.set(true); }}
Final code output
// use-bounds.jsimport { ref, onMounted, onUnmounted } from "vue";export const useBounds = () => { const elRef = ref(); const bounds = ref({ height: 0, width: 0, x: 0, y: 0, }); function resizeListener() { if (!elRef.value) return; bounds.value = elRef.value.getBoundingClientRect(); } onMounted(() => { resizeListener(); window.addEventListener("resize", resizeListener); }); onUnmounted(() => { window.removeEventListener("resize", resizeListener); }); return { bounds, ref: elRef };};
And finally, we'll use this composition in App
:
<!-- App.vue --><script setup>import { onMounted, onUnmounted, ref } from "vue";import ContextMenu from "./ContextMenu.vue";import { useBounds } from "./use-bounds";const isOpen = ref(false);const { ref: contextOrigin, bounds } = useBounds();const contextMenu = ref();function close() { isOpen.value = false;}function open(e) { e.preventDefault(); isOpen.value = true; setTimeout(() => { contextMenu.value.focusMenu(); }, 0);}</script><template> <div :style="{ marginTop: '5rem', marginLeft: '5rem' }"> <div ref="contextOrigin" @contextmenu="open($event)"> Right click on me! </div> </div> <ContextMenu ref="contextMenu" v-if="isOpen" :x="bounds.x" :y="bounds.y" @close="close()" /></template>