Post contents
Before we can dive into how many front-end frameworks work, we need to set a baseline of information. If you're already familiar with how the DOM represents a tree and how the browser takes that information and uses it, great; You're ready to read ahead! Otherwise, it's strongly suggested that you look at our post introducing the concepts required to understand some baselines of this post.
You may have heard about various frameworks and libraries that modern front-end developers use to build large-scale applications. Among these frameworks are Angular, React, and Vue. While each of these libraries brings its own strengths and weaknesses, many of the core concepts are shared between them.
With this book, we will outline core concepts shared between them and how you can implement them in code in all three of the frameworks. This should provide a good reference when trying to learn one of these frameworks without pre-requisite knowledge or even trying to learn another framework with some pre-requisite of a different one.
First, let's explain why frameworks like Angular, React, or Vue differ from other libraries that may have come before them, like jQuery.
It all comes down to a single core concept at the heart of each: Componentization.
What's an App, Anyway?
Before we dive into the technical aspects, let's think about what an app consists of at a high level.
Take the following application into consideration.
Our app has many parts to it. For example, a sidebar containing navigation links, a list of files for a user to navigate, and a details pane about the user's selected file.
What's more, each part of the app needs different things.
The sidebar may not require complex programming logic, but we may want to style it with nice colors and highlight effects when the user hovers. Likewise, the file list may contain complex logic to handle a user right-clicking, dragging, and dropping files.
When you break it down, each part of the app has three primary concerns:
- Logic — What does the app do?
- Styling — How does the app look visually?
- Structure — How is the app laid out?
While the mockup above does a decent job of displaying things visually, let's look at what the app looks like structurally:
Here, each section is laid out without any additional styling: Simply a wireframe of what the page will look like, with each section containing blocks laid out in pretty straightforward ways. This is what HTML will help us build.
Now that we understand the structure, let's add some functionality. First, we'll include a small snippet of text in each section to outline our goals. We'd then use these as "acceptance" criteria in the future. This is what our logic will provide to the app.
Great! Now, let's go back and add the styling to recreate the mockup we had before!
We can think of each step of the process like we're adding in a new programming language.
- HTML is used for adding the structure of an application. The side nav might be a
<nav>
tag, for example. - JavaScript adds the logic of the application on top of the structure.
- CSS makes everything look nice and potentially adds some minor UX improvements.
The way I typically think about these three pieces of tech is:
HTML is like building blueprints. It allows you to see the overarching pictures of what the result will look like. They define the walls, doors, and flow of a home.
JavaScript is like the electrical, plumbing, and appliances of the house. They allow you to interact with the building in a meaningful way.
CSS is like the paint and other decors that go into a home. They're what makes the house feel lived in and inviting. Of course, this decor does little without the rest of the home, but without the decor, it's a miserable experience.
Parts of the App
Now that we've introduced what an app looks like, let's go back for a moment. Remember how I said each app is made of parts? Let's explode the app's mockup into smaller pieces and look at them more in-depth.
Here, we can more distinctly see how each part of the app has its own structure, styling, and logic.
The file list, for example, contains the structure of each file being its own item, logic about what buttons do which actions, and some CSS to make it look engaging.
While the code for this section might look something like this:
<section> <button id="addButton"><span class="icon">plus</span></button> <!-- ... --></section><ul> <li> <a href="/file/file_one">File one<span>12/03/21</span></a> </li> <!-- ... --> <ul> <script> var addButton = document.querySelector("#addButton"); addButton.addEventListener("click", () => { // ... }); </script> </ul></ul>
We might have a mental model to break down each section into smaller ones. If we use pseudocode to represent our mental model of the actual codebase, it might look something like this:
<files-buttons> <add-button /></files-buttons><files-list> <file name="File one" /></files-list>
Luckily, by using frameworks, this mental model can be reflected in real code!
Let's look at what <file>
might look like in each framework:
- React
- Angular
- Vue
const File = () => { return ( <div> <a href="/file/file_one"> File one<span>12/03/21</span> </a> </div> );};
Here, we're defining a component that we call File
, which contains a set of instructions for how React is to create the associated HTML when the component is used.
These HTML creation instructions are defined using a syntax very similar to HTML — but in JavaScript instead. This syntax is called "JSX" and powers the show for every React application.
While JSX looks closer to HTML than standard JS, it is not supported in the language itself. Instead, a compiler (or transpiler) like Babel must compile down to regular JS. Under the hood, this JSX compiles down to function calls.
For example, the above would be turned into:
var spanTag = React.createElement("span", null, "12/03/21");var aTag = React.createElement( "a", { href: "/file/file_one", }, "File one", spanTag,);React.createElement("div", null, aTag);
While the above seems intimidating, it's worth mentioning that you'll likely never need to fall back on using
createElement
in an actual production application. Instead, this demonstrates why you need Babel in React applications.You also likely do not need to set up Babel yourself from scratch. Most tools that integrate with React handle it out-of-the-box for you invisibly.
import { Component, ChangeDetectionStrategy } from "@angular/core";@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div> `,})class FileComponent {}
Here, we're using the @Component
decorator to define a class component in Angular.
This decorator has a few properties passed to it. Going from the bottom-up:
template
: The HTML associated with this component.changeDetection
: A flag we're using to tell Angular to use the non-defaultOnPush
method of detecting changes, which provides better performance.- All of the code examples in this book would work with
OnPush
or without it, but we opted to make it the default for our components to enforce the additional rules required to makeOnPush
works. This will set you up better for production apps in your future.
- All of the code examples in this book would work with
selector
: The name of the component that can be referenced inside thetemplate
of another component- You may have noticed that this component is called
file-item
rather thanfile
. Unlike the other frameworks in this book, Angular requires you to have a dash (-
) in your selector name to avoid confusion with native HTML tags.
- You may have noticed that this component is called
It's important to note that decorators (anything starting with
@
) are not supported in JavaScript itself. Instead, Angular uses TypeScript to add types and other features to the language. From there, TypeScript compiles down to JavaScript.
<!-- File.vue --><template> <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div></template>
This is a specially named .vue
file, which defines a Vue component called "File" and has a template for which HTML should be displayed when this component is used.
Unlike the other frameworks, which require you to explicitly name your components, Vue uses the name of your
.vue
file to define the component's name.
Each Vue component uses an individual .vue
file to contain its layout, styling, and logic. As such, these .vue
files are often called "Single File Components," or SFCs for short.
While this SFC looks precisely like standard HTML with nothing special added, that will quickly change as we learn more about Vue.
These are called "components." Components have various aspects to them, which we'll learn about throughout the course of this book.
We can see that each framework has its own syntax to display these components, but they often share more similarities than you might think.
Now that we've defined our components, there's a question: how do you use these components in HTML?
Rendering the App
While these components might look like simple HTML, they're capable of much more advanced usage. Because of this, each framework actually uses JavaScript under the hood to "draw" these components on-screen.
This process of "drawing" is called "rendering". This is not a one-and-done, however. A component may render at various times throughout its usage on-screen, particularly when it needs to update data shown on-screen; we'll learn more about this later in the chapter.
Traditionally, when you build out a website with just HTML, you'd define an index.html
file like so:
<!-- index.html --><html> <body> <!-- Your HTML here --> </body></html>
Similarly, all apps built with React, Angular, and Vue start with an index.html
file.
- React
- Angular
- Vue
<!-- index.html --><html> <body> <div id="root"></div> </body></html>
<!-- index.html --><html> <body> <!-- This should match the `selector` of the --> <!-- component you want here --> <file-item></file-item> </body></html>
<!-- index.html --><html> <body> <div id="root"></div> </body></html>
Then, in JavaScript, you "render" a component into an element that acts as the "root" injection site for your framework to build your UI around.
- React
- Angular
- Vue
import { createRoot } from "react-dom/client";const File = () => { return ( <div> <a href="/file/file_one"> File one<span>12/03/21</span> </a> </div> );};createRoot(document.getElementById("root")).render(<File />);
import { bootstrapApplication } from '@angular/platform-browser';import {provideExperimentalZonelessChangeDetection} from '@angular/core';import { Component, ChangeDetectionStrategy } from "@angular/core";@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div> `,})class FileComponent {}bootstrapApplication(FileComponent, { providers: [ provideExperimentalZonelessChangeDetection() ]}) .catch((err) => console.error(err))
As the name implies, provideExperimentalZonelessChangeDetection
enables "Zoneless change detection". When combined with the providers
array (our Dependency Injection chapter will explain this more), it disables Zone.js, an older (and slower) method of detecting changes in your component.
What is a "change detection"? Why do we care about it?
We'll touch on this soon!
While
provideExperimentalZonelessChangeDetection
is technically experimental, it's fairly stable. Stable enough that Google's usage of Angular internally has enabled this by default for all of their apps going forward.The reason we've opted to use it in our book is:
- It's the future of Angular. One day, all modern Angular apps will be "Zoneless".
- When used with
OnPush
, you shouldn't notice a difference between this and Zone.js usage.
Because Vue's components all live within dedicated .vue
SFCs, we have to use two distinct files to render a basic Vue app. We start with our File.vue
component:
<!-- File.vue --><template> <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div></template>
Then, we can import this into our main JavaScript file:
// main.jsimport { createApp } from "vue";import File from "./File.vue";createApp(File).mount("#root");
Once a component is rendered, you can do a lot more with it!
For example, just like nodes in the DOM have relationships, so too can components.
Children, Siblings, and More, Oh My!
While our File
component currently contains HTML elements, components may also contain other components!
- React
- Angular
- Vue
const File = () => { return ( <div> <a href="/file/file_one"> File one<span>12/03/21</span> </a> </div> );};const FileList = () => { return ( <ul> <li> <File /> </li> </ul> );};
@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div> `,})class FileComponent {}@Component({ selector: "file-list", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileComponent], template: ` <ul> <li><file-item /></li> </ul> `,})class FileListComponent {}
Notice how we've told our FileListComponent
to import
FileComponent
by passing it the imports
array.
As we mentioned earlier, you can only have one component in a .vue
SFC. Here, we have our existing File
component:
<!-- File.vue --><template> <div> <a href="/file/file_one">File one<span>12/03/21</span></a> </div></template>
Which we can import
into another component to use it there:
<!-- FileList.vue --><script setup>import File from "./File.vue";</script><template> <ul> <li><File /></li> </ul></template>
We can import
and use our component immediately because any variable we expose inside of <script setup>
is automatically available in the <template>
portion of our SFC.
Notice that our
script
tag has asetup
attribute! Without it, our code won't work the right way!
We must import all the components we'll use in our parent component! Otherwise, Vue will throw an error:
Failed to resolve component: file
Looking through our File
component, we'll notice that we're rendering multiple elements inside a single component. Funnily enough, this has the fun side effect that we can also render multiple components inside a parent component.
- React
- Angular
- Vue
const FileList = () => { return ( <ul> <li> <File /> </li> <li> <File /> </li> <li> <File /> </li> </ul> );};
@Component({ selector: "file-list", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileComponent], template: ` <ul> <li><file-item /></li> <li><file-item /></li> <li><file-item /></li> </ul> `,})class FileListComponent {}
<!-- FileList.vue --><script setup>import File from "./File.vue";</script><template> <ul> <li><File /></li> <li><File /></li> <li><File /></li> </ul></template>
This is a handy feature of components. It allows you to reuse aspects of your structure (and styling + logic, but I'm getting ahead of myself) without repeating yourself. It allows for a very DRY architecture where your code is declared once and reused elsewhere.
That stands for "Don't repeat yourself" and is often heralded as a gold standard of code quality!
It's worth remembering that we're using the term "parent" to refer to our FileList
component in relation to our File
component. This is because, like the DOM tree, each framework's set of components reflects a tree.
This means that the related File
components are "siblings" of one another, each with a "parent" of FileList
.
We can extend this hierarchical relationship to have "grandchildren" and beyond as well:
- React
- Angular
- Vue
const FileDate = () => { return <span>12/03/21</span>;};const File = () => { return ( <div> <a href="/file/file_one"> File one <FileDate /> </a> </div> );};const FileList = () => { return ( <ul> <li> <File /> </li> <li> <File /> </li> <li> <File /> </li> </ul> );};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>12/03/21</span>`,})class FileDateComponent {}@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileDateComponent], template: ` <div> <a href="/file/file_one">File one<file-date /></a> </div> `,})class FileComponent {}@Component({ selector: "file-list", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileComponent], template: ` <ul> <li><file-item /></li> <li><file-item /></li> <li><file-item /></li> </ul> `,})class FileListComponent {}
<!-- FileDate.vue --><template> <span>12/03/21</span></template>
<!-- File.vue --><script setup>import FileDate from "./FileDate.vue";</script><template> <div> <a href="/file/file_one">File one<FileDate /></a> </div></template>
<!-- FileList.vue --><script setup>import File from "./File.vue";</script><template> <ul> <li><File /></li> <li><File /></li> <li><File /></li> </ul></template>
Logic
HTML isn't the only thing components can store, however! As we mentioned earlier, apps (and, by extension, each part of the respective apps) require three parts:
- Structure (HTML)
- Styling (CSS)
- Logic (JS)
Components can handle all three!
Let's look at how we can declare logic in a component by making file-date
show the current date instead of a static date.
We'll start by adding a variable containing the current date in a human-readable string of MM/DD/YY
.
- React
- Angular
- Vue
const FileDate = () => { const dateStr = `${ new Date().getMonth() + 1 }/${new Date().getDate()}/${new Date().getFullYear()}`; return <span>12/03/21</span>;};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>12/03/21</span>`,})class FileDateComponent { dateStr = `${ new Date().getMonth() + 1 }/${new Date().getDate()}/${new Date().getFullYear()}`;}
<!-- FileDate.vue --><script setup>const dateStr = `${ new Date().getMonth() + 1}/${new Date().getDate()}/${new Date().getFullYear()}`;</script><template> <span>12/03/21</span></template>
We're not using this new
dateStr
variable yet. This is intentional; we'll use it here shortly.
While this logic to set this variable works, it's a bit verbose (and slow due to recreating the Date
object thrice) - let's break it out into a method contained within the component.
function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}
- React
- Angular
- Vue
function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const FileDate = () => { const dateStr = formatDate(); return <span>12/03/21</span>;};
Because React can easily access functions outside the component declaration, we decided to move it outside the component scope. This allows us to avoid re-declaring this function in every render, which the other frameworks don't do, thanks to different philosophies.
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>12/03/21</span>`,})class FileDateComponent { dateStr = formatDate();}function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}
<!-- FileDate.vue --><script setup>function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const dateStr = formatDate();</script><template> <span>12/03/21</span></template>
Intro to Side Effects
Let's verify that our formatDate
method outputs the correct value by telling our components, "Once you're rendered on screen, console.log
the value of that data."
- React
- Angular
- Vue
import { useEffect } from "react";function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const month = today.getMonth() + 1; const date = today.getDate(); const year = today.getFullYear(); return month + "/" + date + "/" + year;}const FileDate = () => { const dateStr = formatDate(); useEffect(() => { console.log(dateStr); }, []); return <span>12/03/21</span>;};
import { Component, ChangeDetectionStrategy, effect } from "@angular/core";@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>12/03/21</span>`,})class FileDateComponent { dateStr = formatDate(); constructor() { effect(() => { console.log(this.dateStr); }); }}function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}
<!-- FileDate.vue --><script setup>import { onMounted } from "vue";function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const dateStr = formatDate();onMounted(() => { console.log(dateStr);});</script><template> <span>12/03/21</span></template>
Here, we're telling each respective framework to log the value of dateStr
to the console once the component is rendered for the first time.
Wait, "for the first time?"
Yup! React, Angular, and Vue can all update (or "re-render") when needed.
For example, let's say you want to show dateStr
to a user, but later in the day, the time switches over. While you'd have to handle the code to keep track of the time, the respective framework would notice that you've modified the values of dateStr
and re-render the component to display the new value.
While the method each framework uses to tell when to re-render is different, they all have a highly stable method of doing so.
This feature is arguably the most significant advantage of building an application with one of these frameworks.
This ability to track data being changed relies on the concept of handling "side effects". While we'll touch on this more in our future chapter called "Side effects", you can think of a "side effect" as any change made to a component's data: Either through a user's input or the component's output changing.
Speaking of updating data on-screen - let's look at how we can dynamically display data on a page.
Display
While displaying the value in the console works well for debugging, it's not much help to the user. After all, more than likely, your users won't know what a console even is. Let's show dateStr
on-screen
- React
- Angular
- Vue
function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const FileDate = () => { const dateStr = formatDate(); return <span>{dateStr}</span>;};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>{{ dateStr }}</span>`,})class FileDateComponent { dateStr = formatDate();}function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}
Every class property inside the component instance is usable inside the @Component
's template
.
<!-- FileDate.vue --><script setup>import { onMounted } from "vue";function formatDate() { const today = new Date(); // Month starts at 0, annoyingly const monthNum = today.getMonth() + 1; const dateNum = today.getDate(); const yearNum = today.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const dateStr = formatDate();</script><template> <span>{{ dateStr }}</span></template>
Here, we're using the fact that every variable inside of <script setup>
is automatically exposed to our <template>
code.
Here, we're using each framework's method of injecting the state into a component. For React, we'll use the {}
syntax to interpolate JavaScript into the template, while Vue and Angular both rely on {{}}
syntax.
Live Updating
But what happens if we update dateStr
after the fact? Say we have a setTimeout
call that updates the date to tomorrow's date after 5 minutes.
Let's think about what that code might look like:
// This is non-framework-specific pseudocodesetTimeout(() => { // 24 hours, 60 minutes, 60 seconds, 1000 milliseconds const tomorrow = new Date(Date.now() + 24 * 60 * 60 * 1000); const tomorrowDate = formatDate(tomorrow); dateStr = tomorrowDate; // This is not a real method in any of these frameworks // But the idea of re-rendering after data has changed IS // an integral part of these frameworks. They just do it differently rerender();}, 5000);
Let's see what that looks like in practice for each framework:
- React
- Angular
- Vue
In the pseudocode sample we wrote before, we update the value of dateStr
and then re-render the containing component to update a value on-screen using two lines of code.
In React, we use a single line of code to do both and have a special useState
method to tell React what data needs changing.
import { useState, useEffect } from "react";const FileDate = () => { const [dateStr, setDateStr] = useState(formatDate(new Date())); useEffect(() => { setTimeout(() => { // 24 hours, 60 minutes, 60 seconds, 1000 milliseconds const tomorrow = new Date(Date.now() + 24 * 60 * 60 * 1000); const tomorrowDate = formatDate(tomorrow); setDateStr(tomorrowDate); }, 5000); }, []); return <span>{dateStr}</span>;};function formatDate(inputDate) { // Month starts at 0, annoyingly const month = inputDate.getMonth() + 1; const date = inputDate.getDate(); const year = inputDate.getFullYear(); return month + "/" + date + "/" + year;}
useState
is what React uses to store data that the developer wants to persist between renders. Its first argument (that we're passing a string into) sets the initial value.
We then use array destructuring to convert the returned array into two variables. Another way to write this code is:
const dateArr = useState( `${ new Date().getMonth() + 1 }/${new Date().getDate()}/${new Date().getFullYear()}`,);const dateStr = dateArr[0];const setDateStr = dateArr[1];
Here, we're using setDateStr
to tell React that it should re-render, which will update the value of dateStr
. This differs from Angular and Vue, where you don't have to explicitly tell the framework when to re-render.
Rules of React Hooks
useState
and useEffect
are both what are known as "React Hooks". Hooks are React's method of "hooking" functionality into React's framework code. They allow you to do a myriad of functionalities in React components.
Hooks can be identified as a function that starts with the word "use
". Some other Hooks we'll touch on in the future will include useMemo
, useReducer
, and others.
Something to keep in mind when thinking about Hooks is that they have limitations placed on them by React itself. Namely, React Hooks must:
- 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)
// ❌ Not allowed, component names must start with a capital letter, otherwise it's seen as a normal functionconst windowSize = () => { const [dateStr, setDateStr] = useState(formatDate(new Date())); // ...};
// ❌ Not allowed, you must use a hook _inside_ a componentconst [dateStr, setDateStr] = useState(formatDate(new Date()));const Component = () => { return <p>The date is: {dateStr}</p>;};
// ❌ Not allowed, you cannot `return` before using a hookconst WindowSize = () => { if (bool) return "Today"; const [dateStr, setDateStr] = useState(formatDate(new Date())); // ...};
We'll learn more about the nuances surrounding Hooks in the future, but for now, just remember that they're the way you interface with React's APIs.
In Angular, any data that's:
- Updated more than once
- Shown on screen
Needs to be stored in a "Signal". In short, a signal is a data container that notifies other parts of the code when the inner data has changed. Angular can then hook into that notification system to update the DOM live.
To use a signal, you have to create an instance of it:
const count = signal(0);
You then call a signal like a function to read the inner value:
console.log(count());
And use .set
to update the inner value:
count.set(1);
Now that we know how to use a signal, we can see it in effect:
import { Component, ChangeDetectionStrategy, effect, signal } from "@angular/core";@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span>{{ dateStr() }}</span>`,})class FileDateComponent { dateStr = signal(formatDate(new Date())); constructor() { effect(() => { setTimeout(() => { // 24 hours, 60 minutes, 60 seconds, 1000 milliseconds const tomorrow = new Date(Date.now() + 24 * 60 * 60 * 1000); this.dateStr.set(formatDate(tomorrow)); }, 5000); }); }}function formatDate(inputDate: Date) { // Month starts at 0, annoyingly const monthNum = inputDate.getMonth() + 1; const dateNum = inputDate.getDate(); const yearNum = inputDate.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}
Similar to how React has useState
in order to set data in a component, Vue introduces an API called ref
in order to have data updates trigger a re-render.
<!-- FileDate.vue --><script setup>import { ref, onMounted } from "vue";function formatDate(inputDate) { // Month starts at 0, annoyingly const monthNum = inputDate.getMonth() + 1; const dateNum = inputDate.getDate(); const yearNum = inputDate.getFullYear(); return monthNum + "/" + dateNum + "/" + yearNum;}const dateStr = ref(formatDate(new Date()));onMounted(() => { setTimeout(() => { // 24 hours, 60 minutes, 60 seconds, 1000 milliseconds const tomorrow = new Date(Date.now() + 24 * 60 * 60 * 1000); dateStr.value = formatDate(tomorrow); }, 5000);});</script><template> <span>{{ dateStr }}</span></template>
Notice how we're using
.value
to update the value inside of<script>
but don't use.value
inside of<template>
. This isn't a mistake — it's just how Vue'sref
works!
If you sit on these screens for a while, you'll see that they update automatically!
This idea of a data update triggering other code is called "reactivity" and is a central part of these frameworks.
While the frameworks detect reactive changes under the hood differently, they all handle updating the DOM for you. This allows you to focus on the logic intended to update what's on-screen as opposed to the code that updates the DOM itself.
This is important because to update the DOM in an efficient way requires significant heavy lifting. In fact, two of these frameworks (React and Vue) store an entire copy of the DOM in memory to keep that update as lightweight as possible. In the third book of this book series, "Internals", we'll learn how this works under the hood and how to build our work version of this DOM mirroring.
Attribute Binding
Text isn't the only thing that frameworks are capable of live updating, however!
Just like each framework has a way to have state rendered into text on-screen, it can also update HTML attributes for an element.
Currently, our date
component doesn't read out particularly kindly to screen-readers since it would only read out as numbers. Let's change that by adding an aria-label
of a human-readable date to our date
component.
- React
- Angular
- Vue
const FileDate = () => { const [dateStr, setDateStr] = useState(formatDate(new Date())); // ... return <span aria-label="January 10th, 2023">{dateStr}</span>;};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <span aria-label="January 10th, 2023">{{ dateStr }}</span> `,})class FileDateComponent implements OnInit { dateStr = signal(formatDate(new Date())); // ...}
<!-- FileDate.vue --><script setup>// ...const dateStr = ref(formatDate(new Date()));</script><template> <span aria-label="January 10th, 2023">{{ dateStr }}</span></template>
Now, when we use a screen reader, it'll read out "January 10th" instead of "One dash ten".
However, while this may have worked before the date
was dynamically formatted, it won't be correct for most of the year. (Luckily for us, a broken clock is correct at least once a day.)
Let's correct that by adding in a formatReadableDate
method and reflect that in the attribute:
- React
- Angular
- Vue
import { useState, useEffect } from "react";const FileDate = () => { const [dateStr, setDateStr] = useState(formatDate(new Date())); const [labelText, setLabelText] = useState(formatReadableDate(new Date())); // ... return <span aria-label={labelText}>{dateStr}</span>;};// ...function formatReadableDate(inputDate) { const months = [ "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December", ]; const monthStr = months[inputDate.getMonth()]; const dateSuffixStr = dateSuffix(inputDate.getDate()); const yearNum = inputDate.getFullYear(); return monthStr + " " + dateSuffixStr + "," + yearNum;}function dateSuffix(dayNumber) { const lastDigit = dayNumber % 10; if (lastDigit == 1 && dayNumber != 11) { return dayNumber + "st"; } if (lastDigit == 2 && dayNumber != 12) { return dayNumber + "nd"; } if (lastDigit == 3 && dayNumber != 13) { return dayNumber + "rd"; } return dayNumber + "th";}
Notice the
{}
used after the=
to assign the attribute value. This is pretty similar to the syntax to interpolate text into the DOM!
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span [attr.aria-label]="labelText()">{{ dateStr() }}</span>`,})class FileDateComponent { dateStr = signal(formatDate(new Date())); labelText = signal(formatReadableDate(new Date())); // ...}// ...function dateSuffix(dayNumber: number) { const lastDigit = dayNumber % 10; if (lastDigit == 1 && dayNumber != 11) { return dayNumber + "st"; } if (lastDigit == 2 && dayNumber != 12) { return dayNumber + "nd"; } if (lastDigit == 3 && dayNumber != 13) { return dayNumber + "rd"; } return dayNumber + "th";}function formatReadableDate(inputDate: Date) { const months = [ "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December", ]; const monthStr = months[inputDate.getMonth()]; const dateSuffixStr = dateSuffix(inputDate.getDate()); const yearNum = inputDate.getFullYear(); return monthStr + " " + dateSuffixStr + "," + yearNum;}
Unlike the
{{}}
that you'd use to bind text to the DOM, you use[]
to bind attributes in Angular."
attr
" stands for "attribute." We'll see the other usage for the[]
syntax momentarily.
<!-- FileDate.vue --><script setup>// ...function formatReadableDate(inputDate) { const months = [ "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December", ]; const monthStr = months[inputDate.getMonth()]; const dateSuffixStr = dateSuffix(inputDate.getDate()); const yearNum = inputDate.getFullYear(); return monthStr + " " + dateSuffixStr + "," + yearNum;}function dateSuffix(dayNumber) { const lastDigit = dayNumber % 10; if (lastDigit == 1 && dayNumber != 11) { return dayNumber + "st"; } if (lastDigit == 2 && dayNumber != 12) { return dayNumber + "nd"; } if (lastDigit == 3 && dayNumber != 13) { return dayNumber + "rd"; } return dayNumber + "th";}const dateStr = ref(formatDate(new Date()));const labelText = ref(formatReadableDate(new Date()));// ...</script><template> <span v-bind:aria-label="labelText">{{ dateStr }}</span></template>
In Vue,
v-bind
has a shorter syntax that does the same thing. If you ax thev-bind
and leave the:
, it works the same way.This means that:
<span v-bind:aria-label="labelText">{{dateStr}}</span>
And:
<span :aria-label="labelText">{{dateStr}}</span>
Both work to bind to an attribute in HTML.
This code isn't exactly what you might expect to see in production. If you're looking to write production code, you may want to look into derived values to base the
labelText
anddate
values off of the sameDate
object directly. This would let you avoid callingnew Date
twice, but I'm getting ahead of myself - we'll touch on derived values in a future section.
Awesome! Now, it should read the file's date properly to a screen reader!
Inputs
Our file list is starting to look good! That said, a file list containing the same file repeatedly isn't much of a file list. Ideally, we'd like to pass the file's name into our File
component to add a bit of variance.
Luckily for us, components accept arguments just like functions! These arguments are most often called "inputs" or "properties" (shortened to "props") in the component world.
Let's have the file name be an input to our File
component:
- React
- Angular
- Vue
const File = (props) => { return ( <div> <a href="/file/file_one"> {props.fileName} <FileDate /> </a> </div> );};const FileList = () => { return ( <ul> <li> <File fileName={"File one"} /> </li> <li> <File fileName={"File two"} /> </li> <li> <File fileName={"File three"} /> </li> </ul> );};
React uses an object to contain all properties that we want to pass to a component. We can use parameter destructuring to get the properties without having to use
props
before the name of the parameter we really want, like so:const File = ({ fileName }) => { return ( <div> <a href="/file/file_one"> {fileName} <FileDate /> </a> </div> );};
Since this is extremely common in production React applications, we'll be using this style going forward.
Similarly, while
{}
is required to bind a variable to an input or attribute in React, since we're only passing a string here, we could alternatively write:<File fileName="File one" />
import { Component, ChangeDetectionStrategy, input } from "@angular/core";@Component({ selector: "file-item", imports: [FileDateComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <a href="/file/file_one">{{ fileName() }}<file-date /></a> </div> `,})class FileComponent { fileName = input.required<string>();}@Component({ selector: "file-list", imports: [FileComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <ul> <li><file-item [fileName]="'File one'" /></li> <li><file-item [fileName]="'File two'" /></li> <li><file-item [fileName]="'File three'" /></li> </ul> `,})class FileListComponent {}
See? Told you we'd cover what
[]
would be used for. It's the same binding syntax as with attributes!
Required vs Optional Properties
In our code sample, we wrote:
class FileComponent { fileName = input.required<string>();}
This tells Angular that this input must be passed, and that an error should be thrown if it is not.
However, not all inputs are this mandatory. We could, for example, choose to have a default fileName
if one is not passed:
class FileComponent { fileName = input("file.txt");}
We can see that we can remove the manual <string>
when we pass a default value, but now it will no longer throw an error if you forget to pass it.
Make sure to pick the right one for any given example! Don't just hastily default to one or another.
<!-- File.vue --><script setup>import FileDate from "./FileDate.vue";const props = defineProps(["fileName"]);</script><template> <div> <a href="/file/file_one">{{ props.fileName }}<FileDate /></a> </div></template>
<!-- FileList.vue --><script setup>import File from "./File.vue";</script><template> <ul> <li><File :fileName="'File one'" /></li> <li><File :fileName="'File two'" /></li> <li><File :fileName="'File three'" /></li> </ul></template>
We don't need to import
defineProps
, instead, Vue uses some compiler magic to provide it as a globally accessible method.Here, we need to declare each property using
defineProps
on our component; otherwise, the input value won't be available to the rest of the component.Also, when we talked about attribute binding, we mentioned
:
is shorthand forv-bind:
. The same applies here too. You could alternatively write:<File v-bind:fileName="'File three'" />
Here, we can see each File
being rendered with its own name.
One way of thinking about passing properties to a component is to "pass down" data to our children's components. Remember, these components make a parent/child relationship to one another.
It's exciting what progress we're making! But oh no - the links are still static! Each file has the same href
property as the last. Let's fix that!
Multiple Properties
Like functions, components can accept as many properties as you'd like to pass. Let's add another for href
:
- React
- Angular
- Vue
const File = ({ fileName, href }) => { return ( <div> <a href={href}> {fileName} <FileDate /> </a> </div> );};const FileList = () => { return ( <ul> <li> <File fileName="File one" href="/file/file_one" /> </li> <li> <File fileName="File two" href="/file/file_two" /> </li> <li> <File fileName="File three" href="/file/file_three" /> </li> </ul> );};
@Component({ selector: "file-item", imports: [FileDateComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <a [attr.href]="href()">{{ fileName() }}<file-date /></a> </div> `,})class FileComponent { fileName = input.required<string>(); href = input.required<string>();}@Component({ selector: "file-list", imports: [FileComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <ul> <li><file-item [fileName]="'File one'" [href]="'/file/file_one'" /></li> <li><file-item [fileName]="'File two'" [href]="'/file/file_two'" /></li> <li> <file-item [fileName]="'File three'" [href]="'/file/file_three'" /> </li> </ul> `,})class FileListComponent {}
<!-- File.vue --><script setup>import FileDate from "./FileDate.vue";const props = defineProps(["fileName", "href"]);</script><template> <div> <a :href="props.href">{{ props.fileName }}<FileDate /></a> </div></template>
<!-- FileList.vue --><script setup>import File from "./File.vue";</script><template> <ul> <li><File :fileName="'File one'" :href="'/file/file_one'" /></li> <li><File :fileName="'File two'" :href="'/file/file_two'" /></li> <li><File :fileName="'File three'" :href="'/file/file_three'" /></li> </ul></template>
Object Passing
While we've been using strings to pass values to a component as an input, this isn't always the case.
Input properties can be of any JavaScript type. This can include objects, strings, numbers, arrays, class instances, or anything in between!
To showcase this, let's add the ability to pass a Date
class instance to our file-date
component. After all, each file in the list of our files will likely be created at different times.
- React
- Angular
- Vue
const FileDate = ({ inputDate }) => { const [dateStr, setDateStr] = useState(formatDate(inputDate)); const [labelText, setLabelText] = useState(formatReadableDate(inputDate)); // ... return <span aria-label={labelText}>{dateStr}</span>;};const File = ({ href, fileName }) => { return ( <div> <a href={href}> {fileName} <FileDate inputDate={new Date()} /> </a> </div> );};
import { Component, ChangeDetectionStrategy, input, signal, afterRender } from "@angular/core";@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span [attr.aria-label]="labelText()">{{ dateStr() }}</span>`,})class FileDateComponent { inputDate = input.required<Date>(); /** * You cannot access `input` data from the root (constructor) * of the class */ dateStr = signal(""); labelText = signal(""); constructor() { afterRender(() => { this.dateStr.set(formatDate(this.inputDate())); this.labelText.set(formatReadableDate(this.inputDate())); }); } // ...}@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileDateComponent], template: ` <div> <a [attr.href]="href"> {{ fileName }} <file-date [inputDate]="inputDate" /> </a> </div> `,})class FileComponent { inputDate = new Date(); // ...}
You'll notice that we had to move the logic to set the dateStr
and labelText
values to the afterRender
method. This tells Angular that after its first render, it should update the values of the signals based on the input
method.
This is because Angular doesn't allow you to access input
values in the root (AKA the "constructor") of a component's class.
If you're unfamiliar with what a class constructor is and how it associates with root-level class properties, I'd suggest reading through this guide I wrote about using JavaScript classes without the
class
keyword
<!-- FileDate.vue --><script setup>// ...const props = defineProps(["inputDate"]);const dateStr = ref(formatDate(props.inputDate));const labelText = ref(formatReadableDate(props.inputDate));// ...</script><template> <span :aria-label="labelText">{{ dateStr }}</span></template>
<!-- File.vue --><script setup>import FileDate from "./FileDate.vue";const props = defineProps(["fileName", "href"]);const inputDate = new Date();</script><template> <div> <a :href="props.href" >{{ props.fileName }} <FileDate :inputDate="inputDate" /> </a> </div></template>
Once again, I have to add a minor asterisk next to this code sample. Right now, if you update the
inputDate
value after the initial render, it will not show the new date string inFileDate
. This is because we're setting the value ofdateStr
andlabelText
only once and not updating the values.Each framework has a way of live-updating this value for us, as we might usually expect, by using a derived value, but we'll touch on that in a future section.
Props Rules
While it's true that a component property can be passed a JavaScript object, there's a rule you must follow when it comes to object props:
You must not mutate component prop values.
For example, here's some code that will not work as expected:
- React
- Angular
- Vue
const GenericList = ({ inputArray }) => { // This is NOT allowed and will break things inputArray.push("some value"); // ...};
@Component({ selector: "generic-list", // ...})class GenericListComponent { inputArray = input.required<string[]>(); constructor() { afterRender(() => { // This is NOT allowed and will break things this.inputArray().push("some value"); }) } // ...}
<!-- GenericList.vue --><script setup>import { onMounted } from "vue";const props = defineProps(["inputArray"]);onMounted(() => { // This is NOT allowed and will break things props.inputArray.push("some value");});</script><!-- ... -->
You're not intended to mutate properties because it breaks two key concepts of application architecture with components:
Event Binding
Binding values to an HTML attribute is a powerful way to control your UI, but that's only half the story. Showing information to the user is one thing, but you must also react to a user's input.
One way you can do this is by binding a DOM event emitted by a user's behavior.
In the mockup we saw before, the list of our files has a hover state for the file list. However, when the user clicks on a file, it should be highlighted more distinctly.
Let's add an isSelected
property to our file
component to add hover styling conditionally, then update it when the user clicks on it.
While we're at it, let's migrate our File
component to use a button
instead of a div
. After all, it's important for accessibility and SEO to use semantic elements to indicate what element is which in the DOM.
- React
- Angular
- Vue
const File = ({ fileName }) => { const [isSelected, setSelected] = useState(false); const selectFile = () => { setSelected(!isSelected); }; return ( <button onClick={selectFile} style={ isSelected ? { backgroundColor: "blue", color: "white" } : { backgroundColor: "white", color: "blue" } } > {fileName} <FileDate inputDate={new Date()} /> </button> );};
There are three major things of note in this code sample:
-
The
style
property differs from what you might see in a typical HTML code sample.EG:
"background-color: blue; color: white;"
becomes{backgroundColor: 'blue', color: 'white'}
.This is required to embed CSS directly inside JSX's
style
property. If you move this code out to a dedicated CSS file and use classes, you can use the more traditional syntax. -
We're using the second value of the
useState
returned array.The second value of the array returned by
useState
is used to update the value assigned to the first variable. So, whensetSelected
is called, it will then update the value ofisSelected
, and the component is re-rendered.
-
We prefix the event name we're listening for with
on
and capitalize the first letter of the event name.EG:
click
becomesonClick
.
@Component({ selector: "file-item", imports: [FileDateComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: ` <button (click)="selectFile()" [style]=" isSelected() ? 'background-color: blue; color: white' : 'background-color: white; color: blue' " > {{ fileName() }} <file-date [inputDate]="inputDate" /> </button> `,})class FileComponent { fileName = input.required<string>(); inputDate = new Date(); isSelected = signal(false); selectFile() { this.isSelected.set(!this.isSelected()); }}
Instead of the []
symbols to do input binding, we're using the ()
symbols to bind to any built-in browser event name.
<!-- File.vue --><script setup>import { ref } from "vue";const props = defineProps(["fileName"]);const inputDate = new Date();const isSelected = ref(false);function selectFile() { isSelected.value = !isSelected.value;}</script><template> <button v-on:click="selectFile()" :style=" isSelected ? 'background-color: blue; color: white' : 'background-color: white; color: blue' " > {{ props.fileName }} <FileDate :inputDate="inputDate" /> </button></template>
Here, we're binding the style
property using Vue's binding. You may notice that when binding via style
, you use an object notation for styling instead of the usual string.
We're also using a ternary statement (condition ? trueVal : falseVal
) to act as a single-line if
statement to decide which style to use.
We can use v-on
bind prefix to bind a method to any event. This supports any built-in browser event name.
There's also a shorthand syntax, just like there is one for attribute bindings. Instead of v-on:
, we can use the @
symbol.
This means:
<button v-on:click="selectFile()"></button>
Can be rewritten into:
<button @click="selectFile()"></button>
Outputs
Components aren't limited to only being able to receive a value from its parent; You're also able to send values back to the parent from the child component.
The way this usually works is by passing data upwards via a custom event, much like those emitted by the browser. Similar to how our event binding used some new syntax along with familiar concepts, we'll do the same with event emitting.
While listening for a click
event in our File
component works well enough when we only have one file, it introduces some odd behavior with multiple files. Namely, it allows us to select more than one file at a time simply by clicking. Let's assume this isn't the expected behavior and instead emit a selected
custom event to allow for only one selected file at a time.
- React
- Angular
- Vue
React expects you to pass in a function as opposed to emitting an event and listening for it.
This differs slightly from Vue and Angular but has the same fundamental idea of "sending data to a parent component."
import { useState } from "react";const File = ({ href, fileName, isSelected, onSelected }) => { // `href` is temporarily unused return ( <button onClick={onSelected} style={ isSelected ? { backgroundColor: "blue", color: "white" } : { backgroundColor: "white", color: "blue" } } > {fileName} {/* ... */} </button> );};const FileList = () => { const [selectedIndex, setSelectedIndex] = useState(-1); const onSelected = (idx) => { if (selectedIndex === idx) { setSelectedIndex(-1); return; } setSelectedIndex(idx); }; return ( <ul> <li> <File isSelected={selectedIndex === 0} onSelected={() => onSelected(0)} fileName="File one" href="/file/file_one" /> </li> <li> <File isSelected={selectedIndex === 1} onSelected={() => onSelected(1)} fileName="File two" href="/file/file_two" /> </li> <li> <File isSelected={selectedIndex === 2} onSelected={() => onSelected(2)} fileName="File three" href="/file/file_three" /> </li> </ul> );};
Angular provides us a simple output
method that enables us to emit()
events from a child component up to the parent. This is fairly similar to how we pass in data using an input
method.
import { Component, ChangeDetectionStrategy, signal, input, output } from "@angular/core";@Component({ selector: "file-item", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileDateComponent], template: ` <button (click)="selected.emit()" [style]=" isSelected() ? 'background-color: blue; color: white' : 'background-color: white; color: blue' " > {{ fileName() }} <!-- ... --> </button> `,})class FileComponent { fileName = input.required<string>(); // `href` is temporarily unused href = input.required<string>(); isSelected = input<boolean>(false); selected = output(); // ...}@Component({ selector: "file-list", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileComponent], template: ` <ul> <li> <file-item (selected)="onSelected(0)" [isSelected]="selectedIndex() === 0" fileName="File one" href="/file/file_one" /> </li> <li> <file-item (selected)="onSelected(1)" [isSelected]="selectedIndex() === 1" fileName="File two" href="/file/file_two" /> </li> <li> <file-item (selected)="onSelected(2)" [isSelected]="selectedIndex() === 2" fileName="File three" href="/file/file_three" /> </li> </ul> `,})class FileListComponent { selectedIndex = signal(-1); onSelected(idx: number) { if (this.selectedIndex() === idx) { this.selectedIndex.set(-1); return; } this.selectedIndex.set(idx); }}
Vue introduces the idea of an emitted event using the defineEmits
global function:
<!-- File.vue --><script setup>// ...// `href` is temporarily unusedconst props = defineProps(["isSelected", "fileName", "href"]);const emit = defineEmits(["selected"]);</script><template> <button v-on:click="emit('selected')" :style=" isSelected ? 'background-color: blue; color: white' : 'background-color: white; color: blue' " > {{ fileName }} <!-- ... --> </button></template>
The
defineEmits
function does not need to be imported fromvue
, since Vue's compiler handles that for us.
<!-- FileList.vue --><script setup>import { ref } from "vue";import File from "./File.vue";const selectedIndex = ref(-1);function onSelected(idx) { if (selectedIndex.value === idx) { selectedIndex.value = -1; return; } selectedIndex.value = idx;}</script><template> <ul> <li> <File @selected="onSelected(0)" :isSelected="selectedIndex === 0" fileName="File one" href="/file/file_one" /> </li> <li> <File @selected="onSelected(1)" :isSelected="selectedIndex === 1" fileName="File two" href="/file/file_two" /> </li> <li> <File @selected="onSelected(2)" :isSelected="selectedIndex === 2" fileName="File three" href="/file/file_three" /> </li> </ul></template>
Keep in mind: This code isn't quite production-ready. There are some accessibility concerns with this code and might require things like
aria-selected
and more to fix.
Here, we're using a simple number-based index to act as an id
of sorts for each file. This allows us to keep track of which file is currently selected or not. Likewise, if the user selects an index that's already been selected, we will set the isSelected
index to a number that no file has associated.
You may notice that we've also removed our isSelected
state and logic from our file
component. This is because we're following the practices of "raising state".
Challenge
Now that we have a solid grasp on the fundamentals of components, let's build some ourselves!
Namely, I want us to create a primitive version of the following:
To do this, let's:
- Create a sidebar component
- Add a list of buttons with sidebar items' names
- Make an
ExpandableDropdown
component - Add a
name
input to the dropdown and display it - Add an
expanded
input to the dropdown and display it - Use an output to toggle the
expanded
input - Make our
expanded
property functional
Creating Our First Components
Let's kick off this process by creating our index.html
and a basic component to render:
- React
- Angular
- Vue
<!-- index.html --><html> <body> <div id="root"></div> </body></html>
import { createRoot } from "react-dom";const Sidebar = () => { return <p>Hello, world!</p>;};createRoot(document.getElementById("root")).render(<Sidebar />);
<!-- index.html --><html> <body> <app-sidebar></app-sidebar> </body></html>
import { Component, ChangeDetectionStrategy } from "@angular/core";import { bootstrapApplication } from "@angular/platform-browser";@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <p>Hello, world!</p> `,})class SidebarComponent {}bootstrapApplication(SidebarComponent);
<!-- index.html --><html> <body> <div id="root"></div> </body></html>
// main.jsimport { createApp } from "vue";import Sidebar from "./Sidebar.vue";createApp(Sidebar).mount("#root");
<!-- Sidebar.vue --><template> <p>Hello, world!</p></template>
Now that we have an initial testbed for our component let's add a list of buttons with the names of the sidebar list items:
- React
- Angular
- Vue
const Sidebar = () => { return ( <div> <h1>My Files</h1> <div> <button>Movies</button> </div> <div> <button>Pictures</button> </div> <div> <button>Concepts</button> </div> <div> <button>Articles I'll Never Finish</button> </div> <div> <button>Website Redesigns v5</button> </div> <div> <button>Invoices</button> </div> </div> );};
@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <h1>My Files</h1> <div><button>Movies</button></div> <div><button>Pictures</button></div> <div><button>Concepts</button></div> <div><button>Articles I'll Never Finish</button></div> <div><button>Website Redesigns v5</button></div> <div><button>Invoices</button></div> </div> `,})class SidebarComponent {}
<!-- Sidebar.vue --><template> <div> <h1>My Files</h1> <div><button>Movies</button></div> <div><button>Pictures</button></div> <div><button>Concepts</button></div> <div><button>Articles I'll Never Finish</button></div> <div><button>Website Redesigns v5</button></div> <div><button>Invoices</button></div> </div></template>
This repeated div
and button
combo makes me think that we should extract each of these items to a component since we want to both:
- Reuse the HTML layout
- Expand the current functionality
Start by extracting the div
and button
to their own component, which we'll call ExpandableDropdown
.
- React
- Angular
- Vue
const ExpandableDropdown = ({ name }) => { return ( <div> <button>{name}</button> </div> );};const Sidebar = () => { return ( <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" /> <ExpandableDropdown name="Pictures" /> <ExpandableDropdown name="Concepts" /> <ExpandableDropdown name="Articles I'll Never Finish" /> <ExpandableDropdown name="Website Redesigns v5" /> <ExpandableDropdown name="Invoices" /> </div> );};
@Component({ selector: "expandable-dropdown", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <button> {{ name() }} </button> </div> `,})class ExpandableDropdownComponent { name = input.required<string>();}@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, imports: [ExpandableDropdownComponent], template: ` <div> <h1>My Files</h1> <expandable-dropdown name="Movies" /> <expandable-dropdown name="Pictures" /> <expandable-dropdown name="Concepts" /> <expandable-dropdown name="Articles I'll Never Finish" /> <expandable-dropdown name="Website Redesigns v5" /> <expandable-dropdown name="Invoices" /> </div> `,})class SidebarComponent {}
<!-- Sidebar.vue --><script setup>import ExpandableDropdown from "./ExpandableDropdown.vue";</script><template> <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" /> <ExpandableDropdown name="Pictures" /> <ExpandableDropdown name="Concepts" /> <ExpandableDropdown name="Articles I'll Never Finish" /> <ExpandableDropdown name="Website Redesigns v5" /> <ExpandableDropdown name="Invoices" /> </div></template>
<!-- ExpandableDropdown.vue --><script setup>const props = defineProps(["name"]);</script><template> <div> <button> {{ props.name }} </button> </div></template>
We should now see a list of buttons with a name associated with each!
Making Our Components Functional
Now that we've created the initial structure of our components, let's work on making them functional.
To start, we'll:
- Create an
expanded
property for each button - Pass the
expanded
property using an input - Display the value of
expanded
inside of ourExpandableDropdown
component
- React
- Angular
- Vue
const ExpandableDropdown = ({ name, expanded }) => { return ( <div> <button>{name}</button> <div>{expanded ? "Expanded" : "Collapsed"}</div> </div> );};const Sidebar = () => { // Just to show that the value is displaying properly const [moviesExpanded, setMoviesExpanded] = useState(true); const [picturesExpanded, setPicturesExpanded] = useState(false); const [conceptsExpanded, setConceptsExpanded] = useState(false); const [articlesExpanded, setArticlesExpanded] = useState(false); const [redesignExpanded, setRedesignExpanded] = useState(false); const [invoicesExpanded, setInvoicesExpanded] = useState(false); return ( <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" expanded={moviesExpanded} /> <ExpandableDropdown name="Pictures" expanded={picturesExpanded} /> <ExpandableDropdown name="Concepts" expanded={conceptsExpanded} /> <ExpandableDropdown name="Articles I'll Never Finish" expanded={articlesExpanded} /> <ExpandableDropdown name="Website Redesigns v5" expanded={redesignExpanded} /> <ExpandableDropdown name="Invoices" expanded={invoicesExpanded} /> </div> );};
@Component({ selector: "expandable-dropdown", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <button> {{ name() }} </button> <div> {{ expanded() ? "Expanded" : "Collapsed" }} </div> </div> `,})class ExpandableDropdownComponent { name = input.required<string>(); expanded = input.required<boolean>();}@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, imports: [ExpandableDropdownComponent], template: ` <div> <h1>My Files</h1> <expandable-dropdown name="Movies" [expanded]="moviesExpanded()" /> <expandable-dropdown name="Pictures" [expanded]="picturesExpanded()" /> <expandable-dropdown name="Concepts" [expanded]="conceptsExpanded()" /> <expandable-dropdown name="Articles I'll Never Finish" [expanded]="articlesExpanded()" /> <expandable-dropdown name="Website Redesigns v5" [expanded]="redesignExpanded()" /> <expandable-dropdown name="Invoices" [expanded]="invoicesExpanded()" /> </div> `,})class SidebarComponent { // Just to show that the value is displaying properly moviesExpanded = signal(true); picturesExpanded = signal(false); conceptsExpanded = signal(false); articlesExpanded = signal(false); redesignExpanded = signal(false); invoicesExpanded = signal(false);}
<!-- Sidebar.vue --><script setup>import { ref } from "vue";import ExpandableDropdown from "./ExpandableDropdown.vue";// Just to show that the value is displaying properlyconst moviesExpanded = ref(true);const picturesExpanded = ref(false);const conceptsExpanded = ref(false);const articlesExpanded = ref(false);const redesignExpanded = ref(false);const invoicesExpanded = ref(false);</script><template> <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" :expanded="moviesExpanded" /> <ExpandableDropdown name="Pictures" :expanded="picturesExpanded" /> <ExpandableDropdown name="Concepts" :expanded="conceptsExpanded" /> <ExpandableDropdown name="Articles I'll Never Finish" :expanded="articlesExpanded" /> <ExpandableDropdown name="Website Redesigns v5" :expanded="redesignExpanded" /> <ExpandableDropdown name="Invoices" :expanded="invoicesExpanded" /> </div></template>
<!-- ExpandableDropdown.vue --><script setup>const props = defineProps(["name", "expanded"]);</script><template> <div> <button> {{ props.name }} </button> <div> {{ props.expanded ? "Expanded" : "Collapsed" }} </div> </div></template>
Remember to add our new
expanded
property name inside ofdefineProps
! Otherwise, this component won't bind the value correctly.
Let's now add an output to allow our component to toggle the expanded
input.
- React
- Angular
- Vue
const ExpandableDropdown = ({ name, expanded, onToggle }) => { return ( <div> <button onClick={onToggle}>{name}</button> <div>{expanded ? "Expanded" : "Collapsed"}</div> </div> );};const Sidebar = () => { // Just to show that the value is displaying properly const [moviesExpanded, setMoviesExpanded] = useState(true); const [picturesExpanded, setPicturesExpanded] = useState(false); const [conceptsExpanded, setConceptsExpanded] = useState(false); const [articlesExpanded, setArticlesExpanded] = useState(false); const [redesignExpanded, setRedesignExpanded] = useState(false); const [invoicesExpanded, setInvoicesExpanded] = useState(false); return ( <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" expanded={moviesExpanded} onToggle={() => setMoviesExpanded(!moviesExpanded)} /> <ExpandableDropdown name="Pictures" expanded={picturesExpanded} onToggle={() => setPicturesExpanded(!picturesExpanded)} /> <ExpandableDropdown name="Concepts" expanded={conceptsExpanded} onToggle={() => setConceptsExpanded(!conceptsExpanded)} /> <ExpandableDropdown name="Articles I'll Never Finish" expanded={articlesExpanded} onToggle={() => setArticlesExpanded(!articlesExpanded)} /> <ExpandableDropdown name="Website Redesigns v5" expanded={redesignExpanded} onToggle={() => setRedesignExpanded(!redesignExpanded)} /> <ExpandableDropdown name="Invoices" expanded={invoicesExpanded} onToggle={() => setInvoicesExpanded(!invoicesExpanded)} /> </div> );};
@Component({ selector: "expandable-dropdown", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <button (click)="toggle.emit()"> {{ name() }} </button> <div> {{ expanded() ? "Expanded" : "Collapsed" }} </div> </div> `,})class ExpandableDropdownComponent { name = input.required<string>(); expanded = input.required<boolean>(); toggle = output();}@Component({ selector: "app-sidebar", changeDetection: ChangeDetectionStrategy.OnPush, imports: [ExpandableDropdownComponent], template: ` <div> <h1>My Files</h1> <expandable-dropdown name="Movies" [expanded]="moviesExpanded()" (toggle)="moviesExpanded.set(!moviesExpanded())" /> <expandable-dropdown name="Pictures" [expanded]="picturesExpanded()" (toggle)="picturesExpanded.set(!picturesExpanded())" /> <expandable-dropdown name="Concepts" [expanded]="conceptsExpanded()" (toggle)="conceptsExpanded.set(!conceptsExpanded())" /> <expandable-dropdown name="Articles I'll Never Finish" [expanded]="articlesExpanded()" (toggle)="articlesExpanded.set(!articlesExpanded())" /> <expandable-dropdown name="Website Redesigns v5" [expanded]="redesignExpanded()" (toggle)="redesignExpanded.set(!redesignExpanded())" /> <expandable-dropdown name="Invoices" [expanded]="invoicesExpanded()" (toggle)="invoicesExpanded.set(!invoicesExpanded())" /> </div> `,})class SidebarComponent { moviesExpanded = signal(true); picturesExpanded = signal(false); conceptsExpanded = signal(false); articlesExpanded = signal(false); redesignExpanded = signal(false); invoicesExpanded = signal(false);}
<!-- Sidebar.vue --><script setup>import { ref } from "vue";import ExpandableDropdown from "./ExpandableDropdown.vue";// Just to show that the value is displaying properlyconst moviesExpanded = ref(true);const picturesExpanded = ref(false);const conceptsExpanded = ref(false);const articlesExpanded = ref(false);const redesignExpanded = ref(false);const invoicesExpanded = ref(false);</script><template> <div> <h1>My Files</h1> <ExpandableDropdown name="Movies" :expanded="moviesExpanded" @toggle="moviesExpanded = !moviesExpanded" /> <ExpandableDropdown name="Pictures" :expanded="picturesExpanded" @toggle="picturesExpanded = !picturesExpanded" /> <ExpandableDropdown name="Concepts" :expanded="conceptsExpanded" @toggle="conceptsExpanded = !conceptsExpanded" /> <ExpandableDropdown name="Articles I'll Never Finish" :expanded="articlesExpanded" @toggle="articlesExpanded = !articlesExpanded" /> <ExpandableDropdown name="Website Redesigns v5" :expanded="redesignExpanded" @toggle="redesignExpanded = !redesignExpanded" /> <ExpandableDropdown name="Invoices" :expanded="invoicesExpanded" @toggle="invoicesExpanded = !invoicesExpanded" /> </div></template>
<!-- ExpandableDropdown.vue --><script setup>const props = defineProps(["name", "expanded"]);const emit = defineEmits(["toggle"]);</script><template> <div> <button @click="emit('toggle')"> {{ props.name }} </button> <div> {{ props.expanded ? "Expanded" : "Collapsed" }} </div> </div></template>
Finally, we can update our ExpandableDropdown
component to hide and show the contents of the dropdown by using an HTML attribute called "hidden". When this attribute is true
, it will hide the contents, but when false
, it will display them.
- React
- Angular
- Vue
const ExpandableDropdown = ({ name, expanded, onToggle }) => { return ( <div> <button onClick={onToggle}> {expanded ? "V " : "> "} {name} </button> <div hidden={!expanded}>More information here</div> </div> );};
Final code output
@Component({ selector: "expandable-dropdown", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <button (click)="toggle.emit()"> {{ expanded() ? "V" : ">" }} {{ name() }} </button> <div [hidden]="!expanded()">More information here</div> </div> `,})class ExpandableDropdownComponent { name = input.required<string>(); expanded = input.required<boolean>(); toggle = output();}
Final code output
<!-- ExpandableDropdown.vue --><script setup>const props = defineProps(["name", "expanded"]);const emit = defineEmits(["toggle"]);</script><template> <div> <button @click="emit('toggle')"> {{ expanded ? "V" : ">" }} {{ name }} </button> <div :hidden="!expanded">More information here</div> </div></template>