Derived Values

March 11, 2024

2,845 words

Post contents

We've previously touched on how to pass values to a component as properties earlier in the book:

const FileDate = ({ inputDate }) => {	const [dateStr, setDateStr] = useState(formatDate(inputDate));	const [labelText, setLabelText] = useState(formatReadableDate(inputDate));	return <span aria-label={labelText}>{dateStr}</span>;};
import { Component, OnInit } from "@angular/core";@Component({	selector: "file-date",	standalone: true,	template: `<span [attr.aria-label]="labelText">{{ dateStr }}</span>`,})class FileDateComponent implements OnInit {	@Input() inputDate!: Date;	dateStr = "";	labelText = "";	ngOnInit() {		this.dateStr = this.formatDate(this.inputDate);		this.labelText = this.formatReadableDate(this.inputDate);	}	// ...}
<!-- 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>

You may notice that we're deriving two values from the same property. This works fine at first, but an issue arises with how we're doing things when we realize that our formatDate and formatReadableDate methods are only running once during the initial render.

Because of this, if we pass in an updated inputDate to the FileDate component, the values of formatDate and formatReadableDate will become out of sync from the parent's passed inputDate.

const File = ({ href, fileName, isSelected, onSelected, isFolder }) => {	const [inputDate, setInputDate] = useState(new Date());	useEffect(() => {		// Check if it's a new day every 10 minutes		const timeout = setTimeout(			() => {				const newDate = new Date();				if (inputDate.getDate() === newDate.getDate()) return;				setInputDate(newDate);			},			10 * 60 * 1000,		);		return () => clearTimeout(timeout);	}, [inputDate]);	// JSX shortened for focus	// This may not show the most up-to-date `formatDate` or `formatReadableDate`	return <FileDate inputDate={inputDate} />;};
@Component({	selector: "file-item",	standalone: true,	imports: [FileDateComponent, NgIf],	template: `		<!-- ... -->		<!-- This may not show the most up-to-date 'formatDate' or 'formatReadableDate' -->		<file-date *ngIf="!isFolder" [inputDate]="inputDate" />		<!-- ... -->	`,})class FileComponent implements OnInit, OnDestroy {	// ...	inputDate = new Date();	interval: any = null;	ngOnInit() {		// Check if it's a new day every 10 minutes		this.interval = setInterval(			() => {				const newDate = new Date();				if (this.inputDate.getDate() === newDate.getDate()) return;				this.inputDate = newDate;			},			10 * 60 * 1000,		);	}	ngOnDestroy() {		clearInterval(this.interval);	}}
<!-- File.vue --><script setup>import { ref, onMounted, onUnmounted } from "vue";import FileDate from "./FileDate.vue";// ...const inputDate = ref(new Date());const interval = ref(null);onMounted(() => {	// Check if it's a new day every 10 minutes	interval.value = setInterval(		() => {			const newDate = new Date();			if (inputDate.value.getDate() === newDate.getDate()) return;			inputDate.value = newDate;		},		10 * 60 * 1000,	);});onUnmounted(() => {	clearInterval(interval.value);});</script><template>	<!-- ... -->	<!-- This may not show the most up-to-date `formatDate` or `formatReadableDate` -->	<FileDate v-if="isFolder" :inputDate="inputDate" />	<!-- ... --></template>

While the above File component updates inputDate correctly, our FileDate component is never listening for the changed input value and, as such, never recomputed the formatDate or formatReadableDate value to display to the user.

How can we fix this?

Method 1: Prop Listening

The first - and arguably easiest to mentally model - method to solve this disparity between prop value and display value is to simply listen for when a property's value has been updated and re-calculate the display value.

Luckily, we can use our existing knowledge of side effects to do so:

const FileDate = ({ inputDate }) => {	const [dateStr, setDateStr] = useState(formatDate(inputDate));	const [labelText, setLabelText] = useState(formatReadableDate(inputDate));	useEffect(() => {		setDateStr(formatDate(inputDate));		setLabelText(formatReadableDate(inputDate));		// Every time `inputDate` changes, it'll trigger a render and therefore call the `useEffect`	}, [inputDate]);	return <span aria-label={labelText}>{dateStr}</span>;};

While we didn't touch on this lifecycle method in our previous chapter, Angular has a lifecycle method specifically for when a component's props change: ngOnChanges.

We can use this new lifecycle method to update the value of a component's state based off of the parent's props:

import { Component, OnChanges, SimpleChanges } from "@angular/core";@Component({	selector: "file-date",	standalone: true,	template: `<span [attr.aria-label]="labelText">{{ dateStr }}</span>`,})class FileDateComponent implements OnChanges {	@Input() inputDate: Date;	dateStr = "";	labelText = "";	// Notice that we no longer need `ngOnInit`	ngOnChanges(changes: SimpleChanges) {		/**		 * ngOnChanges runs for EVERY prop change. As such, we can		 * restrict the recalculation to only when `inputDate` changes		 */		if (changes["inputDate"]) {			this.dateStr = this.formatDate(this.inputDate);			this.labelText = this.formatReadableDate(this.inputDate);		}	}	// ...}
<!-- FileDate.vue --><script setup>import { ref, watch } from "vue";// ...const props = defineProps(["inputDate"]);const dateStr = ref(formatDate(props.inputDate));const labelText = ref(formatReadableDate(props.inputDate));watch(	() => props.inputDate,	(newDate, oldDate) => {		dateStr.value = formatDate(newDate);		labelText.value = formatReadableDate(newDate);	},);</script><template>	<span :aria-label="labelText">{{ dateStr }}</span></template>

Vue's watch logic allows you to track a given property or state value changes based on its key.

Here, we're watching the inputDate props key and, when changed, updating dateStr and labelText based off of the new property value.

While this method works, it tends to introduce duplicate developmental logic. For example, notice how we have to repeat the declaration of the dateStr and labelText values twice: Once when they're initially defined and again inside the property listener.

Luckily for us, there's an easy solution for this problem called "computed values."

Method 2: Computed Values

Our previous method of deriving a value from a property follows two steps:

  1. Set an initial value
  2. Update and recompute the value when its base changes

However, what if we could instead simplify this idea to a single step:

  1. Run a function over a value and live update as it changes.

This may remind you of a similar pattern we've used already for live updated text and attribute binding.

Luckily for us, all three frameworks have a way of doing just this!

import { useMemo } from "react";const FileDate = ({ inputDate }) => {	const dateStr = useMemo(() => formatDate(inputDate), [inputDate]);	const labelText = useMemo(() => formatReadableDate(inputDate), [inputDate]);	return <span aria-label={labelText}>{dateStr}</span>;};

useMemo is a method for computing values based on an input or series of inputs. This works because it does the computation and regenerates the calculation whenever the second argument array's values have changed in a render.

Like useEffect, this array's values' changes are only tracked when the component is rendered. Unlike useEffect, however, there's no option to remove the second array argument entirely.

Instead, if you want to recalculate the logic in every render, you'd remove the useMemo entirely. So, for simple computations, you can take this code:

const AddComp = ({ baseNum, addNum }) => {	const val = useMemo(() => baseNum + addNum, [baseNum, addNum]);	return <p>{val}</p>;};

And refactor it to look like this:

const AddComp = ({ baseNum, addNum }) => {	const val = baseNum + addNum;	return <p>{val}</p>;};

While it's technically possible to use this trick to never use useMemo, your application's performance will suffer drastically. That said, it's a bit of a science to know when and where to use useMemo. We'll touch on this more in our third book titled "Internals".

To solve the derived value problem without recomputing the values manually, Angular introduces the concept of a "pipe" into the mix of things. The idea is that a pipe runs over an input (or series of inputs), just like React's useMemo.

import { Pipe, PipeTransform } from "@angular/core";@Pipe({ name: "formatDate", standalone: true })class FormatDatePipe implements PipeTransform {	transform(value: Date): string {		return formatDate(value);	}}@Pipe({ name: "formatReadableDate", standalone: true })class FormatReadableDatePipe implements PipeTransform {	transform(value: Date): string {		return formatReadableDate(value);	}}

You may then use these pipes in your components directly inside the template.

@Component({	selector: "file-date",	standalone: true,	imports: [FormatReadableDatePipe, FormatDatePipe],	template: `		<span [attr.aria-label]="inputDate | formatReadableDate">			{{ inputDate | formatDate }}		</span>	`,})class FileDateComponent {	@Input() inputDate!: Date;}
Multiple Input Pipes

You may notice the similarities between pipes and functions. After all, pipes are effectively functions you're able to call in your template. Much like functions, they're not limited to a single input property, either.

Let's add a second input to see if the formatDate pipe should return a readable date or not.

@Pipe({ name: "formatDate", standalone: true })class FormatDatePipe implements PipeTransform {	// `dateFormat` is an optional argument. If left empty, will simply `formatDate`	transform(value: Date, dateFormat?: string): string {		// Stands for "Long format month, day of month, year"		if (dateFormat === "MMMM d, Y") return formatReadableDate(value);		return formatDate(value);	}}

Then, we can use it in our template while passing a second argument:

@Component({	selector: "file-date",	standalone: true,	imports: [FormatDatePipe],	template: `		<span [attr.aria-label]="inputDate | formatDate: 'MMMM d, Y'">			{{ inputDate | formatDate }}		</span>	`,})class FileDateComponent {	@Input() inputDate: Date;}
Built-In Pipes

Luckily, Angular's all-in-one methodology means that there's a slew of pipes that the Angular team has written for us. One such pipe is actually a date formatting pipe. We can remove our own implementation in favor of one built right into Angular!

To use the built-in pipes, we need to import them from CommonModule into the component. In this case, the pipe we're looking to use is called DatePipe. This provided date pipe is, expectedly, called date when used in the template and can be used like so:

import { DatePipe } from "@angular/common";@Component({	selector: "file-date",	standalone: true,	imports: [DatePipe],	template: `		<span [attr.aria-label]="inputDate | date: 'MMMM d, Y'">			{{ inputDate | date }}		</span>	`,})class FileDateComponent {	@Input() inputDate!: Date;}
<!-- FileDate.vue --><script setup>import { computed } from "vue";// ...const props = defineProps(["inputDate"]);const dateStr = computed(() => formatDate(props.inputDate));const labelText = computed(() => formatReadableDate(props.inputDate));</script><template>	<span :aria-label="labelText">{{ dateStr }}</span></template>

Instead of using ref to construct a set of variables and then re-initializing the values once we watch a prop, we can simply tell Vue to do that same process for us using computed props.

Vue is able to ✨ magically ✨ detect what data inside the computed function is dynamic, just like watchEffect. When this dynamic data is changed, it will automatically re-initialize the variable it's assigned to with a new value returned from the inner function.

These computed props are then accessible in the same way a data property is, both from the template and from Vue's <script> alike.

Non-Prop Derived Values

While we've primarily used component inputs to demonstrate derived values today, both of the methods we've used thus far work for the internal component state and inputs.

Let's say that we have a piece of state called number in our component and want to display the doubled value of this property without passing this state to a new component:

const CountAndDoubleComp = () => {	const [number, setNumber] = useState(0);	const doubleNum = useMemo(() => number * 2, [number]);	return (		<div>			<p>{number}</p>			<p>{doubleNum}</p>			<button onClick={() => setNumber(number + 2)}>Add one</button>		</div>	);};
@Pipe({ name: "doubleNum", standalone: true })class DoubleNumPipe implements PipeTransform {	transform(value: number): number {		return value * 2;	}}@Component({	selector: "count-and-double",	standalone: true,	imports: [DoubleNumPipe],	template: `		<div>			<p>{{ number }}</p>			<p>{{ number | doubleNum }}</p>			<button (click)="addOne()">Add one</button>		</div>	`,})class CountAndDoubleComponent {	number = 0;	addOne() {		this.number++;	}}
<!-- CountAndDouble.vue --><script setup>import { ref, computed } from "vue";const number = ref(0);function addOne() {	number.value++;}const doubleNum = computed(() => number.value * 2);</script><template>	<div>		<p>{{ number }}</p>		<p>{{ doubleNum }}</p>		<button @click="addOne()">Add one</button>	</div></template>

In this component, we can see two numbers — one doubling the value of the other. We then have a button that allows us to increment the first number, and therefore, using a derived value, the second number also updates.

Challenge

While building through our continued file hosting application, let's think through how our Size can be calculated to be displayed in the UI like so:

A table of files and folders with "Name", "LAst modified", "Type", and "Size" headings

File sizes are usually measured in how many bytes it takes to store the file. However, this isn't exactly useful information past a certain size. Let's instead use the following JavaScript to figure out how large a file size is, given the number of bytes:

const kilobyte = 1024;const megabyte = kilobyte * 1024;const gigabyte = megabyte * 1024;function formatBytes(bytes) {	if (bytes < kilobyte) {		return `${bytes} B`;	} else if (bytes < megabyte) {		return `${Math.floor(bytes / kilobyte)} KB`;	} else if (bytes < gigabyte) {		return `${Math.floor(bytes / megabyte)} MB`;	} else {		return `${Math.floor(bytes / gigabyte)} GB`;	}}

Fun code challenge for you at home — can you write the above in fewer lines of code? 🤔

With this JavaScript, we can use a derived value to display the relevant display size. Let's build this out using a dedicated DisplaySize component:

function DisplaySize({ bytes }) {	const humanReadableSize = useMemo(() => formatBytes(bytes), [bytes]);	return <p>{humanReadableSize}</p>;}const kilobyte = 1024;const megabyte = kilobyte * 1024;const gigabyte = megabyte * 1024;function formatBytes(bytes) {	if (bytes < kilobyte) {		return `${bytes} B`;	} else if (bytes < megabyte) {		return `${Math.floor(bytes / kilobyte)} KB`;	} else if (bytes < gigabyte) {		return `${Math.floor(bytes / megabyte)} MB`;	} else {		return `${Math.floor(bytes / gigabyte)} GB`;	}}
@Pipe({ name: "formatBytes", standalone: true })class FormatBytesPipe implements PipeTransform {	kilobyte = 1024;	megabyte = this.kilobyte * 1024;	gigabyte = this.megabyte * 1024;	transform(bytes: number): string {		if (bytes < this.kilobyte) {			return `${bytes} B`;		} else if (bytes < this.megabyte) {			return `${Math.floor(bytes / this.kilobyte)} KB`;		} else if (bytes < this.gigabyte) {			return `${Math.floor(bytes / this.megabyte)} MB`;		} else {			return `${Math.floor(bytes / this.gigabyte)} GB`;		}	}}@Component({	selector: "display-size",	standalone: true,	imports: [FormatBytesPipe],	template: `<p>{{ bytes | formatBytes }}</p>`,})class DisplaySizeComponent {	@Input() bytes!: number;}
<!-- DisplaySize.vue --><script setup>import { computed } from "vue";const props = defineProps(["bytes"]);const humanReadableSize = computed(() => formatBytes(props.bytes));const kilobyte = 1024;const megabyte = kilobyte * 1024;const gigabyte = megabyte * 1024;function formatBytes(bytes) {	if (bytes < kilobyte) {		return `${bytes} B`;	} else if (bytes < megabyte) {		return `${Math.floor(bytes / kilobyte)} KB`;	} else if (bytes < gigabyte) {		return `${Math.floor(bytes / megabyte)} MB`;	} else {		return `${Math.floor(bytes / gigabyte)} GB`;	}}</script><template>	<p>{{ humanReadableSize }}</p></template>
Previous articleSide Effects

Subscribe to our newsletter!

Subscribe to our newsletter to get updates on new content we create, events we have coming up, and more! We'll make sure not to spam you and provide good insights to the content we have.