Passing Children

March 11, 2024

4,395 words

Post contents

As we've mentioned before, in the DOM, your HTML elements have a relationship with respect to one another.

For example, the following:

<div>	<ul>		<li>One</li>		<li>Two</li>		<li>Three</li>	</ul></div>

Would construct the following DOM tree:

A top-down DOM tree showing a div at the top, then an ul, then lis beneath the ul, and finally text nodes below the li elements

This is how the DOM constructs nodes as parents and children. Notice how the <li> is distinctly below the <ul> tag rather than a syntax like:

<!-- This isn't correct HTML to do what we want --><div	ul="        li='One'        li='Two'        li='Three'    "/>

While the above looks strange and counter-intuitive, let's look at how we define the same list if each element is a dedicated component using the methods we've created thus far:

const ListItem = ({ name }) => {	return <li>{name}</li>;};const List = () => {	return (		<ul>			<ListItem name="One" />			<ListItem name="Two" />			<ListItem name="Three" />		</ul>	);};const Container = () => {	return (		<div>			<List />		</div>	);};
@Component({	selector: "list-item",	standalone: true,	template: ` <li>{{ name }}</li> `,})class ListItemComponent {	@Input() name!: string;}@Component({	selector: "app-list",	standalone: true,	imports: [ListItemComponent],	template: `		<ul>			<list-item name="One" />			<list-item name="Two" />			<list-item name="Three" />		</ul>	`,})class ListComponent {}@Component({	selector: "app-container",	standalone: true,	imports: [ListComponent],	template: `		<div>			<app-list />		</div>	`,})class ContainerComponent {}
<!-- ListItem.vue --><script setup>const props = defineProps(["name"]);</script><template>	<li>{{ props.name }}</li></template>
<!-- List.vue --><script setup>import ListItem from "./ListItem.vue";</script><template>	<ul>		<ListItem name="One" />		<ListItem name="Two" />		<ListItem name="Three" />	</ul></template>
<!-- Container.vue --><script setup>import List from "./List.vue";</script><template>	<div>		<List />	</div></template>

This is fairly similar to that strange fake nested HTML syntax. The alternative component usage syntax that matches closer to the DOM might otherwise look like this:

<Component>	<OtherComponent /></Component>

This mismatch occurs because if we look at how our components are defined, we're building out our previous components deeply rather than broadly.

A left-to-right chart of a div, then an ul, then back to a vertical top-down chart of li elements with text nodes

This is the difference between building apps with HTML alone and building them with a frontend framework; while the DOM is typically thought of as one-dimensional, there are really two dimensions that are exposed more thoroughly by the framework's ability to construct this tree in a more fine-grained manner.

Let's move the component tree back to being breadth first by using a feature that may sound familiar: Passing children.

Passing Basic Children

Before we explore passing children with our frameworks, let's first think of a potential use case for when we want to do this.

For example, say you want the button to have a "pressed" effect whenever you click on it. Then, when you click on it for a second time, it unclicks. This might look something like the following:

const ToggleButton = ({ text }) => {	const [pressed, setPressed] = useState(false);	return (		<button			onClick={() => setPressed(!pressed)}			style={{				backgroundColor: pressed ? "black" : "white",				color: pressed ? "white" : "black",			}}			type="button"			aria-pressed={pressed}		>			{text}		</button>	);};const ToggleButtonList = () => {	return (		<>			<ToggleButton text="Hello world!" />			<ToggleButton text="Hello other friends!" />		</>	);};
@Component({	selector: "toggle-button",	standalone: true,	template: `		<button			(click)="togglePressed()"			[style]="				pressed					? 'background-color: black; color: white;'					: 'background-color: white;color: black'			"			type="button"			[attr.aria-pressed]="pressed"		>			{{ text }}		</button>	`,})class ToggleButtonComponent {	@Input() text!: string;	pressed = false;	togglePressed() {		this.pressed = !this.pressed;	}}@Component({	selector: "toggle-button-list",	standalone: true,	imports: [ToggleButtonComponent],	template: `		<toggle-button text="Hello world!" />		<toggle-button text="Hello other friends!" />	`,})class ToggleButtonListComponent {}
<!-- ToggleButton.vue --><script setup>import { ref } from "vue";const pressed = ref(false);const props = defineProps(["text"]);function togglePressed() {	pressed.value = !pressed.value;}</script><template>	<button		@click="togglePressed()"		:style="			pressed				? 'background-color: black; color: white'				: 'background-color: white; color: black'		"		type="button"		:aria-pressed="pressed"	>		{{ props.text || "Test" }}	</button></template>
<!-- ToggleButtonList.vue --><script setup>import ToggleButton from "./ToggleButton.vue";</script><template>	<ToggleButton text="Hello world!" />	<ToggleButton text="Hello other friends!" /></template>

