Post contents
We've previously touched on how to pass values to a component as properties earlier in the book:
- 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>;};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span [attr.aria-label]="labelText()">{{ dateStr() }}</span>`,})class FileDateComponent { inputDate = input.required<Date>(); dateStr = signal(""); labelText = signal(""); constructor() { afterRender(() => { this.dateStr.set(formatDate(this.inputDate())); this.labelText.set(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
.
- React
- Angular
- Vue
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", changeDetection: ChangeDetectionStrategy.OnPush, imports: [FileDateComponent], template: ` <!-- ... --> <!-- This may not show the most up-to-date 'formatDate' or 'formatReadableDate' --> @if (!isFolder()) { <file-date [inputDate]="inputDate()" /> } <!-- ... --> `,})class FileComponent { // ... inputDate = signal(new Date()); constructor() { effect((onCleanup) => { // Check if it's a new day every 10 minutes const interval = setInterval( () => { const newDate = new Date(); if (this.inputDate().getDate() === newDate.getDate()) return; this.inputDate.set(newDate); }, 10 * 60 * 1000, ); onCleanup(() => { clearInterval(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:
- React
- Angular
- Vue
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>;};
@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: `<span [attr.aria-label]="labelText()">{{ dateStr() }}</span>`,})class FileDateComponent { inputDate = input.required<Date>(); dateStr = signal(""); labelText = signal(""); constructor() { /** * effect runs for EVERY change of signals read in the effect */ effect(() => { this.dateStr.set(formatDate(this.inputDate())); this.labelText.set(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:
- Set an initial value
- Update and recompute the value when its base changes
However, what if we could instead simplify this idea to a single step:
- 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!
- React
- Angular
- Vue
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 useuseMemo
. We'll touch on this more in our third book titled "Internals".
Angular is able to derive state from a signal using a function to transform the signal being read and a computed
method:
import { Component, input, computed, ChangeDetectionStrategy } from "@angular/core";@Component({ selector: "file-date", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <span [attr.aria-label]="labelText()"> {{ dateStr() }} </span> `,})class FileDateComponent { inputDate = input.required<Date>(); dateStr = computed(() => formatDate(this.inputDate())); labelText = computed(() => formatReadableDate(this.inputDate()));}
Angular has another method of deriving state called "Pipes". You can learn all about Angular pipes in our "Complete guide to Angular pipes" article.
<!-- 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:
- React
- Angular
- Vue
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> );};
@Component({ selector: "count-and-double", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <p>{{ number() }}</p> <p>{{ doubleNum() }}</p> <button (click)="addOne()">Add one</button> </div> `,})class CountAndDoubleComponent { number = signal(0); doubleNum = computed(() => this.number() * 2); addOne() { this.number.set(this.number() + 1); }}
Writable Derived State
What if we wanted to have a bit of derived state that you can write temporary updates to that get reset when you update the base signal?
Luckily for us, we can do this using Angular's linkedSignal
:
import { Component, ChangeDetectionStrategy, linkedSignal, signal,} from '@angular/core';@Component({ selector: 'count-and-double', changeDetection: ChangeDetectionStrategy.OnPush, template: ` <div> <p>{{ number() }}</p> <button (click)="addOne()">Add one</button> <p>{{ doubleNum() }}</p> <button (click)="addOneToDouble()">Add one</button> </div> `,})class CountAndDoubleComponent { number = signal(0); doubleNum = linkedSignal(() => this.number() * 2); addOne() { this.number.set(this.number() + 1); } addOneToDouble() { this.doubleNum.set(this.doubleNum() + 1); }}
Now we can add 1
to doubleNum()
any time we want, but when we update number()
it resets doubleNum()
to the calculated value of number() * 2
.
<!-- 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:
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:
- React
- Angular
- Vue
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`; }}
Final code output
const kilobyte = 1024;const megabyte = kilobyte * 1024;const gigabyte = megabyte * 1024;function formatBytes(bytes: number) { 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`; }}@Component({ selector: "display-size", changeDetection: ChangeDetectionStrategy.OnPush, template: `<p>{{ readableBytes() }}</p>`,})class DisplaySizeComponent { bytes = input.required<number>(); readableBytes = computed(() => formatBytes(this.bytes()));}
Final code output
<!-- 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>