include ../_util-fns :marked We’ve all used a form to log in, submit a help request, place an order, book a flight, schedule a meeting and perform countless other data entry tasks. Forms are the mainstay of business applications. Any seasoned web developer can slap together an HTML form with all the right tags. It's more challenging to create a cohesive data entry experience that guides the user efficiently and effectively through the workflow behind the form. *That* takes design skills that are, to be frank, well out of scope for this guide. It also takes framework support for **two-way data binding, change tracking, validation, and error handling** ... which we shall cover in this guide on Angular forms. We will build a simple form from scratch, one step at a time. Along the way we'll learn how to - build an Angular form with a component and template - two-way data bind with `[(ngModel)]` syntax for reading and writing values to input controls - track the change state and validity of form controls using `ngModel` in combination with a form - provide strong visual feedback using special CSS classes that track the state of the controls - display validation errors to users and enable/disable form controls - use [template reference variables](./template-syntax.html#ref-vars) for sharing information among HTML elements Run the . .l-main-section :marked ## Template-Driven Forms Many of us will build forms by writing templates in the Angular [template syntax](./template-syntax.html) with the form-specific directives and techniques described in this guide. .l-sub-section :marked That's not the only way to create a form but it's the way we'll cover in this guide. :marked We can build almost any form we need with an Angular template — login forms, contact forms ... pretty much any business forms. We can lay out the controls creatively, bind them to data, specify validation rules and display validation errors, conditionally enable or disable specific controls, trigger built-in visual feedback, and much more. It will be pretty easy because Angular handles many of the repetitive, boiler plate tasks we'd otherwise wrestle with ourselves. We'll discuss and learn to build the following template-driven form: figure.image-display img(src="/resources/images/devguide/forms/hero-form-1.png" width="400px" alt="Clean Form") :marked Here at the *Hero Employment Agency* we use this form to maintain personal information about the heroes in our stable. Every hero needs a job. It's our company mission to match the right hero with the right crisis! Two of the three fields on this form are required. Required fields have a green bar on the left to make them easy to spot. If we delete the hero name, the form displays a validation error in an attention grabbing style: figure.image-display img(src="/resources/images/devguide/forms/hero-form-2.png" width="400px" alt="Invalid, Name Required") :marked Note that the submit button is disabled and the "required" bar to the left of the input control changed from green to red. .l-sub-section p We'll customize the colors and location of the "required" bar with standard CSS. :marked We will build this form in the following sequence of small steps 1. Create the `Hero` model class 1. Create the component that controls the form 1. Create a template with the initial form layout 1. Bind data properties to each form input control with the `ngModel` two-way data binding syntax 1. Add the `name` attribute to each form input control 1. Add custom CSS to provide visual feedback 1. Show and hide validation error messages 1. Handle form submission with **ngSubmit** 1. Disable the form’s submit button until the form is valid :marked ## Setup Follow the [setup](setup.html) instructions for creating a new project named angular-forms. ## Create the Hero Model Class As users enter form data, we capture their changes and update an instance of a model. We can't layout the form until we know what the model looks like. A model can be as simple as a "property bag" that holds facts about a thing of application importance. That describes well our `Hero` class with its three required fields (`id`, `name`, `power`) and one optional field (`alterEgo`). Create a new file in the app folder called `hero.ts` and give it the following class definition: +makeExample('forms/ts/app/hero.ts', null, 'app/hero.ts') :marked It's an anemic model with few requirements and no behavior. Perfect for our demo. The TypeScript compiler generates a public field for each `public` constructor parameter and assigns the parameter’s value to that field automatically when we create new heroes. The `alterEgo` is optional and the constructor lets us omit it; note the (?) in `alterEgo?`. We can create a new hero like this: code-example(format=""). let myHero = new Hero(42, 'SkyDog', 'Fetch any object at any distance', 'Leslie Rollover'); console.log('My hero is called ' + myHero.name); // "My hero is called SkyDog" :marked .l-main-section :marked ## Create a Form component An Angular form has two parts: an HTML-based _template_ and a component _class_ to handle data and user interactions programmatically. We begin with the class because it states, in brief, what the hero editor can do. Create a new file called `hero-form.component.ts` and give it the following definition: +makeExample('forms/ts/app/hero-form.component.ts', 'first', 'app/hero-form.component.ts') :marked There’s nothing special about this component, nothing form-specific, nothing to distinguish it from any component we've written before. Understanding this component requires only the Angular concepts we’ve learned in previous guides 1. We import the `Component` decorator from the Angular library as we usually do. 1. We import the `Hero` model we just created. 1. The `@Component` selector value of "hero-form" means we can drop this form in a parent template with a `` tag. 1. The `moduleId: module.id` property sets the base for module-relative loading of the `templateUrl`. 1. The `templateUrl` property points to a separate file for the template HTML called `hero-form.component.html`. 1. We defined dummy data for `model` and `powers` as befits a demo. Down the road, we can inject a data service to get and save real data or perhaps expose these properties as [inputs and outputs](./template-syntax.html#inputs-outputs) for binding to a parent component. None of this concerns us now and these future changes won't affect our form. 1. We threw in a `diagnostic` property at the end to return a JSON representation of our model. It'll help us see what we're doing during our development; we've left ourselves a cleanup note to discard it later. ### Why the separate template file? Why don't we write the template inline in the component file as we often do elsewhere? There is no “right” answer for all occasions. We like inline templates when they are short. Most form templates won't be short. TypeScript and JavaScript files generally aren't the best place to write (or read) large stretches of HTML and few editors are much help with files that have a mix of HTML and code. We also like short files with a clear and obvious purpose like this one. Form templates tend to be quite large even when displaying a small number of fields so it's usually best to put the HTML template in a separate file. We'll write that template file in a moment. Before we do, we'll take a step back and revise the `app.module.ts` and `app.component.ts` to make use of the new `HeroFormComponent`. .l-main-section :marked ## Revise the *app.module.ts* `app.module.ts` defines the application's root module. In it we identify the external modules we'll use in our application and declare the components that belong to this module, such as our `HeroFormComponent`. Because template-driven forms are in their own module, we need to add the `FormsModule` to the array of `imports` for our application module before we can use forms. Replace the contents of the "QuickStart" version with the following: +makeExample('forms/ts/app/app.module.ts', null, 'app/app.module.ts') :marked .l-sub-section :marked There are three changes: 1. We import `FormsModule` and our new `HeroFormComponent`. 1. We add the `FormsModule` to the list of `imports` defined in the `ngModule` decorator. This gives our application access to all of the template-driven forms features, including `ngModel`. 1. We add the `HeroFormComponent` to the list of `declarations` defined in the `ngModule` decorator. This makes the `HeroFormComponent` component visible throughout this module. .alert.is-important :marked If a component, directive, or pipe belongs to a module in the `imports` array, ​_DON'T_​ re-declare it in the `declarations` array. If you wrote it and it should belong to this module, ​_DO_​ declare it in the `declarations` array. .l-main-section :marked ## Revise the *app.component.ts* `app.component.ts` is the application's root component. It will host our new `HeroFormComponent`. Replace the contents of the "QuickStart" version with the following: +makeExample('forms/ts/app/app.component.ts', null, 'app/app.component.ts') :marked .l-sub-section :marked There is only one change. The `template` is simply the new element tag identified by the component's `selector` property. This will display the hero form when the application component is loaded. .l-main-section :marked ## Create an initial HTML Form Template Create a new template file called `hero-form.component.html` and give it the following definition: +makeExample('forms/ts/app/hero-form.component.html', 'start', 'app/hero-form.component.html') :marked That is plain old HTML 5. We're presenting two of the `Hero` fields, `name` and `alterEgo`, and opening them up for user input in input boxes. The *Name* `` control has the HTML5 `required` attribute; the *Alter Ego* `` control does not because `alterEgo` is optional. We've got a *Submit* button at the bottom with some classes on it for styling. **We are not using Angular yet**. There are no bindings. No extra directives. Just layout. The `container`, `form-group`, `form-control`, and `btn` classes come from [Twitter Bootstrap](http://getbootstrap.com/css/). Purely cosmetic. We're using Bootstrap to give the form a little style! .callout.is-important header Angular Forms Do Not Require A Style Library :marked Angular makes no use of the `container`, `form-group`, `form-control`, and `btn` classes or the styles of any external library. Angular apps can use any CSS library ... or none at all. :marked Let's add the stylesheet. ol li Open a terminal window in the application root folder and enter the command: code-example(language="html" escape="html"). npm install bootstrap --save li Open index.html and add the following link to the <head>. +makeExample('forms/ts/index.html', 'bootstrap')(format=".") :marked .l-main-section :marked ## Add Powers with ***ngFor** Our hero may choose one super power from a fixed list of Agency-approved powers. We maintain that list internally (in `HeroFormComponent`). We'll add a `select` to our form and bind the options to the `powers` list using `ngFor`, a technique seen previously in the [Displaying Data](./displaying-data.html) guide. Add the following HTML *immediately below* the *Alter Ego* group. +makeExample('forms/ts/app/hero-form.component.html', 'powers', 'app/hero-form.component.html (excerpt)')(format=".") :marked We are repeating the `` tag for each power in the list of Powers. The `pow` template input variable is a different power in each iteration; we display its name using the interpolation syntax with the double-curly-braces. .l-main-section :marked ## Two-way data binding with **_ngModel_** Running the app right now would be disappointing. figure.image-display img(src="/resources/images/devguide/forms/hero-form-3.png" width="400px" alt="Early form with no binding") :marked We don't see hero data because we are not binding to the `Hero` yet. We know how to do that from earlier guides. [Displaying Data](./displaying-data.html) taught us Property Binding. [User Input](./user-input.html) showed us how to listen for DOM events with an Event Binding and how to update a component property with the displayed value. Now we need to display, listen, and extract at the same time. We could use those techniques again in our form. Instead we'll introduce something new, the `[(ngModel)]` syntax, that makes binding our form to the model super-easy. Find the `` tag for the "Name" and update it like this +makeExample('forms/ts/app/hero-form.component.html', 'ngModelName-1','app/hero-form.component.html (excerpt)')(format=".") .l-sub-section :marked We appended a diagnostic interpolation after the input tag so we can see what we're doing. We left ourselves a note to throw it away when we're done. :marked Focus on the binding syntax: `[(ngModel)]="..."`. If we ran the app right now and started typing in the *Name* input box, adding and deleting characters, we'd see them appearing and disappearing from the interpolated text. At some point it might look like this. figure.image-display img(src="/resources/images/devguide/forms/ng-model-in-action.png" width="400px" alt="ngModel in action") :marked The diagnostic is evidence that we really are flowing values from the input box to the model and back again. **That's two-way data binding!** Notice that we also added a `name` attribute to our `` tag and set it to "name" which makes sense for the hero's name. Any unique value will do, but using a descriptive name is helpful. Defining a `name` attribute is a requirement when using `[(ngModel)]` in combination with a form. .l-sub-section :marked Internally Angular creates `FormControls` and registers them with an `NgForm` directive that Angular attached to the `
` tag. Each `FormControl` is registered under the name we assigned to the `name` attribute. We'll talk about `NgForm` [later in this guide](#ngForm). :marked Let's add similar `[(ngModel)]` bindings and `name` attributes to *Alter Ego* and *Hero Power*. We'll ditch the input box binding message and add a new binding at the top to the component's `diagnostic` property. Then we can confirm that two-way data binding works *for the entire hero model*. After revision the core of our form should have three `[(ngModel)]` bindings and `name` attributes that look much like this: +makeExample('forms/ts/app/hero-form.component.html', 'ngModel-2', 'app/hero-form.component.html (excerpt)') .l-sub-section :marked - Each input element has an `id` property that is used by the `label` element's `for` attribute to match the label to its input control. - Each input element has a `name` property that is required by Angular Forms to register the control with the form. :marked If we ran the app right now and changed every hero model property, the form might display like this: figure.image-display img(src="/resources/images/devguide/forms/ng-model-in-action-2.png" width="400px" alt="ngModel in super action") :marked The diagnostic near the top of the form confirms that all of our changes are reflected in the model. **Delete** the `{{diagnostic}}` binding at the top as it has served its purpose. .l-sub-section :marked ### Inside _[(ngModel)]_ *This section is an optional deep dive into [(ngModel)]. Not interested? Skip ahead!* The punctuation in the binding syntax, [()], is a good clue to what's going on. In a Property Binding, a value flows from the model to a target property on screen. We identify that target property by surrounding its name in brackets, []. This is a one-way data binding **from the model to the view**. In an Event Binding, we flow the value from the target property on screen to the model. We identify that target property by surrounding its name in parentheses, (). This is a one-way data binding in the opposite direction **from the view to the model**. No wonder Angular chose to combine the punctuation as [()] to signify a two-way data binding and a **flow of data in both directions**. In fact, we can break the `NgModel` binding into its two separate modes as we do in this re-write of the "Name" `` binding: +makeExample('forms/ts/app/hero-form.component.html', 'ngModel-3','app/hero-form.component.html (excerpt)')(format=".") :marked
The Property Binding should feel familiar. The Event Binding might seem strange. The `ngModelChange` is not an `` element event. It is actually an event property of the `NgModel` directive. When Angular sees a binding target in the form [(x)], it expects the `x` directive to have an `x` input property and an `xChange` output property. The other oddity is the template expression, `model.name = $event`. We're used to seeing an `$event` object coming from a DOM event. The `ngModelChange` property doesn't produce a DOM event; it's an Angular `EventEmitter` property that returns the input box value when it fires — which is precisely what we should assign to the model's `name` property. Nice to know but is it practical? We almost always prefer `[(ngModel)]`. We might split the binding if we had to do something special in the event handling such as debounce or throttle the key strokes. Learn more about `NgModel` and other template syntax in the [Template Syntax](./template-syntax.html) guide. .l-main-section :marked ## Track change-state and validity with **_ngModel_** A form isn't just about data binding. We'd also like to know the state of the controls on our form. Using `ngModel` in a form gives us more than just two way data binding. It also tells us if the user touched the control, if the value changed, or if the value became invalid. The *NgModel* directive doesn't just track state; it updates the control with special Angular CSS classes that reflect the state. We can leverage those class names to change the appearance of the control and make messages appear or disappear. table tr th State th Class if true th Class if false tr td Control has been visited td ng-touched td ng-untouched tr td Control's value has changed td ng-dirty td ng-pristine tr td Control's value is valid td ng-valid td ng-invalid :marked Let's add a temporary [template reference variable](./template-syntax.html#ref-vars) named **spy** to the "Name" `` tag and use the spy to display those classes. +makeExample('forms/ts/app/hero-form.component.html', 'ngModelName-2','app/hero-form.component.html (excerpt)')(format=".") :marked Now run the app and focus on the *Name* input box. Follow the next four steps *precisely* 1. Look but don't touch 1. Click in the input box, then click outside the text input box 1. Add slashes to the end of the name 1. Erase the name The actions and effects are as follows: figure.image-display img(src="/resources/images/devguide/forms/control-state-transitions-anim.gif" alt="Control State Transition") :marked We should be able to see the following four sets of class names and their transitions: figure.image-display img(src="/resources/images/devguide/forms/ng-control-class-changes.png" width="400px" alt="Control State Transitions") :marked The (`ng-valid` | `ng-invalid`) pair are most interesting to us. We want to send a strong visual signal when the data are invalid and we want to mark required fields. So we add custom CSS for visual feedback. **Delete** the `#spy` template reference variable and `TODO` as they have served their purpose. .l-main-section :marked ## Add Custom CSS for Visual Feedback We realize we can mark required fields and invalid data at the same time with a colored bar on the left of the input box: figure.image-display img(src="/resources/images/devguide/forms/validity-required-indicator.png" width="400px" alt="Invalid Form") :marked We achieve this effect by adding two styles to a new `forms.css` file that we add to our project as a sibling to `index.html`. +makeExample('forms/ts/forms.css',null,'forms.css')(format=".") :marked These styles select for the two Angular validity classes and the HTML 5 "required" attribute. We update the `` of the `index.html` to include this style sheet. +makeExample('forms/ts/index.html', 'styles', 'index.html (excerpt)')(format=".") :marked ## Show and Hide Validation Error messages We can do better. The "Name" input box is required. Clearing it turns the bar red. That says *something* is wrong but we don't know *what* is wrong or what to do about it. We can leverage the `ng-invalid` class to reveal a helpful message. Here's the way it should look when the user deletes the name: figure.image-display img(src="/resources/images/devguide/forms/name-required-error.png" width="400px" alt="Name required") :marked To achieve this effect we extend the `` tag with 1. a [template reference variable](./template-syntax.html#ref-vars) 1. the "*is required*" message in a nearby `
` which we'll display only if the control is invalid. Here's how we do it for the *name* input box: +makeExample('forms/ts/app/hero-form.component.html', 'name-with-error-msg', 'app/hero-form.component.html (excerpt)')(format=".") :marked We need a template reference variable to access the input box's Angular control from within the template. Here we created a variable called `name` and gave it the value "ngModel". .l-sub-section :marked Why "ngModel"? A directive's [exportAs](../api/core/index/Directive-decorator.html) property tells Angular how to link the reference variable to the directive. We set `name` to `ngModel` because the `ngModel` directive's `exportAs` property happens to be "ngModel". Now we can control visibility of the "name" error message by binding properties of the `name` control to the message `
` element's `hidden` property. +makeExample('forms/ts/app/hero-form.component.html', 'hidden-error-msg', 'app/hero-form.component.html (excerpt)')(format='.') :marked In this example, we hide the message when the control is valid or pristine; pristine means the user hasn't changed the value since it was displayed in this form. This user experience is the developer's choice. Some folks want to see the message at all times. If we ignore the `pristine` state, we would hide the message only when the value is valid. If we arrive in this component with a new (blank) hero or an invalid hero, we'll see the error message immediately, before we've done anything. Some folks find that behavior disconcerting. They only want to see the message when the user makes an invalid change. Hiding the message while the control is "pristine" achieves that goal. We'll see the significance of this choice when we [add a new hero](#new-hero) to the form. The hero *Alter Ego* is optional so we can leave that be. Hero *Power* selection is required. We can add the same kind of error handling to the `