Here, we're passing text as a string property to assign text. But oh no! What if we wanted to add a span inside of the button to add bolded text? After all, if you pass Hello, <span>world</span>!, it wouldn't render the span, but instead render the <span> as text.

A button with the inner text of "Hello, span, world, end span,!"

Instead, let's allow the parent of our ToggleButton to pass in a template that's then rendered into the component.

In React, JSX that's passed as a child to a component can be accessed through a special children component property name:

// "children" is a preserved property name by React. It reflects passed child nodesconst ToggleButton = ({ children }) => {	const [pressed, setPressed] = useState(false);	return (		<button			onClick={() => setPressed(!pressed)}			style={{				backgroundColor: pressed ? "black" : "white",				color: pressed ? "white" : "black",			}}			type="button"			aria-pressed={pressed}		>			{/* We then utilize this special property name as any */}			{/* other JSX variable to display its contents */}			{children}		</button>	);};const ToggleButtonList = () => {	return (		<>			<ToggleButton>				Hello <span style={{ fontWeight: "bold" }}>world</span>!			</ToggleButton>			<ToggleButton>Hello other friends!</ToggleButton>		</>	);};

Angular has a special tag called ng-content that acts as a pass-through for all children's content passed to a component.

@Component({	selector: "toggle-button",	standalone: true,	template: `		<button			(click)="togglePressed()"			[style]="				pressed					? 'background-color: black; color: white;'					: 'background-color: white;color: black'			"			type="button"			[attr.aria-pressed]="pressed"		>			<ng-content />		</button>	`,})class ToggleButtonComponent {	pressed = false;	togglePressed() {		this.pressed = !this.pressed;	}}@Component({	selector: "toggle-button-list",	standalone: true,	imports: [ToggleButtonComponent],	template: `		<toggle-button>			Hello <span style="font-weight: bold;">world</span>!		</toggle-button>		<toggle-button>Hello other friends!</toggle-button>	`,})class ToggleButtonListComponent {}

Because ng-content is built into Angular's compiler, we do not need to import anything into our component to use the feature.

When in Vue-land, the slot tag is used to pass children through to a component's template.

<!-- ToggleButton.vue --><script setup>import { ref } from "vue";const pressed = ref(false);function togglePressed() {	pressed.value = !pressed.value;}</script><template>	<button		@click="togglePressed()"		:style="			pressed				? 'background-color: black; color: white'				: 'background-color: white; color: black'		"		type="button"		:aria-pressed="pressed"	>		<slot></slot>	</button></template>
<!-- ToggleButtonList.vue --><script setup>import ToggleButton from "./ToggleButton.vue";</script><template>	<ToggleButton>		Hello <span style="font-weight: bold">world</span>!	</ToggleButton>	<ToggleButton>Hello other friends!</ToggleButton></template>

Because slot is a built-in component of Vue, we do not need to import it from the vue package.

Here, we're able to pass a span and other elements directly to our ToggleButton component as children.

Using Other Framework Features with Component Children

However, because these templates have the full power of the frameworks at their disposal, these children have superpowers! Let's add a for loop into our children's template to say hello to all of our friends:

