Description
A design pattern I created to solve the problem of showing conditional form elements when certain options were selected based on the data populating the form.Angular
Code
<div class="mat-app-background layout"> <mat-card class="form-card"> <mat-card-header> <mat-card-title>Favorite Sci-Fi Universes</mat-card-title> </mat-card-header> <mat-card-content> <form [formGroup]="universesForm"> <mat-form-field appearance="outline"> <mat-label>Universes</mat-label> <mat-select #universes formControlName="universes" name="universes" multiple> <mat-select-trigger> {{universes.value?.[0]?.name || ''}} @if ((universes.value?.length || 0) > 1) { <span class="example-additional-selection"> ({{(universes.value?.length || 0)}} selected) </span> } </mat-select-trigger> @for (universe of universeList; track universe) { <mat-option [value]="universe">{{universe.name}}</mat-option> } </mat-select> </mat-form-field> @if (subUniverses.length > 0) { <mat-card appearance="outlined" class="conditional-form-card"> <mat-card-content formArrayName="subUniverses"> @for (subUniverse of subUniverses.controls; track subUniverse; let i = $index) { <mat-form-field appearance="outline"> <mat-label>{{ selectedSubUniverses[i]?.name }}</mat-label> <mat-select [formControlName]="i"> <mat-option [value]="{ key: selectedSubUniverses[i].key, name: selectedSubUniverses[i].name, sub: {key: 'all', name: 'All'} }">All</mat-option> @for (sub of selectedSubUniverses[i].sub; track sub; let j = $index) { <mat-option [value]="{ key: selectedSubUniverses[i].key, name: selectedSubUniverses[i].name, sub: [sub] }">{{sub.name}}</mat-option> } </mat-select> </mat-form-field> } </mat-card-content> <mat-card-footer></mat-card-footer> </mat-card> } </form> </mat-card-content> <mat-card-footer> <mat-card-actions align="end"> <button mat-flat-button (click)="openSnackBar()">Open SnackBar</button> </mat-card-actions> </mat-card-footer> </mat-card></div>
import { CommonModule } from '@angular/common';import { Component, type OnDestroy, type OnInit } from '@angular/core';import { FormArray, FormBuilder, FormControl, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';import { MatSelectModule } from '@angular/material/select';import { MatFormFieldModule } from '@angular/material/form-field';import { MatCardModule } from '@angular/material/card';import { MatButtonModule } from '@angular/material/button';import { MatSnackBar } from '@angular/material/snack-bar';import { provideAnimations } from '@angular/platform-browser/animations';import { Subject, takeUntil } from 'rxjs';
interface UniverseOption { key: string; name: string; sub?: UniverseOption[];}
@Component({ selector: 'app-conditional-form-demo', standalone: true, imports: [MatFormFieldModule, MatSelectModule, FormsModule, ReactiveFormsModule, CommonModule, MatCardModule, MatButtonModule], styleUrl: "conditional-form-demo.scss", templateUrl: "conditional-form-demo.html",})export class ConditionalFormDemo implements OnInit, OnDestroy { private destroyed$ = new Subject<void>(); static clientProviders = [provideAnimations()]; public universesForm: FormGroup = new FormGroup([]);
universeList: UniverseOption[] = [ { key: 'starTrek', name: 'Star Trek', sub: [ { key: 'tos', name: 'The Original Series' }, { key: 'ds9', name: 'Deep Space Nine' }, { key: 'tng', name: 'The Next Generation' }, { key: 'dsc', name: 'Discovery' }, { key: 'snw', name: 'Strange New Worlds' }, { key: 'ld', name: 'Lower Decks' }, { key: 'pcd', name: 'Picard' } ] }, { key: 'marvel', name: 'Marvel', sub: [ { key: 'movies', name: 'Movies' }, { key: 'comics', name: 'Comics' } ] }, { key: 'dc', name: 'DC', sub: [ { key: 'movies', name: 'Movies' }, { key: 'comics', name: 'Comics' } ] }, { key: 'battlestar-galactica', name: 'Battlestar Galactica' }, { key: 'stargate', name: 'Stargate', sub: [ { key: 'sg1', name: 'SG-1' }, { key: 'atlantis', name: 'Atlantis' }, { key: 'universe', name: 'Universe' } ] }, { key: 'dune', name: 'Dune' }, { key: 'starWars', name: 'Star Wars' } ];
selectedUniverses: UniverseOption[] = []; selectedSubUniverses: UniverseOption[] = []; showSub: boolean = false;
constructor(private formBuilder: FormBuilder, private snackBar: MatSnackBar) { this.formInit(); }
ngOnInit() { this.universesForm.get('universes')?.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((selectedUniverses: UniverseOption[]) => { this.selectedSubUniverses = selectedUniverses.filter(universe => universe.sub && universe.sub.length > 0); this.updateSubControlList(selectedUniverses); }); this.universesForm.get('subUniverses')?.valueChanges.pipe(takeUntil(this.destroyed$)).subscribe((selectedSubUniverses: UniverseOption[]) => { }) }
ngOnDestroy() { this.destroyed$.next(); this.destroyed$.complete(); }
formInit() { this.universesForm = this.formBuilder.group({ universes: this.formBuilder.control<UniverseOption[]>([]), subUniverses: this.formBuilder.array<FormControl<UniverseOption | null>>([]), }); }
get subUniverses() { return this.universesForm.controls.subUniverses as FormArray; }
updateSubControlList(selected: UniverseOption[]) { const selectedKeys = new Set(selected.map(u => u.key));
// Remove controls for deselected universes for (let i = this.subUniverses.length - 1; i >= 0; i--) { const control = this.subUniverses.at(i); const universeKey = control.value?.key;
if (universeKey && !selectedKeys.has(universeKey)) { this.subUniverses.removeAt(i); } }
// Add/Update controls for selected universes selected.forEach(universe => { if (universe.sub && universe.sub.length > 0) { let existingControl = this.subUniverses.controls.find(c => c.value?.key === universe.key);
if (!existingControl) { const defaultValue = universe; this.subUniverses.push(this.formBuilder.control<UniverseOption | null>(defaultValue)); } else if (existingControl.value && !existingControl.value?.sub) { existingControl.setValue(universe.sub[0]); } } }); }
combineUniverseSelections(universes: UniverseOption[], subUniverseControls: FormControl<UniverseOption>[]): UniverseOption[] { const combinedSelections: UniverseOption[] = [];
// Add universes without sub-universes universes.forEach((universe: UniverseOption) => { if (!universe.sub || universe.sub.length === 0) { combinedSelections.push(universe); } });
// Add universes with sub-universes (using selected sub-universe) subUniverseControls.forEach(control => { const selectedSubUniverse = control.value; if (selectedSubUniverse) { combinedSelections.push({ key: selectedSubUniverse.key, name: selectedSubUniverse.name, ...(selectedSubUniverse.sub && { sub: selectedSubUniverse.sub, }), }); } }); return combinedSelections; }
buildSnackStrings(allUniverses: UniverseOption[]): string[] { const snackStrings: string[] = [];
allUniverses.forEach((universe: UniverseOption, index: number) => { let tempString = ''; if (allUniverses.length > 1 && index === (allUniverses.length - 1)) { tempString += " and "; } tempString += universe.name.toString(); if (universe.sub && universe.sub.length > 0) { if (universe.sub[0].name === 'All' || universe.sub.length > 1) { tempString += ""; } else { tempString += ': ' + universe.sub[0].name.toString(); } } snackStrings.push(tempString); }); return snackStrings; }
openSnackBar() { const combined = this.combineUniverseSelections( this.universesForm.controls.universes.value, this.subUniverses.controls as FormControl[] ); const snackStrings = this.buildSnackStrings(combined); let snackString = ''; if (combined.length === 1) { snackString += 'Your favorite Sci-Fi Universe is ' + snackStrings.toString() + "!"; } else if (combined.length > 1) { snackString += 'Your favorite Sci-Fi Universes are ' + snackStrings.join(', ') + "!"; } else { snackString += 'Please choose your favorite Sci-Fi Universe(s)'; } this.snackBar.open(snackString, 'Dismiss'); }
}
.example-additional-selection { opacity: 0.75; font-size: 0.75em; line-height: 1;}.layout { display: grid; grid-template-columns: 1fr minmax(300px, 500px) 1fr; padding: 1rem; .form-card { grid-column: 2; margin: 1rem; padding: 0.5rem; mat-form-field { width: 100%; margin-top: 0.5rem; } .conditional-form-card { padding: 0.5rem; margin: 0.5rem; } }}
React
coming soon
VueJS
coming soon
Future Plans
In the future, I plan to add state management (which will also be brought to its own page to represent each library) and other versions of this example in various frameworks/libraries (starting with React and VueJS)