Cory Rylan

My name is , Google Developer Expert, Speaker, Software Developer. Building Design Systems and Web Components.

Follow @coryrylan
Angular

Using Web Components in Angular Forms with Element Internals

Cory Rylan

-

This article has been updated to the latest version Angular 17 and tested with Angular 16. The content is likely still applicable for all Angular 2 + versions.

Web Components provide a native component model for UI components. In addition, web Components encapsulate several stand-alone APIs such as Custom Elements, Shadow DOM for CSS encapsulation, slots for content projection, and Element Internals for accessibility and form participation. In this post, we will see how a basic text input Web Component can be used inside an Angular form using the Reactive Forms API.

By default, when creating a Web Component with Shadow DOM, the HTML, CSS, and ID attributes are all encapsulated to that element's shadow root. The shadow root ensures styles and id attributes are no longer global, making maintenance much easier but also means native inputs are not accessible to any code outside of the element.

Using the Element Internals API, we can enable our Web Components to participate or enable native form control support. This API allows us to have encapsulated styles and behaviors while enabling consumers of our components to use them just like any other form control.

Form Associated Custom Elements

Using the new Element Internals API, we can register our custom elements to a form element so our component can interact with the form providing form data and validation.

export class MyControl extends HTMLElement {
  #internals = this.attachInternals();

  static formAssociated = true;

  constructor() {
    super();
    this.attachShadow({ mode: 'open', delegatesFocus: true });
    this.shadowRoot.innerHTML = `
      <style>
        input {
          box-sizing: border-box;
          padding: 4px 12px;
          border: 2px solid #2d2d2d;
          border-radius: 24px;
          width: 100%;
        }
      </style>
      <input type="text" />`;
  }
}

customElements.define('my-control', MyControl);

Our component has two particular lines that enable form participation. First is the #internals = this.attachInternals(). The attachInternals call enables a suite of APIs for our component to access, allowing it to use features native HTML Elements use. This include behaviors for accessibility, CSS custom states and forms.

Next is the static call static formAssociated = true. This line enables a set of APIs for our component to access and use for forms. Our template encapsulates some styles and a native input. This input, however, is not accessible, but with the Element Internals, the my-control element now has the basics needed for implementing form controls, just like the native input.

To enable the basic value and validation logic, we need to add additional setters/getters to consumers of my-control can treat it just like any other input type.

export class MyControl extends HTMLElement {
  #internals = this.attachInternals() as any;
  #input: HTMLInputElement;

  static formAssociated = true;

  static observedAttributes = ['disabled', 'placeholder'];

  get form() { return this.#internals.form; }
  get name() { return this.getAttribute('name')};
  get type() { return this.localName; }
  get value() { return this.#input.value; }
  set value(v) { this.#input.value = v; }
  get validity() { return this.#internals.validity; }
  get validationMessage() { return this.#internals.validationMessage; }
  get willValidate() { return this.#internals.willValidate; }

  constructor() {
    super();
    this.attachShadow({ mode: 'open', delegatesFocus: true });
    this.shadowRoot.innerHTML = `
      <style>
        input {
          box-sizing: border-box;
          padding: 4px 12px;
          border: 2px solid #2d2d2d;
          border-radius: 24px;
          width: 100%;
        }
      </style>
      <input type="text" />`;

    this.#input = this.shadowRoot.querySelector('input');
    this.#input.addEventListener('input', () => this.#internals.setFormValue(this.value));
  }

  checkValidity() { return this.#internals.checkValidity(); }

  reportValidity() { return this.#internals.reportValidity(); }

  attributeChangedCallback(name, _oldValue, newValue) { this.#input[name] = newValue; }
}

customElements.define('my-control', MyControl);

Now, this component just scratches the surface of what is possible for a robust form-based Web Component. However, this is enough for our primary use case of integrating into Angular.

Within our Angular component, we can create a Form with reactive forms and access my-control just like any other input.

<form [formGroup]="form" (ngSubmit)="submit()">
  <label for="name">label</label>
  <input formControlName="name" id="name" />

  <label for="custom">custom</label>
  <my-control formControlName="custom" id="custom" ngDefaultControl></my-control>
  <div class="error" *ngIf="form.controls.custom.invalid && (form.controls.custom.dirty || form.controls.custom.touched)">required</div>

  <button>submit</button>
</form>
import { Component } from '@angular/core';
import { FormBuilder, FormGroup, Validators } from '@angular/forms';

@Component({
  selector: 'my-app',
  templateUrl: './app.component.html',
  styleUrls: [ './app.component.css' ]
})
export class AppComponent  {
  form: FormGroup;

  constructor(private formBuilder: FormBuilder) {
    this.form = this.formBuilder.group({
      name: ['hello'],
      custom: ['', Validators.required]
    });
  }

  submit() {
    this.form.controls.custom.markAsTouched();
    console.log(this.form.value);
  }
}
Angular Form and Web Component form control

Angular Compatibility Issues

One essential line to call out is on the my-control tag in our Angular template.

 <my-control formControlName="custom" id="custom" ngDefaultControl></my-control>

Note the ngDefaultControl directive. This directive tells the Angular form API to treat this element tag as a standard string text input. With this, the Angular forms API will fully work with your custom element. You can attach this directive automatically by creating a global directive that uses the my-control tag name.

Also, note the ngDefaultControl only works with string value types. Form inputs, including Web Components, can have other types such as number, boolean and date. If you are interested in Angular adding support for this, I recommend opening an issue or commenting/upvoting this PR Add explicit selectors for all built-in ControlValueAccessors.

Another possible workaround is providing directives that extend the internal Control Value Accessors to enable Angular Forms to be compatible with other control types.

import { CUSTOM_ELEMENTS_SCHEMA, Directive, forwardRef, NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser';
import { CheckboxControlValueAccessor, DefaultValueAccessor, NG_VALUE_ACCESSOR, RadioControlValueAccessor, ReactiveFormsModule } from '@angular/forms';

@Directive({ selector: 'ui-radio', providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => UIRadio), multi: true }] })
export class UIRadio extends RadioControlValueAccessor { }

@Directive({ selector: 'ui-checkbox', providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => UICheckbox), multi: true }] })
export class UICheckbox extends CheckboxControlValueAccessor { }

@Directive({ selector: 'ui-input', providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => UIInput), multi: true }] })
export class UIInput extends DefaultValueAccessor { }

@NgModule({
  imports: [BrowserModule, ReactiveFormsModule],
  declarations: [AppComponent, UIRadio, UICheckbox, UIInput],
  bootstrap: [AppComponent],
  schemas: [CUSTOM_ELEMENTS_SCHEMA]
})
export class AppModule { }

Each Web Component ui-checkbox, ui-input, and ui-radio implement their respective form control input types. The Value Accessor directives enable Angular Forms to treat them as the native checkbox, input, and radio inputs. You can find how this works internally in Angular forms here.

I highly recommend checking out "More capable form controls" on web.dev for a deep dive into how to build your own Web Component form inputs. You can check out the working demo in the link below!

View Demo Code   
Twitter Facebook LinkedIn Email
 

No spam. Short occasional updates on Web Development articles, videos, and new courses in your inbox.

Related Posts

Angular

Creating Dynamic Tables in Angular

Learn how to easily create HTML tables in Angular from dynamic data sources.

Read Article
Web Components

Reusable Component Patterns - Default Slots

Learn about how to use default slots in Web Components for a more flexible API design.

Read Article
Web Components

Reusable Component Anti-Patterns - Semantic Obfuscation

Learn about UI Component API design and one of the common anti-patterns, Semantic Obfuscation.

Read Article