function ToggleButtonList() {	const friends = ["Kevin,", "Evelyn,", "and James"];	return (		<>			<ToggleButton>				Hello{" "}				{friends.map((friend) => (					<span>{friend} </span>				))}				!			</ToggleButton>			<ToggleButton>				Hello other friends				<RainbowExclamationMark />			</ToggleButton>		</>	);}function RainbowExclamationMark() {	const rainbowGradient = `    linear-gradient(      180deg,      #fe0000 16.66%,      #fd8c00 16.66%,      33.32%,      #ffe500 33.32%,      49.98%,      #119f0b 49.98%,      66.64%,      #0644b3 66.64%,      83.3%,      #c22edc 83.3%    )  `;	return (		<span			style={{				fontSize: "3rem",				background: rainbowGradient,				backgroundSize: "100%",				WebkitBackgroundClip: "text",				WebkitTextFillColor: "transparent",				MozBackgroundClip: "text",			}}		>			!		</span>	);}
@Component({	selector: "rainbow-exclamation-mark",	standalone: true,	template: ` <span>!</span> `,	// These styles will only apply to this component	styles: [		`			span {				font-size: 3rem;				background: linear-gradient(					180deg,					#fe0000 16.66%,					#fd8c00 16.66%,					33.32%,					#ffe500 33.32%,					49.98%,					#119f0b 49.98%,					66.64%,					#0644b3 66.64%,					83.3%,					#c22edc 83.3%				);				background-size: 100%;				-webkit-background-clip: text;				-webkit-text-fill-color: transparent;				-moz-background-clip: text;			}		`,	],})class RainbowExclamationMarkComponent {}@Component({	selector: "toggle-button-list",	standalone: true,	imports: [ToggleButtonComponent, RainbowExclamationMarkComponent],	template: `		<toggle-button>			Hello @for (friend of friends; track friend) { <span>{{ friend }} </span> }!		</toggle-button>		<toggle-button>			Hello other friends<rainbow-exclamation-mark />		</toggle-button>	`,})class ToggleButtonListComponent {	friends = ["Kevin,", "Evelyn,", "and James"];}
<!-- ToggleButtonList.vue --><script setup>import ToggleButton from "./ToggleButton.vue";import RainbowExclamationMark from "./RainbowExclamationMark.vue";const friends = ["Kevin,", "Evelyn,", "and James"];</script><template>	<ToggleButton		>Hello <span v-for="friend of friends">{{ friend }} </span>!</ToggleButton	>	<ToggleButton		>Hello other friends		<RainbowExclamationMark />	</ToggleButton></template>
<!-- RainbowExclamationMark.vue --><template>	<span>!</span></template><!-- "scoped" means the styles only apply to this component --><style scoped>span {	font-size: 3rem;	background: linear-gradient(		180deg,		#fe0000 16.66%,		#fd8c00 16.66%,		33.32%,		#ffe500 33.32%,		49.98%,		#119f0b 49.98%,		66.64%,		#0644b3 66.64%,		83.3%,		#c22edc 83.3%	);	background-size: 100%;	-webkit-background-clip: text;	-webkit-text-fill-color: transparent;	-moz-background-clip: text;}</style>

As you can see, we can use any features inside our children - even other components!

Thanks to Sarah Fossheim for the guide on how to add clipped background text like our exclamation mark!

Named Children

While passing one set of elements is useful in its own right, many components require there to be more than one "slot" of data you can pass.

For example, take this dropdown component:

Let's build this dropdown component
These tend to be useful for FAQ pages, hidden content, and more!

This dropdown component has two potential places where passing elements would be beneficial:

<Dropdown>	<DropdownHeader>Let's build this dropdown component</DropdownHeader>	<DropdownBody>		These tend to be useful for FAQ pages, hidden contents, and more!	</DropdownBody></Dropdown>

Let's build this component with an API similar to the above using "named children."

Something worth reminding is that JSX constructs a value, just like a number or string, that you can then store to a variable.

const table = <p>Test</p>;

This can be passed to a function, like console.log, or anything any other JavaScript value can do.

console.log(<p>Test</p>); // ReactElement

Because of this behavior, to pass more than one JSX value to a component, we can use function parameters and pass them that way.

const Dropdown = ({ children, header, expanded, toggle }) => {	return (		<>			<button				onClick={toggle}				aria-expanded={expanded}				aria-controls="dropdown-contents"			>				{expanded ? "V" : ">"} {header}			</button>			<div id="dropdown-contents" role="region" hidden={!expanded}>				{children}			</div>		</>	);};function App() {	const [expanded, setExpanded] = useState(false);	return (		<Dropdown			expanded={expanded}			toggle={() => setExpanded(!expanded)}			header={<>Let's build this dropdown component</>}		>			These tend to be useful for FAQ pages, hidden contents, and more!		</Dropdown>	);}

ng-content allows you to pass a select property to have specific children projected in dedicated locations. This select property takes CSS selector query values. Knowing this, we can pass the attribute query for header by wrapping the attribute name in square brackets like so:

@Component({	selector: "dropdown-comp",	standalone: true,	template: `		<button			(click)="toggle.emit()"			:aria-expanded="expanded"			aria-controls="dropdown-contents"		>			{{ expanded ? "V" : ">" }} <ng-content select="[header]" />		</button>		<div id="dropdown-contents" role="region" [hidden]="!expanded">			<ng-content />		</div>	`,})class DropdownComponent {	@Input() expanded!: boolean;	@Output() toggle = new EventEmitter();}@Component({	selector: "app-root",	standalone: true,	imports: [DropdownComponent],	template: `		<dropdown-comp [expanded]="expanded" (toggle)="expanded = !expanded">			<ng-container header>Let's build this dropdown component</ng-container>			These tend to be useful for FAQ pages, hidden contents, and more!		</dropdown-comp>	`,})class AppComponent {	expanded = false;}

Once ng-content finds related elements that match the select query, they will be content projected into the appropriate locations. If not matched by a ng-content[select], they will be projected to a non select enabled ng-content.

Similar to how Angular's ng-content[select] query works, Vue allows you to pass a name to the slot component to a projected and named content.

<!-- Dropdown.vue --><script setup>const props = defineProps(["expanded"]);const emit = defineEmits(["toggle"]);</script><template>	<button		@click="emit('toggle')"		:aria-expanded="expanded"		aria-controls="dropdown-contents"	>		{{ props.expanded ? "V" : ">" }} <slot name="header" />	</button>	<div id="dropdown-contents" role="region" :hidden="!props.expanded">		<slot />	</div></template>
<!-- App.vue --><script setup>import { ref } from "vue";import Dropdown from "./Dropdown.vue";const expanded = ref(false);</script><template>	<Dropdown :expanded="expanded" @toggle="expanded = !expanded">		<template v-slot:header>Let's build this dropdown component</template>		These tend to be useful for FAQ pages, hidden contents, and more!	</Dropdown></template>

Here, we can see that slot is querying for a header template slot. This query is then satisfied by App's template for the heading template element.

v-slot also has a shorthand of #, similar to how v-bind has a shorthand of :. Using this shorthand, we can modify our App component to look like:

<!-- App.vue --><script setup>import { ref } from "vue";import Dropdown from "./Dropdown.vue";const expanded = ref(false);</script><template>	<Dropdown :expanded="expanded" @toggle="expanded = !expanded">		<template #header>Let's build this dropdown component</template>		These tend to be useful for FAQ pages, hidden contents, and more!	</Dropdown></template>

A simple version of this dropdown component is actually built into the browser as <details> and <summary> HTML tags. Building our own is an experiment intended mostly for learning. For production environment, it's highly suggested to use those built-in elements instead.

Using Passed Children to Build a Table

Now that we're familiar with how to pass a child to a component, let's apply it to one of the components we've been building for our file hosting application: our files "list."

A table of files with headings for "Name", "Last modified", "Type", and "Size"

While this does constitute a list of files, there are actually two dimensions of data: Down and right. This makes this "list" really more of a "table". As such, it's actually a bit of a misconception to use the Unordered List (<ul>) and List Item (<li>) elements for this specific UI element.

Instead, let's build out an HTML table element. A normal HTML table might look something like this:

<table>	<thead>		<tr>			<th>Name</th>		</tr>	</thead>	<tbody>		<tr>			<td>Movies</td>		</tr>	</tbody></table>

Where th acts as a heading data item, and td acts as a bit of data on a given row and column.

Let's refactor our file list to use this DOM layout:

const File = ({ href, fileName, isSelected, onSelected, isFolder }) => {	const [inputDate, setInputDate] = useState(new Date());	// ...	return (		<tr			onClick={onSelected}			aria-selected={isSelected}			style={				isSelected					? { backgroundColor: "blue", color: "white" }					: { backgroundColor: "white", color: "blue" }			}		>			<td>				<a href={href} style={{ color: "inherit" }}>					{fileName}				</a>			</td>			<td>{isFolder ? "Type: Folder" : "Type: File"}</td>			<td>{!isFolder && <FileDate inputDate={inputDate} />}</td>		</tr>	);};const filesArray = [	{		fileName: "File one",		href: "/file/file_one",		isFolder: false,	},	{		fileName: "File two",		href: "/file/file_two",		isFolder: false,	},	{		fileName: "File three",		href: "/file/file_three",		isFolder: false,	},];// This was previously called "FileList"const FileTableBody = () => {	// ...	return (		<tbody>			{filesArray.map((file) => {				return (					<Fragment key={file.id}>						{!file.isFolder && (							<File								fileName={file.fileName}								href={file.href}								isSelected={false}								isFolder={file.isFolder}								onSelected={() => {}}							/>						)}					</Fragment>				);			})}		</tbody>	);};// This is a new componentconst FileTable = () => {	return (		<table style={{ borderSpacing: 0 }}>			<FileTableBody />		</table>	);};

Angular is unlike the other frameworks covered in this book. For example, let's say we port the table components from the other frameworks one-to-one into Angular.

@Component({	selector: "file-item",	standalone: true,	imports: [FileDateComponent],	template: `		<tr [attr.aria-selected]="isSelected" (click)="selected.emit()">			<!-- Removed other children for readability -->			<td>				<a [href]="href" style="color: inherit">{{ fileName }}</a>			</td>		</tr>	`,})class FileComponent {	// ...}@Component({	selector: "file-table-body",	standalone: true,	imports: [FileComponent],	template: `		<tbody>			@for (file of filesArray; let i = $index; track file.id) {				<!-- Removed props for readability -->				<file-item />			}		</tbody>	`,})class FileTableBody {	// ...}@Component({	selector: "file-table",	standalone: true,	imports: [FileTableBody],	template: `		<table style="border-spacing: 0;">			<file-table-body />		</table>	`,})class FileTableComponent {}

If we render this, we'll see the following incorrectly formatted table:

Our table data in a single line rather than in a grid

This table is not formatted correctly. Where are the rows? Why is everything horizontally lined up next to one-another?

This is because if you look at the output of the FileTableComponent being rendered, you'll find the following markup:

<file-table>	<table>		<file-table-body>			<tbody>				<file-item>					<tr>						<td><a>File one</a></td>					</tr>				</file-item>				<file-item>					<tr>						<td><a>File two</a></td>					</tr>				</file-item>				<file-item>					<tr>						<td><a>File three</a></td>					</tr>				</file-item>			</tbody>		</file-table-body>	</table></file-table>

Here, you'll notice that the <tbody> isn't under <table>; it's under a <file-table-body> in between those two elements. Similarly, <tr> isn't under <tbody>, it's under <file-item>.

These changes to the markup are not allowed or understood by the HTML specification, which is why our table is formatted so weirdly.

Instead, our markup needs to look like this:

<file-table>	<table>		<tbody>			<tr>				<td><a>File one</a></td>			</tr>			<tr>				<td><a>File two</a></td>			</tr>			<tr>				<td><a>File three</a></td>			</tr>		</tbody>	</table></file-table>

Why doesn't our markup look like that already?

The reason our markup isn't correct is that we're creating non-default host elements in our Angular components.

Angular Host Elements and Bindings

What are Angular host elements?

When you use a selector in Angular to create an element, that selector stays in the DOM:

@Component({	selector: "list-item",	standalone: true,	template: "",})class ListItemComponent {}

Rendering this component will yield the following DOM elements:

<list-item></list-item>

This list-item selector-named element is known as a component's "host element." This host element preservation is different from how React and Vue work, which don't have the concept of in-DOM host nodes.

I've written more about host elements in my article on the topic. Please read this if you're left a bit confused on how host elements work.

At first, this might seem like a roadblock to implementing our list-item component properly. However, there are two features that we can use to fix this problem:

  1. Selecting HTML elements with an attribute name
  2. Host element element/property binding

Combined, this might look something like this:

@Component({	selector: "tr[file-item]",	standalone: true,	host: {		"(click)": "selected.emit()",		"[style]": `			isSelected ?				'background-color: blue; color: white' :				'background-color: white; color: blue'		`,	},	template: ` <td>{{ fileName }}</td> `,})class FileItemComponent {	@Input() fileName: string;	@Input() isSelected: boolean;	@Output() selected = new EventEmitter();}

Where host allows us to bind our typical event handlers and attributes to the tr element selected by our new selector of tr[file-item].

The host element might seem confusing at first, but is super useful in the right contexts. See my article on the property for more information.

Building a Functional Angular Table

Knowing how host and selector can properly work together, let's finally build the Angular table that we wanted to from the start:

@Component({	selector: "tr[file-item]",	standalone: true,	imports: [FileDateComponent],	host: {		"[attr.aria-selected]": "isSelected",		"(click)": "selected.emit()",		"[style]": `			isSelected ?				'background-color: blue; color: white' :				'background-color: white; color: blue'		`,	},	template: `		<td>			<a [href]="href" style="color: inherit">{{ fileName }}</a>		</td>		@if (isFolder) {			<td>Type: Folder</td>		} else {			<td>Type: File</td>		}		<td>			@if (!isFolder) {				<file-date [inputDate]="inputDate" />			}		</td>	`,})class FileComponent implements OnInit, OnDestroy {	@Input() fileName!: string;	@Input() href!: string;	@Input() isSelected!: boolean;	@Input() isFolder!: boolean;	@Output() selected = new EventEmitter();	inputDate = new Date();	// ...}// This was previously called "FileList"@Component({	selector: "tbody[file-table-body]",	standalone: true,	imports: [FileComponent],	template: `		@for (file of filesArray; let i = $index; track file.id) {			@if (!file.isFolder) {				<tr					file-item					[isSelected]="false"					[fileName]="file.fileName"					[href]="file.href"					[isFolder]="file.isFolder"				></tr>			}		}	`,})class FileTableBodyComponent {	filesArray: File[] = [		{			fileName: "File one",			href: "/file/file_one",			isFolder: false,			id: 1,		},		{			fileName: "File two",			href: "/file/file_two",			isFolder: false,			id: 2,		},		// ...	];}// This is a new component@Component({	selector: "file-table",	standalone: true,	imports: [FileTableBody],	template: `		<table style="border-spacing: 0;">			<tbody file-table-body></tbody>		</table>	`,})class FileTableComponent {}
<!-- File.vue --><script setup>import { ref } from "vue";import FileDate from "./FileDate.vue";const props = defineProps(["fileName", "href", "isSelected", "isFolder"]);const emit = defineEmits(["selected"]);const inputDate = ref(new Date());// ...</script><template>	<tr		@click="emit('selected')"		:aria-selected="props.isSelected"		:style="			props.isSelected				? { backgroundColor: 'blue', color: 'white' }				: { backgroundColor: 'white', color: 'blue' }		"	>		<td>			<a :href="props.href" style="color: inherit">{{ props.fileName }}</a>		</td>		<td v-if="props.isFolder">Type: Folder</td>		<td v-else>Type: File</td>		<td><FileDate v-if="!props.isFolder" :inputDate="inputDate" /></td>	</tr></template>
<!-- FileTableBody --><!-- This was previously called "FileList" --><script setup>import File from "./File.vue";// ...const filesArray = [	{		fileName: "File one",		href: "/file/file_one",		isFolder: false,	},	{		fileName: "File two",		href: "/file/file_two",		isFolder: false,	},	{		fileName: "File three",		href: "/file/file_three",		isFolder: false,	},];</script><template>	<tbody>		<template v-for="file in filesArray">			<File				v-if="!file.isFolder"				:fileName="file.fileName"				:href="file.href"				:isFolder="file.isFolder"				:isSelected="false"			/>		</template>	</tbody></template>
<!-- FileTable.vue --><script setup>import FileTableBody from "./FileTableBody.vue";</script><template>	<table>		<FileTableBody />	</table></template>

Now that we have an explicit FileTable component, let's see if we're able to style it a bit more with a replacement FileTableContainer component, which uses passing children to style the underlying table element.

const FileTableContainer = ({ children }) => {	return (		<table			style={{				color: "#3366FF",				border: "2px solid #3366FF",				borderSpacing: 0,				padding: "0.5rem",			}}		>			{children}		</table>	);};const FileTable = () => {	return (		<FileTableContainer>			<FileTableBody />		</FileTableContainer>	);};
@Component({	selector: "file-table-container",	standalone: true,	template: `		<table			style="color: #3366FF; border: 2px solid #3366FF; border-spacing: 0; padding: 0.5rem"		>			<ng-content></ng-content>		</table>	`,})class FileTableContainerComponent {}@Component({	selector: "file-table",	standalone: true,	imports: [FileTableContainerComponent, FileTableBody],	template: `		<file-table-container><tbody file-table-body /></file-table-container>	`,})class FileTableComponent {}
<!-- FileTableContainer.vue --><template>	<table		style="			color: #3366ff;			border: 2px solid #3366ff;			border-spacing: 0;			padding: 0.5rem;		"	>		<slot></slot>	</table></template>
<!-- FileTable.vue --><script setup>import FileTableContainer from "./FileTableContainer.vue";import FileTableBody from "./FileTableBody.vue";</script><template>	<FileTableContainer><FileTableBody /></FileTableContainer></template>

Challenge

Let's make this chapter's challenge a continuation of the table that we just built in the last section. See, our previous table only had the files themselves, not the header. Let's change that by adding in a second set of children we can pass, like so:

<table>	<FileHeader />	<FileList /></table>
const FileTableContainer = ({ children, header }) => {	return (		<table			style={{				color: "#3366FF",				border: "2px solid #3366FF",				padding: "0.5rem",				borderSpacing: 0,			}}		>			<thead>{header}</thead>			{children}		</table>	);};const FileTable = () => {	const headerEl = (		<tr>			<th>Name</th>			<th>File Type</th>			<th>Date</th>		</tr>	);	return (		<FileTableContainer header={headerEl}>			<FileTableBody />		</FileTableContainer>	);};
@Component({	selector: "file-table-container",	standalone: true,	template: `		<table			style="color: #3366FF; border: 2px solid #3366FF; border-spacing: 0; padding: 0.5rem"		>			<thead>				<ng-content select="[header]"></ng-content>			</thead>			<ng-content></ng-content>		</table>	`,})class FileTableContainerComponent {}@Component({	selector: "file-table",	standalone: true,	imports: [FileTableContainerComponent, FileTableBodyComponent],	template: `		<file-table-container>			<tr header>				<th>Name</th>				<th>Date</th>			</tr>			<file-table-body />		</file-table-container>	`,})class FileTableComponent {}
<!-- FileTableContainer.vue --><template>	<table		style="			color: #3366ff;			border: 2px solid #3366ff;			border-spacing: 0;			padding: 0.5rem;		"	>		<thead>			<slot name="header"></slot>		</thead>		<slot></slot>	</table></template>
<!-- FileTable.vue --><script setup>import FileTableContainer from "./FileTableContainer.vue";import FileTableBody from "./FileTableBody.vue";</script><template>	<FileTableContainer>		<template #header>			<tr>				<th>Name</th>				<th>File Type</th>				<th>Date</th>			</tr>		</template>		<FileTableBody />	</FileTableContainer></template>
Previous articleTransparent Elements
Next article Element Reference

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.