Tip: This guide assumes you've already read the Essentials Guide. Read that first if you're new to Angular.
Tip: If you're familiar with other web frameworks, input properties are similar to props.
When creating a component, you can mark specific class properties as bindable by adding the @Input
decorator on the property:
@Component({...})export class CustomSlider { @Input() value = 0;}
This lets you bind to the property in a template:
<custom-slider [value]="50" />
Angular refers to properties marked with the @Input
decorator as inputs. When using a component, you pass data to it by setting its inputs.
Angular records inputs statically at compile-time. Inputs cannot be added or removed at run-time.
When extending a component class, inputs are inherited by the child class.
Input names are case-sensitive.
Customizing inputs
The @Input
decorator accepts a config object that lets you change the way that input works.
Required inputs
You can specify the required
option to enforce that a given input must always have a value.
@Component({...})export class CustomSlider { @Input({required: true}) value = 0;}
If you try to use a component without specifying all of its required inputs, Angular reports an error at build-time.
Input transforms
You can specify a transform
function to change the value of an input when it's set by Angular.
@Component({ selector: 'custom-slider', ...})export class CustomSlider { @Input({transform: trimString}) label = '';}function trimString(value: string | undefined) { return value?.trim() ?? '';}
<custom-slider [label]="systemVolume" />
In the example above, whenever the value of systemVolume
changes, Angular runs trimString
and sets label
to the result.
The most common use-case for input transforms is to accept a wider range of value types in templates, often including null
and undefined
.
Input transform function must be statically analyzable at build-time. You cannot set transform functions conditionally or as the result of an expression evaluation.
Input transform functions should always be pure functions. Relying on state outside of the transform function can lead to unpredictable behavior.
Type checking
When you specify an input transform, the type of the transform function's parameter determines the types of values that can be set to the input in a template.
@Component({...})export class CustomSlider { @Input({transform: appendPx}) widthPx: string = '';}function appendPx(value: number) { return `${value}px`;}
In the example above, the widthPx
input accepts a number
while the property on the class is a string
.
Built-in transformations
Angular includes two built-in transform functions for the two most common scenarios: coercing values to boolean and numbers.
import {Component, Input, booleanAttribute, numberAttribute} from '@angular/core';@Component({...})export class CustomSlider { @Input({transform: booleanAttribute}) disabled = false; @Input({transform: numberAttribute}) number = 0;}
booleanAttribute
imitates the behavior of standard
HTML boolean attributes, where the presence of the attribute indicates a "true" value. However, Angular's booleanAttribute
treats the literal string "false"
as the boolean false
.
numberAttribute
attempts to parse the given value to a number, producing NaN
if parsing fails.
Input aliases
You can specify the alias
option to change the name of an input in templates.
@Component({...})export class CustomSlider { @Input({alias: 'sliderValue'}) value = 0;}
<custom-slider [sliderValue]="50" />
This alias does not affect usage of the property in TypeScript code.
While you should generally avoid aliasing inputs for components, this feature can be useful for renaming properties while preserving an alias for the original name or for avoiding collisions with the name of native DOM element properties.
The @Input
decorator also accepts the alias as its first parameter in place of the config object.
Inputs with getters and setters
A property implemented with a getter and setter can be an input:
export class CustomSlider { @Input() get value(): number { return this.internalValue; } set value(newValue: number) { this.internalValue = newValue; } private internalValue = 0;}
You can even create a write-only input by only defining a public setter:
export class CustomSlider { @Input() set value(newValue: number) { this.internalValue = newValue; } private internalValue = 0;}
Prefer using input transforms instead of getters and setters if possible.
Avoid complex or costly getters and setters. Angular may invoke an input's setter multiple times, which may negatively impact application performance if the setter performs any costly behaviors, such as DOM manipulation.
Specify inputs in the @Component
decorator
In addition to the @Input
decorator, you can also specify a component's inputs with the inputs
property in the @Component
decorator. This can be useful when a component inherits a property from a base class:
// `CustomSlider` inherits the `disabled` property from `BaseSlider`.@Component({ ..., inputs: ['disabled'],})export class CustomSlider extends BaseSlider { }
You can additionally specify an input alias in the inputs
list by putting the alias after a colon in the string:
// `CustomSlider` inherits the `disabled` property from `BaseSlider`.@Component({ ..., inputs: ['disabled: sliderDisabled'],})export class CustomSlider extends BaseSlider { }
Choosing input names
Avoid choosing input names that collide with properties on DOM elements like HTMLElement. Name collisions introduce confusion about whether the bound property belongs to the component or the DOM element.
Avoid adding prefixes for component inputs like you would with component selectors. Since a given element can only host one component, any custom properties can be assumed to belong to the component.