Post contents
In our previous chapter, we built context menu functionality into our App
component. This functionality allowed us to right-click on an element and get a list of actions we could take.
This code works as we'd expect, but it doesn't follow a fundamental pattern of React, Angular, or Vue: It's not componentized.
Let's fix this by moving our context menu code into its own component. This way, we're able to do refactors more easily, code cleanup, and more.
- React
- Angular
- Vue
const ContextMenu = ({ isOpen, x, y, onClose }) => { const [contextMenu, setContextMenu] = useState(); useEffect(() => { if (!contextMenu) return; const closeIfOutsideOfContext = (e) => { const isClickInside = contextMenu.contains(e.target); if (isClickInside) return; onClose(false); }; document.addEventListener("click", closeIfOutsideOfContext); return () => document.removeEventListener("click", closeIfOutsideOfContext); }, [contextMenu]); if (!isOpen) return null; return ( <div ref={(el) => setContextMenu(el)} tabIndex={0} style={{ position: "fixed", top: y, left: x, background: "white", border: "1px solid black", borderRadius: 16, padding: "1rem", }} > <button onClick={() => onClose()}>X</button> This is a context menu </div> );};function App() { const [mouseBounds, setMouseBounds] = useState({ x: 0, y: 0, }); const [isOpen, setIsOpen] = useState(false); function onContextMenu(e) { e.preventDefault(); setIsOpen(true); setMouseBounds({ x: e.clientX, y: e.clientY, }); } return ( <> <div style={{ marginTop: "5rem", marginLeft: "5rem" }}> <div onContextMenu={onContextMenu}>Right click on me!</div> </div> <ContextMenu isOpen={isOpen} onClose={() => setIsOpen(false)} x={mouseBounds.x} y={mouseBounds.y} /> </> );}
@Component({ selector: "context-menu", changeDetection: ChangeDetectionStrategy.OnPush, template: ` @if (isOpen()) { <div tabIndex="0" #contextMenu style=" position: fixed; top: {{ y() }}px; left: {{ x() }}px; background: white; border: 1px solid black; border-radius: 16px; padding: 1rem; " > <button (click)="close.emit()">X</button> This is a context menu </div> } `,})class ContextMenuComponent { contextMenu = viewChild("contextMenu", { read: ElementRef<HTMLElement>, }); isOpen = input.required<boolean>(); x = input.required<number>(); y = input.required<number>(); close = output(); closeIfOutsideOfContext = (e: MouseEvent) => { const contextMenuEl = this.contextMenu()?.nativeElement; if (!contextMenuEl) return; const isClickInside = contextMenuEl.contains(e.target as HTMLElement); if (isClickInside) return; this.close.emit(); }; constructor() { afterRenderEffect((onCleanup) => { document.addEventListener("click", this.closeIfOutsideOfContext); onCleanup(() => { document.removeEventListener("click", this.closeIfOutsideOfContext); }); }); }}@Component({ selector: "app-root", imports: [ContextMenuComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div style="margin-top: 5rem; margin-left: 5rem"> <div #contextOrigin (contextmenu)="open($event)">Right click on me!</div> </div> <context-menu (close)="close()" [isOpen]="isOpen()" [x]="mouseBounds().x" [y]="mouseBounds().y" /> `,})class AppComponent { isOpen = signal(false); mouseBounds = signal({ x: 0, y: 0, }); close() { this.isOpen.set(false); } open(e: MouseEvent) { e.preventDefault(); this.isOpen.set(true); this.mouseBounds.set({ x: e.clientX, y: e.clientY, }); }}
<!-- ContextMenu.vue --><script setup>import { ref, onMounted, onUnmounted } from "vue";const props = defineProps(["isOpen", "x", "y"]);const emit = defineEmits(["close"]);const contextMenuRef = ref(null);function closeIfOutside(e) { const contextMenuEl = contextMenuRef.value; if (!contextMenuEl) return; const isClickInside = contextMenuEl.contains(e.target); if (isClickInside) return; emit("close");}onMounted(() => { document.addEventListener("click", closeIfOutside);});onUnmounted(() => { document.removeEventListener("click", closeIfOutside);});</script><template> <div v-if="props.isOpen" ref="contextMenuRef" tabIndex="0" :style="` position: fixed; top: ${props.y}px; left: ${props.x}px; background: white; border: 1px solid black; border-radius: 16px; padding: 1rem; `" > <button @click="emit('close')">X</button> This is a context menu </div></template>
<!-- App.vue --><script setup>import { ref } from "vue";import ContextMenu from "./ContextMenu.vue";const isOpen = ref(false);const mouseBounds = ref({ x: 0, y: 0,});const close = () => { isOpen.value = false;};const open = (e) => { e.preventDefault(); isOpen.value = true; mouseBounds.value = { x: e.clientX, y: e.clientY, };};</script><template> <div style="margin-top: 5rem; margin-left: 5rem"> <div @contextmenu="open($event)">Right click on me!</div> </div> <ContextMenu :isOpen="isOpen" :x="mouseBounds.x" :y="mouseBounds.y" @close="close()" /></template>
You may have noticed that during this migration, we ended up removing a crucial accessibility feature: We're no longer running focus
on the context menu when it opens.
Why was it removed, and how can we add it back?
Introducing Component Reference
The reason we removed the context menu's focus management is to keep control of the context menu in the parent.
While we could have kept the .focus()
logic in the component using a component side effect handler, this muddies the water a bit. Ideally, in a framework, you want your parent to be in charge of the child component's behavior.
This allows you to re-use your context menu component in more places, should you theoretically ever want to use the component without forcing a focus change.
To do this, let's move the .focus
method out of our component. Moving from this:
/* This is valid JS, but is only pseudocode of what each framework is doing */// Child componentfunction onComponentRender() { document.addEventListener("click", closeIfOutsideOfContext); contextMenu.focus();}// Parent componentfunction openContextMenu(e) { e.preventDefault(); setOpen(true);}
To this:
/* This is valid JS, but is only pseudocode of what each framework is doing */// Child componentfunction onComponentRender() { document.addEventListener("click", closeIfOutsideOfContext);}// Parent componentfunction openContextMenu(e) { e.preventDefault(); setOpen(true); contextMenu.focus();}
While this might seem like a straightforward change at first, there's a new problem present: Our contextMenu
is now inside a component. As a result, we need to access not only the underlying DOM node using element reference but the ContextMenu
component instance as well.
Luckily for us, each framework enables us to do just that! Before we implement the focus
logic, let's dive into how component reference works:
- React
- Angular
- Vue
React has two APIs that help us gain insights into a component's internals from its parent:
Let's start with the basics: forwardRef
.
forwardRef
forwardRef
does what it says on the tin: It allows you to forward a ref
property through a component instance.
See, in React, ref
is a special property. This means that to be used properly, React has to have a special syntax to enable its expected functionality.
As a result, the following code does not work:
const Component = ({ ref, style }) => { return <div ref={ref} style={style} />;};const App = () => { return ( <Component ref={(el) => alert(el)} style={{ height: 100, width: 100, backgroundColor: "red" }} /> );};
Doing this will result in our ref
callback not being called as expected, alongside two error messages explaining why:
Warning: Component:
ref
is not a prop. Trying to access it will result inundefined
being returned. If you need to access the same value within the child component, you should pass it as a different prop. (https://reactjs.org/link/special-props)
Warning: Function components cannot be given refs. Attempts to access this ref will fail. Did you mean to use
forwardRef()
?
To solve this, we have two options:
- Rename our
ref
property to another name, likedivRef
:
const Component = ({ divRef, style }) => { return <div ref={divRef} style={style} />;};const App = () => { return ( <Component divRef={(el) => alert(el)} style={{ height: 100, width: 100, backgroundColor: "red" }} /> );};
- Use the
forwardRef
API, as suggested by the error message originally printed.
import { forwardRef } from "react";const Component = forwardRef((props, ref) => { return <div ref={ref} style={props.style} />;});const App = () => { return ( <Component ref={(el) => alert(el)} style={{ height: 100, width: 100, backgroundColor: "red" }} /> );};
As we can see, forwardRef
accepts slightly modified component functions. While the first argument might look familiar as our place to access properties, our special property ref
is passed as a second argument.
We can then forward that ref
to wherever we want to gain access to an underlying DOM node in the child.
But what if we wanted more control over our child component? What if we wanted to access data and methods from the child component using a ref
?
Luckily, useImperativeHandle
does just that!
useImperativeHandle
While forwardRef
enables us to pass a ref
to a child component, useImperativeHandle
allows us to fully customize this ref
to our heart's content.
import { forwardRef, useImperativeHandle } from "react";const Component = forwardRef((props, ref) => { useImperativeHandle(ref, () => { // Anything returned here will be assigned to the forwarded `ref` return { pi: 3.14, sayHi() { alert("Hello, world"); }, }; }); return <div style={props.style} />;});const App = () => { return ( <Component ref={(el) => console.log(el)} style={{ height: 100, width: 100, backgroundColor: "red" }} /> );};
Here, we can assign properties, functions, or any other JavaScript values to the forwarded ref
. If we look at the output of our ref
callback from App
it shows up the object that we assigned using useImperativeHandle
:
({ pi: 3.14, sayHi: sayHi() });
That sayHi
function still works, too! If we change App
to the following:
const App = () => { const compRef = useRef(); return ( <> <button onClick={() => compRef.current.sayHi()}>Say hi</button> <Component ref={compRef} /> </> );};
It will output Hello, world
, just as we would expect it to!
useImperativeHandle
Dependency Array
Let's stop and think about how useImperativeHandle
works under the hood for a moment.
We know that useRef
creates an object with the shape of:
({ current: initialValue });
Which you can then mutate without triggering a re-render.
const App = () => { const numberOfRenders = useRef(0); numberOfRenders.current += 1; return null;};
Now, let's say that we didn't have access to useImperativeHandle
, but still wanted to pass a value from a child component to a parent via the passed ref
. That might look something like this:
const Child = forwardRef((props, ref) => { ref.current += 1; return null;});const Parent = () => { const numberOfChildRenders = useRef(0); return <Child ref={numberOfChildRenders} />;};
But wait a moment! If we think back to our Side Effects chapter, we'll remember that mutating state outside a component's local values is an example of a side effect.
Because this kind of in-render side effect mutation can cause strange issues and edge-cases with React, we need to make sure that we're using useEffect
or useLayoutEffect
.
Because our code ideally should happen during our render (rather than afterward), let's opt to use useLayoutEffect
.
Changing the previous code to use useLayoutEffect
looks something like this:
const Child = forwardRef((props, ref) => { useLayoutEffect(() => { ref.current += 1; }); return null;});
This is similar to how useImperativeHandle
works under the hood. It's so similar, in fact, that it's effectively how the hook is written in React's source code itself.
But wait!
useLayoutEffect
has the option to pass an array to it so that you can avoid re-running the side effect. Does that work inuseImperativeHandle
as well?
Indeed, it does, astute reader! Instead of:
useLayoutEffect(() => { ref.current = someVal;}, [someVal]);
We might do the following instead:
useImperativeHandle(ref, () => someVal, [someVal]);
Just as we can use viewChild
to access an underlying DOM node, we can do the same thing with a component reference. In fact, we can use a template reference variable just like we would to access the DOM node.
@Component({ selector: "child-comp", changeDetection: ChangeDetectionStrategy.OnPush, template: `<div style="height: 100px; width: 100px; background-color: red;" ></div>`,})class ChildComponent { pi = 3.14; sayHi() { console.log("Hello, world"); }}@Component({ selector: "parent-comp", imports: [ChildComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `<child-comp #childVar />`,})class ParentComponent { childComp = viewChild.required("childVar", { read: ChildComponent }); constructor() { afterRenderEffect(() => { console.log(this.childComp()); }); }}
Doing this, we'll see the console output:
/* Object */ ({ pi: 3.14 });
But how do we know that this is properly the ChildComponent
instance? Simple! We'll console.log
childComp.constructor
and we'll see:
class ChildComponent {}
This means that, as a result, we can also call the sayHi
method:
@Component({ selector: "parent-comp", imports: [ChildComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <button (click)="sayHiFromChild()">Say hi</button> <child-comp #childVar /> `,})class ParentComponent { childComp = viewChild.required("childVar", { read: ChildComponent }); sayHiFromChild() { this.childComp().sayHi(); }}
And it will alert:
Hello, world
Using the same ref
API as element nodes, you can access a component's instance:
<!-- Child.vue --><script setup>const pi = 3.14;function sayHi() { alert("Hello, world");}</script><template> <p>Hello, template</p></template>
<!-- Parent.vue --><script setup>import { ref, onMounted } from "vue";import Child from "./Child.vue";const childComp = ref();onMounted(() => { console.log(childComp.value);});</script><template> <Child ref="childComp" /></template>
If we look at our console output, we might see something unexpected:
/* Proxy */ ({ "<target>": { /*…*/ }, "<handler>": { /*…*/ },});
This is because Vue uses Proxies under the hood to power component state. Rest assured, however, this Proxy
is still our component instance.
Exposing Component Variables to References
We're not able to do much with this component instance currently. If we change out Parent
component to console.log
, the pi
value from Child
:
<!-- Parent.vue --><script setup>import { ref, onMounted } from "vue";import Child from "./Child.vue";const childComp = ref();onMounted(() => { alert(childComp.value.pi);});</script><template> <Child ref="childComp" /></template>
We'll see that childComp.value.pi
is undefined
currently. This is because, by default, Vue's setup script
does not "expose" internal variables to component references externally.
To fix this, we can use Vue's defineExpose
global API to allow parent components to access a child component's variables and methods:
<!-- Child.vue --><script setup>const pi = 3.14;function sayHi() { alert("Hello, world");}defineExpose({ pi, sayHi,});</script><template> <p>Hello, template</p></template>
Because we now have access to the component instance, we can access data and call methods similar to how we're able to access data and call methods from an element reference.
<!-- Parent.vue --><script setup>import { ref, onMounted } from "vue";import Child from "./Child.vue";const childComp = ref();onMounted(() => { alert(childComp.value.pi); childComp.value.sayHi();});</script><template> <Child ref="childComp" /></template>
Using Component Reference to Focus Our Context Menu
Now that we sufficiently understand what component references look like in each framework, let's add it to our App
component to re-enable focusing our ContextMenu
component when it opens.
Remember, if you see:
setTimeout(() => { doSomething();}, 0);
It means that we want to defer the
doSomething
call until after all other tasks are complete. We're using this in our code samples to say:"Wait until the element is rendered to run
.focus()
on it"
- React
- Angular
- Vue
const ContextMenu = forwardRef(({ isOpen, x, y, onClose }, ref) => { const [contextMenu, setContextMenu] = useState(); useImperativeHandle(ref, () => ({ focus: () => contextMenu && contextMenu.focus(), })); // ... return ( // Attributes removed for brevity <div ref={(el) => setContextMenu(el)}>{/* ... */}</div> );});function App() { const [mouseBounds, setMouseBounds] = useState({ x: 0, y: 0, }); // ... const [isOpen, setIsOpen] = useState(false); const contextMenuRef = useRef(); useEffect(() => { if (isOpen) { setTimeout(() => { if (!contextMenuRef.current) return; contextMenuRef.current.focus(); }, 0); } }, [isOpen, mouseBounds]); return ( <> {/* ... */} <ContextMenu ref={contextMenuRef} isOpen={isOpen} onClose={() => setIsOpen(false)} x={mouseBounds.x} y={mouseBounds.y} /> </> );}
@Component({ selector: "context-menu", changeDetection: ChangeDetectionStrategy.OnPush, template: ` @if (isOpen()) { <!-- Attributes removed for brevity --> <div #contextMenu> <button (click)="close.emit()">X</button> This is a context menu </div> } `,})class ContextMenuComponent { contextMenu = viewChild("contextMenu", { read: ElementRef<HTMLElement> }); isOpen = input<boolean>(); x = input<number>(); y = input<number>(); close = output(); focus() { this.contextMenu()?.nativeElement?.focus(); } // ...}@Component({ selector: "app-root", imports: [ContextMenuComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div style="margin-top: 5rem; margin-left: 5rem"> <div #contextOrigin (contextmenu)="open($event)">Right click on me!</div> </div> <context-menu #contextMenu (close)="close()" [isOpen]="isOpen()" [x]="mouseBounds().x" [y]="mouseBounds().y" /> `,})class AppComponent { contextMenu = viewChild.required("contextMenu", { read: ContextMenuComponent, }); isOpen = signal(false); mouseBounds = signal({ x: 0, y: 0, }); close() { this.isOpen.set(false); } open(e: MouseEvent) { e.preventDefault(); this.isOpen.set(true); this.mouseBounds.set({ x: e.clientX, y: e.clientY, }); setTimeout(() => { this.contextMenu().focus(); }, 0); }}
<!-- ContextMenu.vue --><script setup>import { ref, onMounted, onUnmounted } from "vue";const props = defineProps(["isOpen", "x", "y"]);const emit = defineEmits(["close"]);const contextMenuRef = ref(null);// ...function focusMenu() { contextMenuRef.value.focus();}defineExpose({ focusMenu,});</script><template> <div v-if="props.isOpen" ref="contextMenuRef" tabIndex="0" :style="` position: fixed; top: ${props.y}px; left: ${props.x}px; background: white; border: 1px solid black; border-radius: 16px; padding: 1rem; `" > <button @click="emit('close')">X</button> This is a context menu </div></template>
<!-- App.vue --><script setup>import { ref } from "vue";import ContextMenu from "./ContextMenu.vue";const isOpen = ref(false);const mouseBounds = ref({ x: 0, y: 0,});const contextMenu = ref();const close = () => { isOpen.value = false;};const open = (e) => { e.preventDefault(); isOpen.value = true; mouseBounds.value = { x: e.clientX, y: e.clientY, }; setTimeout(() => { contextMenu.value.focusMenu(); }, 0);};</script><template> <div style="margin-top: 5rem; margin-left: 5rem"> <div @contextmenu="open($event)">Right click on me!</div> </div> <ContextMenu ref="contextMenu" :isOpen="isOpen" :x="mouseBounds.x" :y="mouseBounds.y" @close="close()" /></template>
Challenge
This information about component reference isn't just theoretically useful. You're able to apply it to your codebase to enable new methods of building out components.
Let's see that in action by building a sidebar component that's able to expand and collapse.
To add an extra special interaction with this sidebar, let's make it so that when the user shrinks their screen to a certain size, it will automatically collapse the sidebar.
To do this, we'll:
- Set up our
App
component to handle a left and main column. - Make a sidebar that can collapse and expand to grow and shrink the main column.
- Automatically expand or collapse the sidebar as the browser grows and shrinks.
Let's dive in.
Step 1: Setup App Component Layout
Let's start creating our sidebar!
Our first step in doing so will be creating a layout file that includes a left-hand sidebar and a main content area on the right side.
To do that might look something like this:
- React
- Angular
- Vue
const Layout = ({ sidebar, sidebarWidth, children }) => { return ( <div style={{ display: "flex", flexWrap: "nowrap", minHeight: "100vh" }}> <div style={{ width: `${sidebarWidth}px`, height: "100vh", overflowY: "scroll", borderRight: "2px solid #bfbfbf", }} > {sidebar} </div> <div style={{ width: "1px", flexGrow: 1 }}>{children}</div> </div> );};const App = () => { return ( <Layout sidebar={<p>Sidebar</p>} sidebarWidth={150}> <p style={{ padding: "1rem" }}>Hi there!</p> </Layout> );};
@Component({ selector: "app-layout", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div style="display: flex; flex-wrap: nowrap; min-height: 100vh"> <div style=" width: {{ sidebarWidth() }}px; height: 100vh; overflow-y: scroll; border-right: 2px solid #bfbfbf; " > <ng-content select="[sidebar]" /> </div> <div style="width: 1px; flex-grow: 1"> <ng-content /> </div> </div> `,})class LayoutComponent { sidebarWidth = input.required<number>();}@Component({ selector: "app-root", changeDetection: ChangeDetectionStrategy.OnPush, imports: [LayoutComponent], template: ` <app-layout [sidebarWidth]="150"> <p sidebar>Sidebar</p> <p style="padding: 1rem">Hi there!</p> </app-layout> `,})class AppComponent {}
<!-- Layout.vue --><script setup>const props = defineProps(["sidebarWidth"]);</script><template> <div style="display: flex; flex-wrap: nowrap; min-height: 100vh"> <div :style="` width: ${props.sidebarWidth}px; height: 100vh; overflow-y: scroll; border-right: 2px solid #bfbfbf; `" > <slot name="sidebar" /> </div> <div style="width: 1px; flex-grow: 1"> <slot /> </div> </div></template>
<!-- App.vue --><script setup>import Layout from "./Layout.vue";</script><template> <Layout :sidebarWidth="150"> <template #sidebar><p>Sidebar</p></template> <p style="padding: 1rem">Hi there!</p> </Layout></template>
Step 2: Make a Collapsible Sidebar
Now that we have a rough sidebar, we'll make it so that the user can manually collapse the sidebar.
This can be done by having an isCollapsed
state that the user toggles with a button.
When isCollapsed
is true
, it will only show the toggle button, but when isCollapsed
is false
, it should display the full sidebar's contents.
We'll also set up constants to support different widths of this sidebar area if it's collapsed or not.
- React
- Angular
- Vue
const Sidebar = ({ toggle }) => { const [isCollapsed, setIsCollapsed] = useState(false); const setAndToggle = (v) => { setIsCollapsed(v); toggle(v); }; const toggleCollapsed = () => { setAndToggle(!isCollapsed); }; if (isCollapsed) { return <button onClick={toggleCollapsed}>Toggle</button>; } return ( <div> <button onClick={toggleCollapsed}>Toggle</button> <ul style={{ padding: "1rem" }}> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div> );};const collapsedWidth = 100;const expandedWidth = 150;const App = () => { const [width, setWidth] = useState(expandedWidth); return ( <Layout sidebarWidth={width} sidebar={ <Sidebar toggle={(isCollapsed) => { if (isCollapsed) { setWidth(collapsedWidth); return; } setWidth(expandedWidth); }} /> } > <p style={{ padding: "1rem" }}>Hi there!</p> </Layout> );};
@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, template: ` @if (isCollapsed()) { <button (click)="toggleCollapsed()">Toggle</button> } @if (!isCollapsed()) { <div> <button (click)="toggleCollapsed()">Toggle</button> <ul style="padding: 1rem"> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div> } `,})class SidebarComponent { toggle = output<boolean>(); isCollapsed = signal(false); setAndToggle(v: boolean) { this.isCollapsed.set(v); this.toggle.emit(v); } toggleCollapsed() { this.setAndToggle(!this.isCollapsed()); }}@Component({ selector: "app-root", changeDetection: ChangeDetectionStrategy.OnPush, imports: [LayoutComponent, SidebarComponent], template: ` <app-layout [sidebarWidth]="width()"> <app-sidebar sidebar (toggle)="onToggle($event)" /> <p style="padding: 1rem">Hi there!</p> </app-layout> `,})class AppComponent { collapsedWidth = 100; expandedWidth = 150; width = signal(this.expandedWidth); onToggle(isCollapsed: boolean) { if (isCollapsed) { this.width.set(this.collapsedWidth); return; } this.width.set(this.expandedWidth); }}
<!-- Sidebar.vue --><script setup>import { ref } from "vue";const emit = defineEmits(["toggle"]);const isCollapsed = ref(false);function setAndToggle(v) { isCollapsed.value = v; emit("toggle", v);}function toggleCollapsed() { setAndToggle(!isCollapsed.value);}</script><template> <button v-if="isCollapsed" @click="toggleCollapsed()">Toggle</button> <div v-if="!isCollapsed"> <button @click="toggleCollapsed()">Toggle</button> <ul style="padding: 1rem"> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div></template>
<!-- App.vue --><script setup>import Layout from "./Layout.vue";import Sidebar from "./Sidebar.vue";import { ref } from "vue";const collapsedWidth = 100;const expandedWidth = 150;const width = ref(expandedWidth);function onToggle(isCollapsed) { if (isCollapsed) { width.value = collapsedWidth; return; } width.value = expandedWidth;}</script><template> <Layout :sidebarWidth="width"> <template #sidebar> <Sidebar @toggle="onToggle($event)" /> </template> <p style="padding: 1rem">Hi there!</p> </Layout></template>
Step 3: Auto-Collapse Sidebar on Small Screens
Finally, let's auto-collapse the sidebar on screens smaller than 600px wide.
We can do this using a side effect handler to add a listener for screen resizes.
Then, we'll use framework-specific code similar to the following pseudocode to expand or collapse the sidebar based on the screen size:
const onResize = () => { if (window.innerWidth < widthToCollapseAt) { sidebarRef.collapse(); } else if (sidebar.isCollapsed) { sidebarRef.expand(); }};window.addEventListener("resize", onResize);// Laterwindow.removeEventListener("resize", onResize);
Let's implement it:
- React
- Angular
- Vue
const Sidebar = forwardRef(({ toggle }, ref) => { const [isCollapsed, setIsCollapsed] = useState(false); const setAndToggle = (v) => { setIsCollapsed(v); toggle(v); }; useImperativeHandle( ref, () => ({ collapse: () => { setAndToggle(true); }, expand: () => { setAndToggle(false); }, isCollapsed: isCollapsed, }), [isCollapsed, setAndToggle], ); const toggleCollapsed = () => { setAndToggle(!isCollapsed); }; if (isCollapsed) { return <button onClick={toggleCollapsed}>Toggle</button>; } return ( <div> <button onClick={toggleCollapsed}>Toggle</button> <ul style={{ padding: "1rem" }}> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div> );});const collapsedWidth = 100;const expandedWidth = 150;const widthToCollapseAt = 600;const App = () => { const [width, setWidth] = useState(expandedWidth); const sidebarRef = useRef(); useEffect(() => { const onResize = () => { if (window.innerWidth < widthToCollapseAt) { sidebarRef.current.collapse(); } else if (sidebarRef.current.isCollapsed) { sidebarRef.current.expand(); } }; window.addEventListener("resize", onResize); return () => window.removeEventListener("resize", onResize); }, [sidebarRef]); return ( <Layout sidebarWidth={width} sidebar={ <Sidebar ref={sidebarRef} toggle={(isCollapsed) => { if (isCollapsed) { setWidth(collapsedWidth); return; } setWidth(expandedWidth); }} /> } > <p style={{ padding: "1rem" }}>Hi there!</p> </Layout> );};
Final code output
@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, template: ` @if (isCollapsed()) { <button (click)="toggleCollapsed()">Toggle</button> } @if (!isCollapsed()) { <div> <button (click)="toggleCollapsed()">Toggle</button> <ul style="padding: 1rem"> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div> } `,})class SidebarComponent { toggle = output<boolean>(); isCollapsed = signal(false); setAndToggle(v: boolean) { this.isCollapsed.set(v); this.toggle.emit(v); } collapse() { this.setAndToggle(true); } expand() { this.setAndToggle(false); } toggleCollapsed() { this.setAndToggle(!this.isCollapsed()); }}@Component({ selector: "app-root", imports: [LayoutComponent, SidebarComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <app-layout [sidebarWidth]="width()"> <app-sidebar #sidebar sidebar (toggle)="onToggle($event)" /> <p style="padding: 1rem">Hi there!</p> </app-layout> `,})class AppComponent { sidebar = viewChild.required("sidebar", { read: SidebarComponent }); collapsedWidth = 100; expandedWidth = 150; widthToCollapseAt = 600; width = signal(this.expandedWidth); onToggle(isCollapsed: boolean) { if (isCollapsed) { this.width.set(this.collapsedWidth); return; } this.width.set(this.expandedWidth); } onResize = () => { if (window.innerWidth < this.widthToCollapseAt) { this.sidebar().collapse(); } else if (this.sidebar().isCollapsed()) { this.sidebar().expand(); } }; constructor() { effect((onCleanup) => { window.addEventListener("resize", this.onResize); onCleanup(() => { window.removeEventListener("resize", this.onResize); }); }); }}
Final code output
<!-- Sidebar.vue --><script setup>import { ref } from "vue";const emits = defineEmits(["toggle"]);const isCollapsed = ref(false);const setAndToggle = (v) => { isCollapsed.value = v; emits("toggle", v);};const collapse = () => { setAndToggle(true);};const expand = () => { setAndToggle(false);};const toggleCollapsed = () => { setAndToggle(!isCollapsed.value);};defineExpose({ expand, collapse, isCollapsed,});</script><template> <button v-if="isCollapsed" @click="toggleCollapsed()">Toggle</button> <div v-if="!isCollapsed"> <button @click="toggleCollapsed()">Toggle</button> <ul style="padding: 1rem"> <li>List item 1</li> <li>List item 2</li> <li>List item 3</li> <li>List item 4</li> <li>List item 5</li> <li>List item 6</li> </ul> </div></template>
<!-- App.vue --><script setup>import { onMounted, onUnmounted, ref } from "vue";import Layout from "./Layout.vue";import Sidebar from "./Sidebar.vue";const collapsedWidth = 100;const expandedWidth = 150;const widthToCollapseAt = 600;const sidebar = ref();const width = ref(expandedWidth);const onToggle = (isCollapsed) => { if (isCollapsed) { width.value = collapsedWidth; return; } width.value = expandedWidth;};const onResize = () => { if (window.innerWidth < widthToCollapseAt) { sidebar.value.collapse(); } else if (sidebar.value.isCollapsed) { sidebar.value.expand(); }};onMounted(() => { window.addEventListener("resize", onResize);});onUnmounted(() => { window.removeEventListener("resize", onResize);});</script><template> <Layout :sidebarWidth="width"> <template #sidebar> <Sidebar ref="sidebar" @toggle="onToggle($event)" /> </template> <p style="padding: 1rem">Hi there!</p> </Layout></template>
Final code output
Now, when the user makes their screen too small, the sidebar automatically collapses. This makes the rest of our app much easier to interact with on mobile devices.
Truth be told, this is not necessarily how I would build this component in production. Instead, I might "raise the state" of "collapsed" from the
Sidebar
component to theApp
component.This would give us greater flexibility in controlling our sidebar's
isCollapsed
state without having to use a component reference.However, if you're building a UI library meant to interact with multiple applications, sometimes having this state lowered can allow you to reduce boilerplate between apps that share this component.