angular/aio/content/guide/template-reference-variables.md
2022-04-12 23:55:32 +00:00

213 lines
9.2 KiB
Markdown

# Template variables
Template variables help you use data from one part of a template in another part of the template.
Use template variables to perform tasks such as respond to user input or finely tune your application's forms.
A template variable can refer to the following:
* A DOM element within a template
* A directive
* An element
* [TemplateRef](api/core/TemplateRef)
* A [web component](https://developer.mozilla.org/docs/Web/Web_Components "MDN: Web Components")
<div class="alert is-helpful">
See the <live-example></live-example> for a working example containing the code snippets in this guide.
</div>
## Syntax
In the template, you use the hash \(`#`\) character to declare a template variable.
The following template variable, `#phone`, declares a `phone` variable on an `<input>` element.
<code-example header="src/app/app.component.html" path="template-reference-variables/src/app/app.component.html" region="ref-var"></code-example>
Refer to a template variable anywhere in the component's template.
Here, a `<button>` further down the template refers to the `phone` variable.
<code-example header="src/app/app.component.html" path="template-reference-variables/src/app/app.component.html" region="ref-phone"></code-example>
## How Angular assigns values to template variables
Angular assigns a template variable a value based on where you declare the variable:
* If you declare the variable on a component, the variable refers to the component instance
* If you declare the variable on a standard HTML tag, the variable refers to the element
* If you declare the variable on an `<ng-template>` element, the variable refers to a `TemplateRef` instance, which represents the template.
For more information on `<ng-template>`, see [How Angular uses the asterisk, `*`, syntax](guide/structural-directives#asterisk) in [Structural directives](guide/structural-directives).
* If the variable specifies a name on the right-hand side, such as `#var="ngModel"`, the variable refers to the directive or component on the element with a matching `exportAs` name.
<!--todo: What does the second half of this mean?^^ Can we explain this more fully? Could I see a working example? -kw -->
### Using `NgForm` with template variables
In most cases, Angular sets the template variable's value to the element on which it occurs.
In the previous example, `phone` refers to the phone number `<input>`.
The button's click handler passes the `<input>` value to the component's `callPhone()` method.
The `NgForm` directive is applied by Angular on `<form>` elements. This example demonstrates getting a reference to a different value by referencing a directive's `exportAs` name.
<code-example header="src/app/hero-form.component.html" path="template-reference-variables/src/app/app.component.html" region="ngForm"></code-example>
Without the `ngForm` attribute value, the reference value of `itemForm` would be
the [HTMLFormElement](https://developer.mozilla.org/en-US/docs/Web/API/HTMLFormElement), `<form>`.
With `NgForm`, `itemForm` is a reference to the [NgForm](api/forms/NgForm "API: NgForm") directive with the ability to track the value and validity of every control in the form.
Unlike the native `<form>` element, the `NgForm` directive has a `form` property.
The `NgForm` `form` property lets you disable the submit button if the `itemForm.form.valid` is invalid.
## Default reference type without assigned value
When declaring a template reference variable on an element without defining a value for it, its returned type will reflect the type of element it's applied to:
- **Native element**: specific subclass of [HTMLElement](https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement)
- **Component**: instance of the specific Component class
- **NgTemplate**: TemplateRef
Referencing an element by its directive needs the directive `exportAs` property set as reference value.
In case of an unspecified variable value, the reference will return an `HTMLElement`, even if the element has one or more directive applied to itself.
## Template variable scope
Refer to a template variable anywhere within its surrounding template.
[Structural directives](guide/built-in-directives), such as `*ngIf` and `*ngFor`, or `<ng-template>` act as a template boundary.
You cannot access template variables outside of these boundaries.
<div class="alert is-helpful">
Define a variable only once in the template so the runtime value remains predictable.
</div>
### Accessing in a nested template
An inner template can access template variables that the outer template defines.
In the following example, changing the text in the `<input>` changes the value in the `<span>` because Angular immediately updates changes through the template variable, `ref1`.
<code-example header="src/app/app.component.html" path="template-reference-variables/src/app/app.component.html" region="template-ref-vars-scope1"></code-example>
In this case, there is an implied `<ng-template>` around the `<span>` and the definition of the variable is outside of it.
Accessing a template variable from the parent template works because the child template inherits the context from the parent template.
Rewriting the preceding code in a more verbose form explicitly shows the `<ng-template>`.
<code-example format="html" language="html">
&lt;input #ref1 type="text" [(ngModel)]="firstExample" /&gt;
&lt;!-- New template --&gt;
&lt;ng-template [ngIf]="true"&gt;
&lt;!-- Because the context is inherited, the value is available to the new template --&gt;
&lt;span&gt;Value: {{ ref1.value }}&lt;/span&gt;
&lt;/ng-template&gt;
</code-example>
However, accessing a template variable from outside the parent template doesn't work.
<code-example format="html" language="html">
&lt;input *ngIf="true" #ref2 type="text" [(ngModel)]="secondExample" /&gt;
&lt;span&gt;Value: {{ ref2?.value }}&lt;/span&gt; &lt;!-- doesn't work --&gt;
</code-example>
The verbose form shows that `ref2` is outside the parent template.
<code-example format="html" language="html">
&lt;ng-template [ngIf]="true"&gt;
&lt;!-- The reference is defined within a template --&gt;
&lt;input #ref2 type="text" [(ngModel)]="secondExample" /&gt;
&lt;/ng-template&gt;
&lt;!-- ref2 accessed from outside that template doesn't work --&gt;
&lt;span&gt;Value: {{ ref2?.value }}&lt;/span&gt;
</code-example>
Consider the following example that uses `*ngFor`.
<code-example format="html" language="html">
&lt;ng-container *ngFor="let i of [1,2]"&gt;
&lt;input #ref type="text" [value]="i" /&gt;
&lt;/ng-container&gt;
{{ ref.value }}
</code-example>
Here, `ref.value` doesn't work.
Verbose syntax of the same loop shows why:
<code-example format="html" language="html">
&lt;ng-template ngFor let-i [ngForOf]="[1,2]"&gt;
&lt;input #ref type="text" [value]="i" /&gt;
&lt;/ng-template&gt;
{{ ref.value }}
</code-example>
The interpolation trying to access property `ref.value` occurs outside of the referenced element's parent template, making it unreachable.
Moving the interpolation inside the template makes the variable available. Now it references the correct distinct value for each element the loop renders.
<code-example format="html" language="html">
&lt;ng-template ngFor let-i [ngForOf]="[1,2]"&gt;
&lt;input #ref type="text" [value]="i" /&gt;
{{ ref.value }}
&lt;/ng-template&gt;
</code-example>
This snippet shows 2 `<input>` elements, with their respective value printed.
### Accessing a template variable within `<ng-template>`
When you declare the variable on an `<ng-template>`, the variable refers to a `TemplateRef` instance, which represents the template.
<code-example header="src/app/app.component.html" path="template-reference-variables/src/app/app.component.html" region="template-ref"></code-example>
In this example, clicking the button calls the `log()` function, which outputs the value of `#ref3` to the console.
Because the `#ref` variable is on an `<ng-template>`, the value is `TemplateRef`.
The following is the expanded browser console output of the `TemplateRef()` function with the name of `TemplateRef`.
<code-example format="shell" language="shell">
&blacktriangledown; ƒ TemplateRef()
name: "TemplateRef"
&lowbar;&lowbar;proto&lowbar;&lowbar;: Function
</code-example>
<a id="template-input-variable"></a>
<a id="template-input-variables"></a>
## Template input variable
A *template input variable* is a variable to reference within a single instance of the template.
You declare a template input variable using the `let` keyword as in `let hero`.
If its value is omitted, it gets the `$implicit` template context's property value.
There are several such variables in this example: `hero`, `i`, and `odd`.
The first one takes the value of the iterated item, because `NgForOf` assigns that to `$implicit`
<code-example format="html" language="html">
&lt;ng-template ngFor #hero let-hero [ngForOf]="heroes" let-i="index" let-odd="odd"&gt;
&lt;div [class]="{'odd-row': odd}"&gt;{{i}}:{{hero.name}}&lt;/div&gt;
&lt;/ng-template&gt;
</code-example>
The variable's scope is limited to a single instance of the repeated template.
Use the same variable name again in the definition of other structural directives.
When in the same template a _template reference variable_ and a _template input variable_ with the same name get declared, the latter takes precedence.